2023-01-13 02:51:32 -05:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
2022-07-09 05:49:20 -04:00
|
|
|
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_and_log: {
|
|
|
|
parameters: [],
|
|
|
|
result: "void",
|
|
|
|
},
|
|
|
|
} as const,
|
|
|
|
);
|
|
|
|
|
|
|
|
const tripleLogCallback = () => {
|
|
|
|
console.log("Sync");
|
|
|
|
Promise.resolve().then(() => {
|
|
|
|
console.log("Async");
|
|
|
|
callback.unref();
|
|
|
|
});
|
|
|
|
setTimeout(() => {
|
|
|
|
console.log("Timeout");
|
|
|
|
callback.unref();
|
|
|
|
}, 10);
|
|
|
|
};
|
|
|
|
|
|
|
|
const callback = new Deno.UnsafeCallback(
|
|
|
|
{
|
|
|
|
parameters: [],
|
|
|
|
result: "void",
|
|
|
|
} as const,
|
|
|
|
tripleLogCallback,
|
|
|
|
);
|
|
|
|
|
|
|
|
// Store function
|
|
|
|
dylib.symbols.store_function(callback.pointer);
|
|
|
|
|
|
|
|
// Synchronous callback logging
|
|
|
|
console.log("SYNCHRONOUS");
|
|
|
|
dylib.symbols.call_stored_function();
|
|
|
|
console.log("STORED_FUNCTION called");
|
|
|
|
|
|
|
|
// Wait to make sure synch logging and async logging
|
|
|
|
await new Promise((res) => setTimeout(res, 100));
|
|
|
|
|
|
|
|
// Ref twice to make sure both `Promise.resolve().then()` and `setTimeout()`
|
|
|
|
// must resolve before isolate exists.
|
|
|
|
callback.ref();
|
|
|
|
callback.ref();
|
|
|
|
|
|
|
|
console.log("THREAD SAFE");
|
|
|
|
dylib.symbols.call_stored_function_thread_safe_and_log();
|