mirror of
https://github.com/denoland/deno.git
synced 2025-01-18 11:53:59 -05:00
947ce41e99
Most uses of `Deno.resources()` within tests, as they previously checked for leaked resources. This is not needed as the test runner does this automatically. Other internal uses of this API have been replaced with the internal `Deno[Deno.internal].core.resources()`.
105 lines
2.7 KiB
JavaScript
105 lines
2.7 KiB
JavaScript
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
// deno-lint-ignore-file
|
|
|
|
const targetDir = Deno.execPath().replace(/[^\/\\]+$/, "");
|
|
const [libPrefix, libSuffix] = {
|
|
darwin: ["lib", "dylib"],
|
|
linux: ["lib", "so"],
|
|
windows: ["", "dll"],
|
|
}[Deno.build.os];
|
|
const libPath = `${targetDir}/${libPrefix}test_ffi.${libSuffix}`;
|
|
|
|
const dylib = Deno.dlopen(libPath, {
|
|
store_function: {
|
|
parameters: ["function"],
|
|
result: "void",
|
|
},
|
|
call_stored_function: {
|
|
parameters: [],
|
|
result: "void",
|
|
},
|
|
call_stored_function_thread_safe: {
|
|
parameters: [],
|
|
result: "void",
|
|
},
|
|
});
|
|
|
|
let resolveWorker;
|
|
let workerResponsePromise;
|
|
|
|
const worker = new Worker(
|
|
new URL("./thread_safe_test_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
|
|
worker.addEventListener("message", () => {
|
|
if (resolveWorker) {
|
|
resolveWorker();
|
|
}
|
|
});
|
|
|
|
const sendWorkerMessage = async (data) => {
|
|
workerResponsePromise = new Promise((res) => {
|
|
resolveWorker = res;
|
|
});
|
|
worker.postMessage(data);
|
|
await workerResponsePromise;
|
|
};
|
|
|
|
// Test step 1: Register main thread callback, trigger on worker thread
|
|
|
|
const mainThreadCallback = new Deno.UnsafeCallback(
|
|
{ parameters: [], result: "void" },
|
|
() => {
|
|
console.log("Callback on main thread");
|
|
},
|
|
);
|
|
|
|
mainThreadCallback.ref();
|
|
|
|
dylib.symbols.store_function(mainThreadCallback.pointer);
|
|
|
|
await sendWorkerMessage("call");
|
|
|
|
// Test step 2: Register on worker thread, trigger on main thread
|
|
|
|
await sendWorkerMessage("register");
|
|
|
|
dylib.symbols.call_stored_function();
|
|
|
|
// Unref both main and worker thread callbacks and terminate the worker: Note, the stored function pointer in lib is now dangling.
|
|
|
|
dylib.symbols.store_function(null);
|
|
|
|
mainThreadCallback.unref();
|
|
await sendWorkerMessage("unref");
|
|
worker.terminate();
|
|
|
|
// Test step 3: Register a callback that will be the only thing left keeping the isolate from exiting.
|
|
// Rely on it to keep Deno running until the callback comes in and unrefs the callback, after which Deno should exit.
|
|
|
|
const cleanupCallback = new Deno.UnsafeCallback(
|
|
{ parameters: [], result: "void" },
|
|
() => {
|
|
console.log("Callback being called");
|
|
// Defer the cleanup to give the spawned thread all the time it needs to properly shut down
|
|
setTimeout(() => cleanup(), 100);
|
|
},
|
|
);
|
|
|
|
cleanupCallback.ref();
|
|
|
|
function cleanup() {
|
|
cleanupCallback.unref();
|
|
dylib.symbols.store_function(null);
|
|
mainThreadCallback.close();
|
|
cleanupCallback.close();
|
|
console.log("Isolate should now exit");
|
|
}
|
|
|
|
dylib.symbols.store_function(cleanupCallback.pointer);
|
|
|
|
console.log(
|
|
"Calling callback, isolate should stay asleep until callback is called",
|
|
);
|
|
dylib.symbols.call_stored_function_thread_safe();
|