diff --git a/.github/workflows/rust.yml b/.github/workflows/rust.yml index 71f3d153..ff3d7b30 100644 --- a/.github/workflows/rust.yml +++ b/.github/workflows/rust.yml @@ -46,7 +46,7 @@ jobs: runs-on: ${{ matrix.os }} strategy: matrix: - os: [ "macos-15" ] + os: [ "macos-15", "ubuntu-latest" ] # Test on the pinned rust-toolchain version. toolchain: [ "1.80.0" ] steps: @@ -78,10 +78,10 @@ jobs: --release --verbose --workspace + --exclude compile-errors --exclude does-it-work --exclude test-json --exclude test-unique-id - --exclude compile-errors stable-test-no-op: name: Test with probes disabled diff --git a/tests/compile-errors/src/different-serializable-type.stderr b/tests/compile-errors/src/different-serializable-type.stderr index 5d7367ae..f6c6c241 100644 --- a/tests/compile-errors/src/different-serializable-type.stderr +++ b/tests/compile-errors/src/different-serializable-type.stderr @@ -1,21 +1,3 @@ -error[E0277]: the trait bound `Expected: Clone` is not satisfied - --> src/different-serializable-type.rs:31:20 - | -31 | fn my_probe(_: Expected) {} - | ^^^^^^^^ the trait `Clone` is not implemented for `Expected` - | -note: required by a bound in `usdt_types_must_be_clone_and_serialize` - --> src/different-serializable-type.rs:28:1 - | -28 | #[usdt::provider] - | ^^^^^^^^^^^^^^^^^ required by this bound in `usdt_types_must_be_clone_and_serialize` - = note: this error originates in the attribute macro `usdt::provider` (in Nightly builds, run with -Z macro-backtrace for more info) -help: consider annotating `Expected` with `#[derive(Clone)]` - | -19 + #[derive(Clone)] -20 | struct Expected { - | - error[E0277]: the trait bound `Different: Borrow` is not satisfied --> src/different-serializable-type.rs:28:1 | diff --git a/usdt-attr-macro/src/lib.rs b/usdt-attr-macro/src/lib.rs index 9c894120..21d3ee5c 100644 --- a/usdt-attr-macro/src/lib.rs +++ b/usdt-attr-macro/src/lib.rs @@ -1,6 +1,6 @@ //! Generate USDT probes from an attribute macro -// Copyright 2021 Oxide Computer Company +// Copyright 2024 Oxide Computer Company // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. @@ -144,7 +144,7 @@ fn generate_provider_item( quote! { const _: fn() = || { #(#use_statements)* - fn usdt_types_must_be_clone_and_serialize() {} + fn usdt_types_must_be_serialize() {} #(#check_fns)* }; } @@ -269,16 +269,12 @@ fn build_serializable_check_function(ident: &T, fn_index: usize, arg_index: u where T: quote::ToTokens, { - let fn_name = quote::format_ident!( - "usdt_types_must_be_clone_and_serialize_{}_{}", - fn_index, - arg_index - ); + let fn_name = quote::format_ident!("usdt_types_must_be_serialize_{}_{}", fn_index, arg_index); quote! { fn #fn_name() { // #ident must be in scope here, because this function is defined in the same module as // the actual probe functions, and thus shares any imports the consumer wants. - usdt_types_must_be_clone_and_serialize::<#ident>() + usdt_types_must_be_serialize::<#ident>() } } } diff --git a/usdt-impl/src/common.rs b/usdt-impl/src/common.rs index d95c106a..e56b8d47 100644 --- a/usdt-impl/src/common.rs +++ b/usdt-impl/src/common.rs @@ -1,6 +1,6 @@ //! Shared code used in both the linker and no-linker implementations of this crate. -// Copyright 2021 Oxide Computer Company +// Copyright 2024 Oxide Computer Company // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. @@ -14,39 +14,29 @@ // See the License for the specific language governing permissions and // limitations under the License. -use crate::{DataType, Provider}; +use crate::DataType; use proc_macro2::TokenStream; use quote::{format_ident, quote}; -// Construct function call that is used internally in the UDST-generated macros, to allow -// compile-time type checking of the lambda arguments. -pub fn generate_type_check( +/// Construct a function to type-check the argument closure. +/// +/// This constructs a function that is never called, but is used to ensure that +/// the closure provided to each probe macro returns arguments of the right +/// type. +pub fn construct_type_check( provider_name: &str, - use_statements: &[syn::ItemUse], probe_name: &str, + use_statements: &[syn::ItemUse], types: &[DataType], ) -> TokenStream { - // If the probe has zero arguments, verify that the result of calling the closure is `()` - // Note that there's no need to clone the closure here, since () is Copy. + // If there are zero arguments, we need to make sure we can assign the + // result of the closure to (). if types.is_empty() { return quote! { - let __usdt_private_args_lambda = $args_lambda; - let _ = || { - let _: () = __usdt_private_args_lambda(); - }; + let _: () = ($args_lambda)(); }; } - - // For one or more arguments, verify that we can unpack the closure into a tuple of type - // `(arg0, arg1, ...)`. We verify that we can pass those arguments to a type-check function - // that is _similar to_, but not exactly the probe function signature. In particular, we try to - // support passing things by value or reference, and take some form of reference to that thing. - // The mapping is generally: - // - // T or &T -> Borrow - // Strings -> AsRef - // [T; N] or &[T] -> AsRef<[T]> - let type_check_args = types + let type_check_params = types .iter() .map(|typ| match typ { DataType::Serializable(ty) => { @@ -84,28 +74,22 @@ pub fn generate_type_check( }) .collect::>(); - // Unpack the tuple from the closure to `args.0, args.1, ...`. - let expanded_lambda_args = (0..types.len()) + // Create a list of arguments `arg.0`, `arg.1`, ... to pass to the check + // function. + let type_check_args = (0..types.len()) .map(|i| { let index = syn::Index::from(i); quote! { args.#index } }) .collect::>(); - let preamble = unpack_argument_lambda(types, /* clone = */ true); - - let type_check_function = - format_ident!("__usdt_private_{}_{}_type_check", provider_name, probe_name); + let type_check_fn = format_ident!("__usdt_private_{}_{}_type_check", provider_name, probe_name); quote! { - let __usdt_private_args_lambda = $args_lambda; #[allow(unused_imports)] #(#use_statements)* #[allow(non_snake_case)] - fn #type_check_function (#(#type_check_args),*) { } - let _ = || { - #preamble - #type_check_function(#(#expanded_lambda_args),*); - }; + fn #type_check_fn(#(#type_check_params),*) {} + let _ = || { #type_check_fn(#(#type_check_args),*); }; } } @@ -156,28 +140,24 @@ pub fn construct_probe_args(types: &[DataType]) -> (TokenStream, TokenStream) { (destructured_arg, register_arg) }) .unzip(); - let preamble = unpack_argument_lambda(types, /* clone = */ false); + let arg_lambda = call_argument_closure(types); let unpacked_args = quote! { - #preamble + #arg_lambda #(#unpacked_args)* }; let in_regs = quote! { #(#in_regs,)* }; (unpacked_args, in_regs) } -fn unpack_argument_lambda(types: &[DataType], clone: bool) -> TokenStream { - let maybe_clone = if clone { - quote! { .clone() } - } else { - quote! {} - }; +/// Call the argument closure, assigning its output to `args`. +pub fn call_argument_closure(types: &[DataType]) -> TokenStream { match types.len() { - // Don't bother with arguments if there are none. - 0 => quote! { __usdt_private_args_lambda #maybe_clone (); }, + // Don't bother with any closure if there are no arguments. + 0 => quote! {}, // Wrap a single argument in a tuple. - 1 => quote! { let args = (__usdt_private_args_lambda #maybe_clone (),); }, + 1 => quote! { let args = (($args_lambda)(),); }, // General case. - _ => quote! { let args = __usdt_private_args_lambda #maybe_clone (); }, + _ => quote! { let args = ($args_lambda)(); }, } } @@ -218,17 +198,18 @@ fn asm_type_convert(typ: &DataType, input: TokenStream) -> (TokenStream, TokenSt } } +/// Create the top-level probe macro. +/// +/// This takes the implementation block constructed elsewhere, and builds out +/// the actual macro users call in their code to fire the probe. pub(crate) fn build_probe_macro( config: &crate::CompileProvidersConfig, - provider: &Provider, probe_name: &str, types: &[DataType], impl_block: TokenStream, ) -> TokenStream { let module = config.module_ident(); let macro_name = config.probe_ident(probe_name); - let type_check_block = - generate_type_check(&provider.name, &provider.use_statements, probe_name, types); let no_args_match = if types.is_empty() { quote! { () => { crate::#module::#macro_name!(|| ()) }; } } else { @@ -243,7 +224,6 @@ pub(crate) fn build_probe_macro( }; ($args_lambda:expr) => { { - #type_check_block #impl_block } }; @@ -263,20 +243,16 @@ mod tests { use dtrace_parser::Sign; #[test] - fn test_generate_type_check_empty() { - let types = &[]; + fn test_construct_type_check_empty() { let expected = quote! { - let __usdt_private_args_lambda = $args_lambda; - let _ = || { - let _: () = __usdt_private_args_lambda(); - }; + let _ : () = ($args_lambda)(); }; - let block = generate_type_check("", &[], "", types); + let block = construct_type_check("", "", &[], &[]); assert_eq!(block.to_string(), expected.to_string()); } #[test] - fn test_generate_type_check_native() { + fn test_construct_type_check_native() { let provider = "provider"; let probe = "probe"; let types = &[ @@ -290,7 +266,6 @@ mod tests { })), ]; let expected = quote! { - let __usdt_private_args_lambda = $args_lambda; #[allow(unused_imports)] #[allow(non_snake_case)] fn __usdt_private_provider_probe_type_check( @@ -298,72 +273,65 @@ mod tests { _: impl ::std::borrow::Borrow ) { } let _ = || { - let args = __usdt_private_args_lambda.clone()(); __usdt_private_provider_probe_type_check(args.0, args.1); }; }; - let block = generate_type_check(provider, &[], probe, types); + let block = construct_type_check(provider, probe, &[], types); assert_eq!(block.to_string(), expected.to_string()); } #[test] - fn test_generate_type_check_with_string() { + fn test_construct_type_check_with_string() { let provider = "provider"; let probe = "probe"; let types = &[DataType::Native(dtrace_parser::DataType::String)]; let use_statements = vec![]; let expected = quote! { - let __usdt_private_args_lambda = $args_lambda; #[allow(unused_imports)] #[allow(non_snake_case)] fn __usdt_private_provider_probe_type_check(_: impl AsRef) { } let _ = || { - let args = (__usdt_private_args_lambda.clone()(),); __usdt_private_provider_probe_type_check(args.0); }; }; - let block = generate_type_check(provider, &use_statements, probe, types); + let block = construct_type_check(provider, probe, &use_statements, types); assert_eq!(block.to_string(), expected.to_string()); } #[test] - fn test_generate_type_check_with_shared_slice() { + fn test_construct_type_check_with_shared_slice() { let provider = "provider"; let probe = "probe"; let types = &[DataType::Serializable(syn::parse_str("&[u8]").unwrap())]; let use_statements = vec![]; let expected = quote! { - let __usdt_private_args_lambda = $args_lambda; #[allow(unused_imports)] #[allow(non_snake_case)] fn __usdt_private_provider_probe_type_check(_: impl AsRef<[u8]>) { } let _ = || { - let args = (__usdt_private_args_lambda.clone()(),); __usdt_private_provider_probe_type_check(args.0); }; }; - let block = generate_type_check(provider, &use_statements, probe, types); + let block = construct_type_check(provider, probe, &use_statements, types); assert_eq!(block.to_string(), expected.to_string()); } #[test] - fn test_generate_type_check_with_custom_type() { + fn test_construct_type_check_with_custom_type() { let provider = "provider"; let probe = "probe"; let types = &[DataType::Serializable(syn::parse_str("MyType").unwrap())]; let use_statements = vec![syn::parse2(quote! { use my_module::MyType; }).unwrap()]; let expected = quote! { - let __usdt_private_args_lambda = $args_lambda; #[allow(unused_imports)] use my_module::MyType; #[allow(non_snake_case)] fn __usdt_private_provider_probe_type_check(_: impl ::std::borrow::Borrow) { } let _ = || { - let args = (__usdt_private_args_lambda.clone()(),); __usdt_private_provider_probe_type_check(args.0); }; }; - let block = generate_type_check(provider, &use_statements, probe, types); + let block = construct_type_check(provider, probe, &use_statements, types); assert_eq!(block.to_string(), expected.to_string()); } @@ -382,7 +350,7 @@ mod tests { let registers = ["x0", "x1"]; let (args, regs) = construct_probe_args(types); let expected = quote! { - let args = __usdt_private_args_lambda(); + let args = ($args_lambda)(); let arg_0 = (*<_ as ::std::borrow::Borrow<*const u8>>::borrow(&args.0) as usize); let arg_1 = [(args.1.as_ref() as &str).as_bytes(), &[0_u8]].concat(); }; diff --git a/usdt-impl/src/empty.rs b/usdt-impl/src/empty.rs index 9c3946df..1279c114 100644 --- a/usdt-impl/src/empty.rs +++ b/usdt-impl/src/empty.rs @@ -2,7 +2,7 @@ //! //! Used on platforms without DTrace. -// Copyright 2021 Oxide Computer Company +// Copyright 2024 Oxide Computer Company // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. @@ -76,8 +76,20 @@ fn compile_probe( probe: &Probe, config: &crate::CompileProvidersConfig, ) -> TokenStream { - let impl_block = quote! { let _ = || (__usdt_private_args_lambda()) ; }; - common::build_probe_macro(config, provider, &probe.name, &probe.types, impl_block) + // We don't need to add any actual probe emission code, but we do still want + // to perform type-checking of its arguments here. + let args = common::call_argument_closure(&probe.types); + let type_check_fn = common::construct_type_check( + &provider.name, + &probe.name, + &provider.use_statements, + &probe.types, + ); + let impl_block = quote! { + #args + #type_check_fn + }; + common::build_probe_macro(config, &probe.name, &probe.types, impl_block) } pub fn register_probes() -> Result<(), crate::Error> { diff --git a/usdt-impl/src/linker.rs b/usdt-impl/src/linker.rs index 668d7597..59295be5 100644 --- a/usdt-impl/src/linker.rs +++ b/usdt-impl/src/linker.rs @@ -51,7 +51,7 @@ //! of the `stability` and `typedefs` symbols could be anything--we just need //! a symbol name we can reference for the asm! macro that won't get garbled. -// Copyright 2022 Oxide Computer Company +// Copyright 2024 Oxide Computer Company // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. @@ -170,6 +170,8 @@ fn compile_probe( syn::parse2::(quote! { _: #ty }).unwrap() }); let (unpacked_args, in_regs) = common::construct_probe_args(types); + let type_check_fn = + common::construct_type_check(&provider.name, probe_name, &provider.use_statements, types); #[cfg(target_arch = "x86_64")] let call_instruction = quote! { "call {extern_probe_fn}" }; @@ -199,6 +201,7 @@ fn compile_probe( unsafe { if #is_enabled_fn() != 0 { #unpacked_args + #type_check_fn ::std::arch::asm!( ".reference {typedefs}", #call_instruction, @@ -213,7 +216,7 @@ fn compile_probe( } }; - common::build_probe_macro(config, provider, probe_name, types, impl_block) + common::build_probe_macro(config, probe_name, types, impl_block) } #[derive(Debug, Default, Clone)] diff --git a/usdt-impl/src/no-linker.rs b/usdt-impl/src/no-linker.rs index 8c7a00fd..313e8d4e 100644 --- a/usdt-impl/src/no-linker.rs +++ b/usdt-impl/src/no-linker.rs @@ -82,6 +82,12 @@ fn compile_probe( let (unpacked_args, in_regs) = common::construct_probe_args(&probe.types); let is_enabled_rec = emit_probe_record(&provider.name, &probe.name, None); let probe_rec = emit_probe_record(&provider.name, &probe.name, Some(&probe.types)); + let type_check_fn = common::construct_type_check( + &provider.name, + &probe.name, + &provider.use_statements, + &probe.types, + ); let impl_block = quote! { { @@ -91,24 +97,25 @@ fn compile_probe( "990: clr rax", #is_enabled_rec, out("rax") is_enabled, - options(nomem, nostack, preserves_flags) + options(nostack) ); } if is_enabled != 0 { #unpacked_args + #type_check_fn unsafe { ::std::arch::asm!( "990: nop", #probe_rec, #in_regs - options(nomem, nostack, preserves_flags) + options(nostack) ); } } } }; - common::build_probe_macro(config, provider, &probe.name, &probe.types, impl_block) + common::build_probe_macro(config, &probe.name, &probe.types, impl_block) } fn extract_probe_records_from_section() -> Result {