1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-21 21:50:00 -05:00

feat(unstable): Improve FFI types (#20215)

Few improvements to FFI types:
1. Export `PointerObject` for convenience. It's fairly commonly used in
library code and thus should be exported.
2. Fix various comments around `PointerValue` and `UnsafePointer` and
expand upon them to better reflect reality.
3. Instead of using a `Record<"value", type>[T]` for determining the
type of an FFI symbol parameter use direct `T extends "value" ? type :
never` comparison.

The last part enables smuggling extra information into the parameter and
return value string declarations at the type level. eg. Instead of just
`"u8"` the parameter can be `"u8" & { [brand]: T }` for some `T extends
number`. That `T` can then be extracted from the parameter to form the
TypeScript function's parameter or return value type. Essentially, this
enables type-safe FFI!

The foremost use-cases for this are enums and pointer safety. These are
implemented in the second commit which should enable, in a backwards
compatible way, for pointer parameters to declare what sort of pointer
they mean, functions to declare what the API definition of the native
function is, and for numbers to declare what Enum they stand for (if
any).
This commit is contained in:
Aapo Alasuutari 2023-08-21 11:06:26 +03:00 committed by GitHub
parent 576d0db372
commit af125c8e70
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 215 additions and 80 deletions

View file

@ -103,6 +103,31 @@ declare namespace Deno {
*/ */
type NativeStructType = { readonly struct: readonly NativeType[] }; type NativeStructType = { readonly struct: readonly NativeType[] };
/** @category FFI */
const brand: unique symbol;
/** @category FFI */
export type NativeU8Enum<T extends number> = "u8" & { [brand]: T };
/** @category FFI */
export type NativeI8Enum<T extends number> = "i8" & { [brand]: T };
/** @category FFI */
export type NativeU16Enum<T extends number> = "u16" & { [brand]: T };
/** @category FFI */
export type NativeI16Enum<T extends number> = "i16" & { [brand]: T };
/** @category FFI */
export type NativeU32Enum<T extends number> = "u32" & { [brand]: T };
/** @category FFI */
export type NativeI32Enum<T extends number> = "i32" & { [brand]: T };
/** @category FFI */
export type NativeTypedPointer<T extends PointerObject> = "pointer" & {
[brand]: T;
};
export type NativeTypedFunction<T extends UnsafeCallbackDefinition> =
& "function"
& {
[brand]: T;
};
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
* All supported types for interfacing with foreign functions. * All supported types for interfacing with foreign functions.
@ -124,21 +149,6 @@ declare namespace Deno {
*/ */
export type NativeResultType = NativeType | NativeVoidType; export type NativeResultType = NativeType | NativeVoidType;
/** **UNSTABLE**: New API, yet to be vetted.
*
* A utility type conversion for foreign symbol parameters and unsafe callback
* return types.
*
* @category FFI
*/
type ToNativeTypeMap =
& Record<NativeNumberType, number>
& Record<NativeBigIntType, number | bigint>
& Record<NativeBooleanType, boolean>
& Record<NativePointerType, PointerValue>
& Record<NativeFunctionType, PointerValue>
& Record<NativeBufferType, BufferSource | null>;
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
* Type conversion for foreign symbol parameters and unsafe callback return * Type conversion for foreign symbol parameters and unsafe callback return
@ -148,15 +158,22 @@ declare namespace Deno {
*/ */
type ToNativeType<T extends NativeType = NativeType> = T extends type ToNativeType<T extends NativeType = NativeType> = T extends
NativeStructType ? BufferSource NativeStructType ? BufferSource
: ToNativeTypeMap[Exclude<T, NativeStructType>]; : T extends NativeNumberType ? T extends NativeU8Enum<infer U> ? U
: T extends NativeI8Enum<infer U> ? U
/** **UNSTABLE**: New API, yet to be vetted. : T extends NativeU16Enum<infer U> ? U
* : T extends NativeI16Enum<infer U> ? U
* A utility type for conversion for unsafe callback return types. : T extends NativeU32Enum<infer U> ? U
* : T extends NativeI32Enum<infer U> ? U
* @category FFI : number
*/ : T extends NativeBigIntType ? number | bigint
type ToNativeResultTypeMap = ToNativeTypeMap & Record<NativeVoidType, void>; : T extends NativeBooleanType ? boolean
: T extends NativePointerType
? T extends NativeTypedPointer<infer U> ? U | null : PointerValue
: T extends NativeFunctionType
? T extends NativeTypedFunction<infer U> ? PointerValue<U> | null
: PointerValue
: T extends NativeBufferType ? BufferSource | null
: never;
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
@ -166,7 +183,23 @@ declare namespace Deno {
*/ */
type ToNativeResultType<T extends NativeResultType = NativeResultType> = type ToNativeResultType<T extends NativeResultType = NativeResultType> =
T extends NativeStructType ? BufferSource T extends NativeStructType ? BufferSource
: ToNativeResultTypeMap[Exclude<T, NativeStructType>]; : T extends NativeNumberType ? T extends NativeU8Enum<infer U> ? U
: T extends NativeI8Enum<infer U> ? U
: T extends NativeU16Enum<infer U> ? U
: T extends NativeI16Enum<infer U> ? U
: T extends NativeU32Enum<infer U> ? U
: T extends NativeI32Enum<infer U> ? U
: number
: T extends NativeBigIntType ? number | bigint
: T extends NativeBooleanType ? boolean
: T extends NativePointerType
? T extends NativeTypedPointer<infer U> ? U | null : PointerValue
: T extends NativeFunctionType
? T extends NativeTypedFunction<infer U> ? PointerObject<U> | null
: PointerValue
: T extends NativeBufferType ? BufferSource | null
: T extends NativeVoidType ? void
: never;
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
@ -184,21 +217,6 @@ declare namespace Deno {
} }
: never; : never;
/** **UNSTABLE**: New API, yet to be vetted.
*
* A utility type for conversion of foreign symbol return types and unsafe
* callback parameters.
*
* @category FFI
*/
type FromNativeTypeMap =
& Record<NativeNumberType, number>
& Record<NativeBigIntType, number | bigint>
& Record<NativeBooleanType, boolean>
& Record<NativePointerType, PointerValue>
& Record<NativeBufferType, PointerValue>
& Record<NativeFunctionType, PointerValue>;
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
* Type conversion for foreign symbol return types and unsafe callback * Type conversion for foreign symbol return types and unsafe callback
@ -208,17 +226,22 @@ declare namespace Deno {
*/ */
type FromNativeType<T extends NativeType = NativeType> = T extends type FromNativeType<T extends NativeType = NativeType> = T extends
NativeStructType ? Uint8Array NativeStructType ? Uint8Array
: FromNativeTypeMap[Exclude<T, NativeStructType>]; : T extends NativeNumberType ? T extends NativeU8Enum<infer U> ? U
: T extends NativeI8Enum<infer U> ? U
/** **UNSTABLE**: New API, yet to be vetted. : T extends NativeU16Enum<infer U> ? U
* : T extends NativeI16Enum<infer U> ? U
* A utility type for conversion for foreign symbol return types. : T extends NativeU32Enum<infer U> ? U
* : T extends NativeI32Enum<infer U> ? U
* @category FFI : number
*/ : T extends NativeBigIntType ? number | bigint
type FromNativeResultTypeMap = : T extends NativeBooleanType ? boolean
& FromNativeTypeMap : T extends NativePointerType
& Record<NativeVoidType, void>; ? T extends NativeTypedPointer<infer U> ? U | null : PointerValue
: T extends NativeBufferType ? PointerValue
: T extends NativeFunctionType
? T extends NativeTypedFunction<infer U> ? PointerObject<U> | null
: PointerValue
: never;
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
@ -228,7 +251,23 @@ declare namespace Deno {
*/ */
type FromNativeResultType<T extends NativeResultType = NativeResultType> = type FromNativeResultType<T extends NativeResultType = NativeResultType> =
T extends NativeStructType ? Uint8Array T extends NativeStructType ? Uint8Array
: FromNativeResultTypeMap[Exclude<T, NativeStructType>]; : T extends NativeNumberType ? T extends NativeU8Enum<infer U> ? U
: T extends NativeI8Enum<infer U> ? U
: T extends NativeU16Enum<infer U> ? U
: T extends NativeI16Enum<infer U> ? U
: T extends NativeU32Enum<infer U> ? U
: T extends NativeI32Enum<infer U> ? U
: number
: T extends NativeBigIntType ? number | bigint
: T extends NativeBooleanType ? boolean
: T extends NativePointerType
? T extends NativeTypedPointer<infer U> ? U | null : PointerValue
: T extends NativeBufferType ? PointerValue
: T extends NativeFunctionType
? T extends NativeTypedFunction<infer U> ? PointerObject<U> | null
: PointerValue
: T extends NativeVoidType ? void
: never;
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
@ -354,42 +393,52 @@ declare namespace Deno {
: StaticForeignSymbol<T[K]>; : StaticForeignSymbol<T[K]>;
}; };
/** @category FFI */
const brand: unique symbol;
/** @category FFI */
type PointerObject = { [brand]: unknown };
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
* Pointer type depends on the architecture and actual pointer value. * A non-null pointer, represented as an object
* at runtime. The object's prototype is `null`
* and cannot be changed. The object cannot be
* assigned to either and is thus entirely read-only.
* *
* On a 32 bit host system all pointer values are plain numbers. On a 64 bit * To interact with memory through a pointer use the
* host system pointer values are represented as numbers if the value is below * {@linkcode UnsafePointerView} class. To create a
* `Number.MAX_SAFE_INTEGER`, otherwise they are provided as bigints. * pointer from an address or the get the address of
* a pointer use the static methods of the
* {@linkcode UnsafePointer} class.
* *
* @category FFI * @category FFI
*/ */
export type PointerValue = null | PointerObject; export type PointerObject<T = unknown> = { [brand]: T };
/** **UNSTABLE**: New API, yet to be vetted. /** **UNSTABLE**: New API, yet to be vetted.
* *
* An unsafe pointer to a memory location for passing and returning pointers * Pointers are represented either with a {@linkcode PointerObject}
* to and from the FFI. * object or a `null` if the pointer is null.
*
* @category FFI
*/
export type PointerValue<T = unknown> = null | PointerObject<T>;
/** **UNSTABLE**: New API, yet to be vetted.
*
* A collection of static functions for interacting with pointer objects.
* *
* @category FFI * @category FFI
*/ */
export class UnsafePointer { export class UnsafePointer {
/** Create a pointer from a numeric value. This one is <i>really</i> dangerous! */ /** Create a pointer from a numeric value. This one is <i>really</i> dangerous! */
static create(value: number | bigint): PointerValue; static create<T = unknown>(value: number | bigint): PointerValue<T>;
/** Returns `true` if the two pointers point to the same address. */ /** Returns `true` if the two pointers point to the same address. */
static equals(a: PointerValue, b: PointerValue): boolean; static equals<T = unknown>(a: PointerValue<T>, b: PointerValue<T>): boolean;
/** Return the direct memory pointer to the typed array in memory. */ /** Return the direct memory pointer to the typed array in memory. */
static of(value: Deno.UnsafeCallback | BufferSource): PointerValue; static of<T = unknown>(
value: Deno.UnsafeCallback | BufferSource,
): PointerValue<T>;
/** Return a new pointer offset from the original by `offset` bytes. */ /** Return a new pointer offset from the original by `offset` bytes. */
static offset( static offset<T = unknown>(
value: NonNullable<PointerValue>, value: PointerObject,
offset: number, offset: number,
): PointerValue; ): PointerValue<T>;
/** Get the numeric value of a pointer */ /** Get the numeric value of a pointer */
static value(value: PointerValue): number | bigint; static value(value: PointerValue): number | bigint;
} }
@ -404,9 +453,9 @@ declare namespace Deno {
* @category FFI * @category FFI
*/ */
export class UnsafePointerView { export class UnsafePointerView {
constructor(pointer: NonNullable<PointerValue>); constructor(pointer: PointerObject);
pointer: NonNullable<PointerValue>; pointer: PointerObject;
/** Gets a boolean at the specified byte offset from the pointer. */ /** Gets a boolean at the specified byte offset from the pointer. */
getBool(offset?: number): boolean; getBool(offset?: number): boolean;
@ -441,14 +490,14 @@ declare namespace Deno {
* pointer. */ * pointer. */
getFloat64(offset?: number): number; getFloat64(offset?: number): number;
/** Gets a pointer at the specified byte offset from the pointer */ /** Gets a pointer at the specified byte offset from the pointer */
getPointer(offset?: number): PointerValue; getPointer<T = unknown>(offset?: number): PointerValue<T>;
/** Gets a C string (`null` terminated string) at the specified byte offset /** Gets a C string (`null` terminated string) at the specified byte offset
* from the pointer. */ * from the pointer. */
getCString(offset?: number): string; getCString(offset?: number): string;
/** Gets a C string (`null` terminated string) at the specified byte offset /** Gets a C string (`null` terminated string) at the specified byte offset
* from the specified pointer. */ * from the specified pointer. */
static getCString( static getCString(
pointer: NonNullable<PointerValue>, pointer: PointerObject,
offset?: number, offset?: number,
): string; ): string;
/** Gets an `ArrayBuffer` of length `byteLength` at the specified byte /** Gets an `ArrayBuffer` of length `byteLength` at the specified byte
@ -457,7 +506,7 @@ declare namespace Deno {
/** Gets an `ArrayBuffer` of length `byteLength` at the specified byte /** Gets an `ArrayBuffer` of length `byteLength` at the specified byte
* offset from the specified pointer. */ * offset from the specified pointer. */
static getArrayBuffer( static getArrayBuffer(
pointer: NonNullable<PointerValue>, pointer: PointerObject,
byteLength: number, byteLength: number,
offset?: number, offset?: number,
): ArrayBuffer; ): ArrayBuffer;
@ -473,7 +522,7 @@ declare namespace Deno {
* *
* Also takes optional byte offset from the pointer. */ * Also takes optional byte offset from the pointer. */
static copyInto( static copyInto(
pointer: NonNullable<PointerValue>, pointer: PointerObject,
destination: BufferSource, destination: BufferSource,
offset?: number, offset?: number,
): void; ): void;
@ -488,11 +537,13 @@ declare namespace Deno {
*/ */
export class UnsafeFnPointer<Fn extends ForeignFunction> { export class UnsafeFnPointer<Fn extends ForeignFunction> {
/** The pointer to the function. */ /** The pointer to the function. */
pointer: NonNullable<PointerValue>; pointer: PointerObject<Fn>;
/** The definition of the function. */ /** The definition of the function. */
definition: Fn; definition: Fn;
constructor(pointer: NonNullable<PointerValue>, definition: Const<Fn>); constructor(pointer: PointerObject<Fn>, definition: Const<Fn>);
/** @deprecated Properly type {@linkcode pointer} using {@linkcode NativeTypedFunction} or {@linkcode UnsafeCallbackDefinition} types. */
constructor(pointer: PointerObject, definition: Const<Fn>);
/** Call the foreign function. */ /** Call the foreign function. */
call: FromForeignFunction<Fn>; call: FromForeignFunction<Fn>;
@ -562,7 +613,7 @@ declare namespace Deno {
); );
/** The pointer to the unsafe callback. */ /** The pointer to the unsafe callback. */
readonly pointer: NonNullable<PointerValue>; readonly pointer: PointerObject<Definition>;
/** The definition of the unsafe callback. */ /** The definition of the unsafe callback. */
readonly definition: Definition; readonly definition: Definition;
/** The callback function. */ /** The callback function. */

View file

@ -173,7 +173,7 @@ result4.then((_0: Deno.BufferSource) => {});
result4.then((_1: null | Deno.UnsafePointer) => {}); result4.then((_1: null | Deno.UnsafePointer) => {});
const fnptr = new Deno.UnsafeFnPointer( const fnptr = new Deno.UnsafeFnPointer(
{} as NonNullable<Deno.PointerValue>, {} as Deno.PointerObject,
{ {
parameters: ["u32", "pointer"], parameters: ["u32", "pointer"],
result: "void", result: "void",
@ -358,6 +358,24 @@ type AssertNotEqual<
$ = [Equal<Expected, Got>] extends [true] ? never : Expected, $ = [Equal<Expected, Got>] extends [true] ? never : Expected,
> = never; > = never;
const enum FooEnum {
Foo,
Bar,
}
const foo = "u8" as Deno.NativeU8Enum<FooEnum>;
declare const brand: unique symbol;
class MyPointerClass {}
type MyPointer = Deno.PointerObject & { [brand]: MyPointerClass };
const myPointer = "pointer" as Deno.NativeTypedPointer<MyPointer>;
type MyFunctionDefinition = Deno.UnsafeCallbackDefinition<
[typeof foo, "u32"],
typeof myPointer
>;
const myFunction = "function" as Deno.NativeTypedFunction<
MyFunctionDefinition
>;
type __Tests__ = [ type __Tests__ = [
empty: AssertEqual< empty: AssertEqual<
{ symbols: Record<never, never>; close(): void }, { symbols: Record<never, never>; close(): void },
@ -442,4 +460,70 @@ type __Tests__ = [
{ foo: { parameters: []; result: "i32" } } { foo: { parameters: []; result: "i32" } }
> >
>, >,
enum_param: AssertEqual<
{
symbols: {
foo: (arg: FooEnum) => void;
};
close(): void;
},
Deno.DynamicLibrary<
{ foo: { parameters: [typeof foo]; result: "void" } }
>
>,
enum_return: AssertEqual<
{
symbols: {
foo: () => FooEnum;
};
close(): void;
},
Deno.DynamicLibrary<
{ foo: { parameters: []; result: typeof foo } }
>
>,
typed_pointer_param: AssertEqual<
{
symbols: {
foo: (arg: MyPointer | null) => void;
};
close(): void;
},
Deno.DynamicLibrary<
{ foo: { parameters: [typeof myPointer]; result: "void" } }
>
>,
typed_pointer_return: AssertEqual<
{
symbols: {
foo: () => MyPointer | null;
};
close(): void;
},
Deno.DynamicLibrary<
{ foo: { parameters: []; result: typeof myPointer } }
>
>,
typed_function_param: AssertEqual<
{
symbols: {
foo: (arg: Deno.PointerObject<MyFunctionDefinition> | null) => void;
};
close(): void;
},
Deno.DynamicLibrary<
{ foo: { parameters: [typeof myFunction]; result: "void" } }
>
>,
typed_function_return: AssertEqual<
{
symbols: {
foo: () => Deno.PointerObject<MyFunctionDefinition> | null;
};
close(): void;
},
Deno.DynamicLibrary<
{ foo: { parameters: []; result: typeof myFunction } }
>
>,
]; ];