diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index a94a688..3851859 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -12,7 +12,7 @@ jobs: - uses: denoland/setup-deno@v1 with: - deno-version: v1.13.1 + deno-version: ab2e0a465e4eafe4de2cc6ac7ef61d1655db4c2d - name: Install rust uses: hecrj/setup-rust-action@v1 @@ -30,6 +30,4 @@ jobs: run: | cd example deno run -A ../cli.ts - deno test --allow-ffi --unstable - - + deno test --no-check --allow-ffi --unstable diff --git a/README.md b/README.md index cf3bcb4..7ffb6fe 100644 --- a/README.md +++ b/README.md @@ -21,8 +21,14 @@ deno_bindgen = { git = "https://github.com/littledivy/deno_bindgen" } use deno_bindgen::deno_bindgen; #[deno_bindgen] -fn add(a: i32, b: i32) -> i32 { - a + b +pub struct Input { + a: i32, + b: i32, +} + +#[deno_bindgen] +fn add(input: Input) -> i32 { + input.a + input.b } ``` @@ -33,5 +39,6 @@ $ deno_bindgen ```typescript // add.ts import { add } from "./bindings/bindings.ts"; -add(1, 2); // 3 + +add({ a: 1, b: 2 }); // 3 ``` diff --git a/cli.ts b/cli.ts index 26a2883..1d7d93f 100644 --- a/cli.ts +++ b/cli.ts @@ -23,16 +23,26 @@ if (Deno.build.os == "windows") { let source = null; async function generate() { + let conf; try { - const conf = JSON.parse(await Deno.readTextFile("bindings.json")); - const pkgName = conf.name; - source = "// Auto-generated with deno_bindgen\n"; - - source += codegen(`target/${profile}/lib${pkgName}${ext}`, conf.bindings); + conf = JSON.parse(await Deno.readTextFile("bindings.json")); } catch (_) { // Nothing to update. return; } + + console.log(conf); + const pkgName = conf.name; + + source = "// Auto-generated with deno_bindgen\n"; + source += codegen( + `target/${profile}/lib${pkgName}${ext}`, + conf.type_defs, + conf.bindings, + { + le: conf.le, + } + ); await Deno.remove("bindings.json"); } @@ -40,6 +50,7 @@ await build(); await generate(); if (source != null) { + await ensureDir("bindings"); await Deno.writeTextFile("bindings/bindings.ts", source); } diff --git a/codegen.ts b/codegen.ts index 1b0675b..4b4cf02 100644 --- a/codegen.ts +++ b/codegen.ts @@ -16,257 +16,25 @@ const Type: Record = { f64: "number", }; -function invalidType(type: string) { - throw new TypeError(`Type not supported: ${type}`); -} - -const span = { start: 0, end: 0, ctxt: 0 }; - -function param(value: string, type: string) { - const kind = Type[type] || invalidType(type); - return { - type: "Parameter", - span, - decorators: [], - pat: { - type: "Identifier", - span, - value, - optional: false, - typeAnnotation: { - type: "TsTypeAnnotation", - span, - typeAnnotation: { - type: "TsKeywordType", - span, - kind, - }, - }, - }, - }; -} - -function bodyStmt(fn: Sig) { - return [ - { - type: "ReturnStatement", - span, - argument: { - type: "TsAsExpression", - span, - typeAnnotation: { - type: "TsKeywordType", - span, - kind: Type[fn.result] || invalidType(fn.result), - }, - expression: { - type: "CallExpression", - span, - callee: { - type: "MemberExpression", - span, - object: { - type: "MemberExpression", - span, - object: { - type: "Identifier", - span, - value: "_lib", - optional: false, - }, - property: { - type: "Identifier", - span, - value: "symbols", - optional: false, - }, - computed: false, - }, - property: { - type: "Identifier", - span, - value: fn.func, - optional: false, - }, - computed: false, - }, - arguments: fn.parameters.map((i) => { - return { - spread: null, - expression: { - type: "Identifier", - span, - value: i.ident, - }, - }; - }), - typeArguments: null, - }, - }, - }, - ]; -} +type TypeDef = { + fields: Record; + ident: string; +}; -function libDecl(target: string, signature: Sig[]) { - return { - type: "VariableDeclaration", - span, - kind: "const", - declare: false, - declarations: [ - { - type: "VariableDeclarator", - span, - id: { - type: "Identifier", - span, - value: "_lib", - optional: false, - typeAnnotation: null, - }, - init: { - type: "CallExpression", - span, - callee: { - type: "MemberExpression", - span, - object: { - type: "Identifier", - span, - value: "Deno", - optional: false, - }, - property: { - type: "Identifier", - span, - value: "dlopen", - optional: false, - }, - computed: false, - }, - arguments: [ - { - spread: null, - expression: { - type: "StringLiteral", - span, - value: target, - hasEscape: false, - kind: { type: "normal", containsQuote: true }, - }, - }, - { - spread: null, - expression: { - type: "ObjectExpression", - span, - properties: signature.map((sig) => { - return { - type: "KeyValueProperty", - key: { - type: "Identifier", - span, - value: sig.func, - optional: false, - }, - value: { - type: "ObjectExpression", - span, - properties: [ - { - type: "KeyValueProperty", - key: { - type: "Identifier", - span, - value: "result", - optional: false, - }, - value: { - type: "StringLiteral", - span, - value: sig.result, - hasEscape: false, - kind: { type: "normal", containsQuote: true }, - }, - }, - { - type: "KeyValueProperty", - key: { - type: "Identifier", - span, - value: "parameters", - optional: false, - }, - value: { - type: "ArrayExpression", - span, - elements: sig.parameters.map((p) => { - return { - spread: null, - expression: { - type: "StringLiteral", - span, - value: p.type, - hasEscape: false, - kind: { - type: "normal", - containsQuote: true, - }, - }, - }; - }), - }, - }, - ], - }, - }; - }), - }, - }, - ], - typeArguments: null, - }, - definite: false, - }, - ], - }; +function resolveType(typeDefs: TypeDef[], type: string): string { + if (Type[type] !== undefined) return Type[type]; + if (typeDefs.find((f) => f.ident == type) !== undefined) { + return type; + } + throw new TypeError(`Type not supported: ${type}`); } -function exportDecl(fn: Sig) { - return { - type: "ExportDeclaration", - span, - declaration: { - type: "FunctionDeclaration", - identifier: { - type: "Identifier", - span, - value: fn.func, - optional: false, - }, - declare: false, - params: fn.parameters.map((p) => param(p.ident, p.type)), - decorators: [], - span, - body: { - type: "BlockStatement", - span, - stmts: bodyStmt(fn), - }, - generator: false, - async: false, - typeParameters: null, - returnType: { - type: "TsTypeAnnotation", - span, - typeAnnotation: { - type: "TsKeywordType", - span, - kind: Type[fn.result] || invalidType(fn.result), - }, - }, - }, - }; +function resolveDlopenParameter(typeDefs: TypeDef[], type: string): string { + if (Type[type] !== undefined) return type; + if (typeDefs.find((f) => f.ident == type) !== undefined) { + return "buffer"; + } + throw new TypeError(`Type not supported: ${type}`); } type Sig = { @@ -275,16 +43,91 @@ type Sig = { result: string; }; -export function codegen(dylib: string, signature: Sig[]) { - const { code } = print({ - type: "Module", - span, - body: [ - libDecl(dylib, signature), - ...signature.map((e) => exportDecl(e)), - ], - interpreter: null, - }); +type Options = { + le?: boolean; +}; + +function createByteTypeImport(le?: boolean) { + // Endianess dependent types to be imported from the `byte_type` module. + let types = [ + "i16", + "u16", + "i32", + "u32", + "i64", + "u64", + "f32", + "f64", + ]; + + // Finalize type name based on endianness. + const typeImports = types.map((ty) => ty + (le ? "le" : "be")); + + // TODO(@littledivy): version imports + let code = `import { Struct, i8, u8, ${ + typeImports.join(", ") + } } from "https://deno.land/x/byte_type/mod.ts";\n`; + + code += types.map((ty, idx) => `const ${ty} = ${typeImports[idx]};`).join('\n'); + + code += `\nconst usize = u64;\n`; + code += `const isize = i64;\n`; return code; } + +export function codegen( + target: string, + decl: TypeDef[], + signature: Sig[], + options?: Options, +) { + return `${createByteTypeImport(options?.le)} +const _lib = Deno.dlopen("${target}", { ${ + signature.map((sig) => + `${sig.func}: { parameters: [ ${ + sig.parameters.map((p) => `"${resolveDlopenParameter(decl, p.type)}"`) + .join(", ") + } ], result: "${sig.result}" }` + ).join(", ") + } }); +${ + decl.map((def) => + `type ${def.ident} = { ${ + Object.keys(def.fields).map((f) => + `${f}: ${resolveType(decl, def.fields[f])}` + ).join("; ") + } };` + ).join("\n") + } +${ + decl.map((def) => + `const _${def.ident} = new Struct({ ${ + Object.keys(def.fields).map((f) => `${f}: ${def.fields[f]}`).join(", ") + } });` + ).join("\n") + } +${ + signature.map((sig) => + `export function ${sig.func}(${ + sig.parameters.map((p) => `${p.ident}: ${resolveType(decl, p.type)}`) + .join(", ") + }) { + ${ + sig.parameters.filter((p) => Type[p.type] == undefined).map((p) => + `const _buf_${p.ident} = new Uint8Array(_${p.type}.size); + const _view_${p.ident} = new DataView(_buf_${p.ident}.buffer); + _${p.type}.write(_view_${p.ident}, 0, ${p.ident});` + ).join("\n") + } + const _result = _lib.symbols.${sig.func}(${ + sig.parameters.map((p) => + Type[p.type] == undefined ? ` _buf_${p.ident}` : p.ident + ).join(", ") + }); + return _result as ${resolveType(decl, sig.result)}; +}` + ).join("\n") + } + `; +} diff --git a/example/add_test.ts b/example/add_test.ts index 0940985..16da8d1 100644 --- a/example/add_test.ts +++ b/example/add_test.ts @@ -1,4 +1,4 @@ -import { add } from "./bindings/bindings.ts"; +import { add, add2 } from "./bindings/bindings.ts"; import { assertEquals } from "https://deno.land/std/testing/asserts.ts"; Deno.test({ @@ -8,3 +8,10 @@ Deno.test({ assertEquals(add(-1, 1), 0); }, }); + +Deno.test({ + name: "add2#test", + fn: () => { + assertEquals(add2({ a: 1, b: 2 }), 3); + }, +}); diff --git a/example/bindings/bindings.ts b/example/bindings/bindings.ts index daa77cc..b37d954 100644 --- a/example/bindings/bindings.ts +++ b/example/bindings/bindings.ts @@ -1,13 +1,42 @@ // Auto-generated with deno_bindgen +import { + f32le, + f64le, + i16le, + i32le, + i64le, + i8, + Struct, + u16le, + u32le, + u64le, + u8, +} from "https://deno.land/x/byte_type/mod.ts"; +const i16 = i16le; +const u16 = u16le; +const i32 = i32le; +const u32 = u32le; +const i64 = i64le; +const u64 = u64le; +const f32 = f32le; +const f64 = f64le; +const usize = u64; +const isize = i64; + const _lib = Deno.dlopen("target/debug/libadd.so", { - add: { - result: "i32", - parameters: [ - "i32", - "i32", - ], - }, + add: { parameters: ["i32", "i32"], result: "i32" }, + add2: { parameters: ["buffer"], result: "i32" }, }); -export function add(a0: number, a1: number): number { - return _lib.symbols.add(a0, a1) as number; +type Input = { a: number; b: number }; +const _Input = new Struct({ a: i32, b: i32 }); +export function add(a0: number, a1: number) { + const _result = _lib.symbols.add(a0, a1); + return _result as number; +} +export function add2(a0: Input) { + const _buf_a0 = new Uint8Array(_Input.size); + const _view_a0 = new DataView(_buf_a0.buffer); + _Input.write(_view_a0, 0, a0); + const _result = _lib.symbols.add2(_buf_a0); + return _result as number; } diff --git a/example/src/lib.rs b/example/src/lib.rs index 40c7969..8e1441f 100644 --- a/example/src/lib.rs +++ b/example/src/lib.rs @@ -2,5 +2,16 @@ use deno_bindgen::deno_bindgen; #[deno_bindgen] fn add(a: i32, b: i32) -> i32 { - a + b + a + b +} + +#[deno_bindgen] +pub struct Input { + a: i32, + b: i32, +} + +#[deno_bindgen] +fn add2(input: Input) -> i32 { + input.a + input.b } diff --git a/src/lib.rs b/src/lib.rs index dd7cb8d..3237128 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -8,17 +8,31 @@ use serde_json::json; use std::fs::OpenOptions; use std::io::Read; use std::io::Write; +use syn::Data; +use syn::DataStruct; +use syn::Fields; +use syn::ItemFn; + +fn is_le() -> bool { + #[cfg(target_endian = "little")] + return true; + + #[cfg(target_endian = "big")] + return false; +} #[derive(Serialize, Deserialize, Default)] struct Bindings { name: String, + le: bool, bindings: Vec, + type_defs: Vec, } #[proc_macro_attribute] pub fn deno_bindgen(_attr: TokenStream, input: TokenStream) -> TokenStream { - let func = syn::parse_macro_input!(input as syn::ItemFn); let mut buf = String::new(); + // Load existing bindings match OpenOptions::new().read(true).open("bindings.json") { Ok(mut fd) => { @@ -28,7 +42,9 @@ pub fn deno_bindgen(_attr: TokenStream, input: TokenStream) -> TokenStream { // We assume this was the first macro run. } } + let mut bindings: Bindings = serde_json::from_str(&buf).unwrap_or_default(); + bindings.le = is_le(); // TODO(@littledivy): Use Cargo's `out` directory // let dir = Path::new(env!("PROC_ARTIFACT_DIR")); let mut config = OpenOptions::new() @@ -37,48 +53,150 @@ pub fn deno_bindgen(_attr: TokenStream, input: TokenStream) -> TokenStream { .open("bindings.json") .unwrap(); - let mut parameters = vec![]; let pkg_name = std::env::var("CARGO_CRATE_NAME").unwrap(); - for (idx, i) in func.sig.inputs.iter().enumerate() { - match i { - syn::FnArg::Typed(ref val) => match &*val.ty { - syn::Type::Path(ref ty) => { - for seg in &ty.path.segments { - let ident = format!("a{}", idx); - parameters.push(json!({ - "ident": ident, - "type": seg.ident.to_string(), - })); + + match syn::parse::(input.clone()) { + // + Ok(func) => { + let mut parameters = vec![]; + let mut foriegn_types = vec![]; + + let fn_name = &func.sig.ident; + let fn_inputs = &func.sig.inputs; + let mut inputs = vec![]; + let fn_output = &func.sig.output; + let fn_block = &func.block; + let fn_params: Vec<_> = fn_inputs + .iter() + .enumerate() + .map(|(idx, i)| match i { + syn::FnArg::Typed(ref val) => { + let mut val = val.clone(); + match *val.ty { + syn::Type::Path(ref ty) => { + for seg in ty.path.segments.clone() { + let ident = format!("a{}", idx); + let ident_str = seg.ident.to_string(); + let ty = match ident_str.as_str() { + "void" | "i8" | "u8" | "i16" | "u16" | "i32" | "u32" + | "i64" | "u64" | "usize" | "isize" | "f32" | "f64" => { + ident_str + } + _ => { + // Check if a type definition already exists + bindings + .type_defs + .iter() + .find(|&def| def["ident"] == ident_str) + .expect(&format!( + "Type definition not found for `{}` identifier", + &ident_str + )); + foriegn_types.push(( + val.pat.clone(), + quote::format_ident!("{}", ident_str), + )); + val.ty = Box::new(syn::Type::Ptr( + syn::parse_quote! { *const u8 }, + )); + ident_str + } + }; + parameters.push(json!( + { + "ident": ident, + "type": ty, + } + )); + } + } + _ => {} + }; + + inputs.push(val); } + _ => unimplemented!(), + }) + .collect(); + + let return_type = match &func.sig.output { + syn::ReturnType::Default => "void".to_string(), + syn::ReturnType::Type(_, box syn::Type::Path(ty)) => { + // TODO(@littledivy): Support multiple `Type` path segments + ty.path.segments[0].ident.to_string() } - _ => {} - }, - _ => unreachable!(), - } - } + _ => unimplemented!(), + }; - let return_type = match &func.sig.output { - syn::ReturnType::Default => "void".to_string(), - syn::ReturnType::Type(_, box syn::Type::Path(ty)) => { - // TODO(@littledivy): Support multiple `Type` path segments - ty.path.segments[0].ident.to_string() - } - _ => panic!("Type not supported"), - }; + bindings.bindings.push(json!( + { + "func": func.sig.ident.to_string(), + "parameters": parameters, + "result": return_type, + } + )); + + bindings.name = pkg_name.to_string(); + config + .write_all(&serde_json::to_vec(&bindings).unwrap()) + .unwrap(); - bindings.bindings.push(json!({ - "func": func.sig.ident.to_string(), - "parameters": parameters, - "result": return_type, + let overrides = foriegn_types.iter().map(|(ident, ty)| quote! { + let _size = std::mem::size_of::<#ty>(); + let buf = unsafe { std::slice::from_raw_parts(#ident, _size) }; + + let #ident: #ty = unsafe { std::ptr::read(buf.as_ptr() as *const _) }; + }) + .fold(quote! {}, |acc, new| quote! { #acc #new }); + let input_idents: Vec<_> = inputs.iter().map(|i| &i.pat).collect(); + TokenStream::from(quote! { + #[no_mangle] + pub extern "C" fn #fn_name (#(#inputs,) *) #fn_output { + fn __inner_impl (#fn_inputs) #fn_output #fn_block + #overrides + let result = __inner_impl(#(#input_idents,) *); + result + } + }) } - )); - bindings.name = pkg_name.to_string(); - config - .write_all(&serde_json::to_vec(&bindings).unwrap()) - .unwrap(); + Err(_) => { + // Try to parse as an DeriveInput + let input = syn::parse_macro_input!(input as syn::DeriveInput); + let fields = match &input.data { + Data::Struct(DataStruct { + fields: Fields::Named(fields), + .. + }) => &fields.named, + _ => panic!("expected a struct with named fields"), + }; + let struct_name = &input.ident; + let mut definition = json!({}); - TokenStream::from(quote! { - #[no_mangle] - pub extern "C" #func - }) + for field in fields.iter() { + if let Some(ident) = &field.ident { + match field.ty { + syn::Type::Path(ref ty) => { + for seg in &ty.path.segments { + definition[ident.to_string()] = + serde_json::Value::String(seg.ident.to_string()); + } + } + _ => {} + } + } + } + + bindings.type_defs.push( + json!({ "ident": struct_name.to_string(), "fields": definition }), + ); + config + .write_all(&serde_json::to_vec(&bindings).unwrap()) + .unwrap(); + TokenStream::from(quote! { + // Preserve the input + #[repr(C)] + #input + }) + } + } }