From de5ed69537ebaaf545c5c9bb253f1df2c03111a5 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Tue, 4 Oct 2022 05:12:18 +0200 Subject: [PATCH 1/5] Reduce indentation in generated macro output Useful when debugging with cargo-expand --- objc2/src/macros/extern_class.rs | 48 +++++++++++++++++++----------- objc2/src/macros/extern_methods.rs | 19 ++++++------ 2 files changed, 39 insertions(+), 28 deletions(-) diff --git a/objc2/src/macros/extern_class.rs b/objc2/src/macros/extern_class.rs index 68d43a519..e0a7fabff 100644 --- a/objc2/src/macros/extern_class.rs +++ b/objc2/src/macros/extern_class.rs @@ -449,17 +449,17 @@ macro_rules! __attribute_helper { $($rest:tt)* ) $($macro_args:tt)* - } => {{ + } => { $crate::__attribute_helper! { @extract_sel_duplicate - $($rest)* + ($($rest)*) + $out_macro!( + $($macro_args)* + // Append selector to the end of the macro arguments + @($($sel)*) + ) } - - $out_macro!( - $($macro_args)* - @($($sel)*) - ) - }}; + }; { @extract_sel ($out_macro:path) @@ -468,14 +468,14 @@ macro_rules! __attribute_helper { $($rest:tt)* ) $($macro_args:tt)* - } => {{ + } => { $crate::__attribute_helper! { @extract_sel ($out_macro) ($($rest)*) $($macro_args)* } - }}; + }; { @extract_sel ($out_macro:path) @@ -487,21 +487,33 @@ macro_rules! __attribute_helper { { @extract_sel_duplicate - #[sel($($_sel_args:tt)*)] - $($rest:tt)* + ( + #[sel($($_sel_args:tt)*)] + $($rest:tt)* + ) + $($output:tt)* } => {{ compile_error!("Cannot not specify a selector twice!"); }}; { @extract_sel_duplicate - #[$($m_checked:tt)*] - $($rest:tt)* - } => {{ + ( + #[$($m_checked:tt)*] + $($rest:tt)* + ) + $($output:tt)* + } => { $crate::__attribute_helper! { @extract_sel_duplicate - $($rest)* + ($($rest:tt)*) + $($output)* } + }; + { + @extract_sel_duplicate + () + $($output:tt)* + } => {{ + $($output)* }}; - {@extract_sel_duplicate} => {}; - } diff --git a/objc2/src/macros/extern_methods.rs b/objc2/src/macros/extern_methods.rs index b0e25559b..45c32050f 100644 --- a/objc2/src/macros/extern_methods.rs +++ b/objc2/src/macros/extern_methods.rs @@ -319,12 +319,12 @@ macro_rules! __inner_extern_methods { @($($args_rest:tt)*) @($($sel:tt)*) } => { - $crate::__collect_msg_send!( + $crate::__collect_msg_send! { $crate::msg_send; $self; ($($sel)*); ($($args_rest)*); - ) + } }; { @unsafe_method_body @@ -336,12 +336,12 @@ macro_rules! __inner_extern_methods { @($($args_rest:tt)*) @($($sel:tt)*) } => { - $crate::__collect_msg_send!( + $crate::__collect_msg_send! { $crate::msg_send; Self::class(); ($($sel)*); ($($args_rest)*); - ) + } }; } @@ -377,16 +377,15 @@ macro_rules! __collect_msg_send { ($sel:ident : $($sel_rest:tt)*); ($arg:ident: $arg_ty:ty $(, $($args_rest:tt)*)?); $($output:tt)* - ) => {{ - $crate::__collect_msg_send!( + ) => { + $crate::__collect_msg_send! { $macro; $obj; ($($sel_rest)*); ($($($args_rest)*)?); - $($output)* - $sel: $arg, - ) - }}; + $($output)* $sel: $arg, + } + }; // If couldn't zip selector and arguments, show useful error message ($($_any:tt)*) => {{ From 155abd32f190d16da82a7c5086e896d30a23e5d5 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Tue, 4 Oct 2022 07:24:04 +0200 Subject: [PATCH 2/5] Simplify internal msg_send_id output --- objc2/src/__macro_helpers.rs | 68 ++++++++++-------- objc2/src/macros.rs | 69 ++++++++++++------- .../test_msg_send_id/expected/apple-aarch64.s | 14 ++-- .../test_msg_send_id/expected/apple-armv7.s | 14 ++-- .../test_msg_send_id/expected/apple-armv7s.s | 14 ++-- .../test_msg_send_id/expected/apple-x86.s | 14 ++-- .../test_msg_send_id/expected/apple-x86_64.s | 14 ++-- .../test_msg_send_id/expected/gnustep-x86.s | 14 ++-- .../expected/gnustep-x86_64.s | 14 ++-- test-assembly/crates/test_msg_send_id/lib.rs | 34 +++++---- .../expected/apple-aarch64.s | 2 +- .../expected/apple-armv7.s | 2 +- .../expected/apple-armv7s.s | 2 +- .../expected/apple-old-x86.s | 2 +- .../expected/apple-x86.s | 2 +- .../expected/apple-x86_64.s | 2 +- .../ui/msg_send_id_invalid_receiver.stderr | 4 +- test-ui/ui/msg_send_id_invalid_return.stderr | 6 +- 18 files changed, 160 insertions(+), 131 deletions(-) diff --git a/objc2/src/__macro_helpers.rs b/objc2/src/__macro_helpers.rs index 675067052..3de840cb4 100644 --- a/objc2/src/__macro_helpers.rs +++ b/objc2/src/__macro_helpers.rs @@ -3,7 +3,6 @@ use alloc::vec::Vec; #[cfg(feature = "verify_message")] use std::collections::HashSet; -use crate::__sel_inner; use crate::declare::ClassBuilder; #[cfg(feature = "verify_message")] use crate::declare::MethodImplementation; @@ -12,6 +11,7 @@ use crate::rc::{Allocated, Id, Ownership}; use crate::runtime::MethodDescription; use crate::runtime::{Class, Object, Protocol, Sel}; use crate::{Message, MessageArguments, MessageReceiver}; +use crate::{__sel_data, __sel_inner}; pub use crate::cache::CachedClass; pub use crate::cache::CachedSel; @@ -38,19 +38,16 @@ pub use std::sync::Once; // actual assembly is as one would expect. #[inline] -pub fn alloc() -> Sel { - // SAFETY: Must have NUL byte - __sel_inner!("alloc\0", "alloc") +pub fn alloc_sel() -> Sel { + __sel_inner!(__sel_data!(alloc), "alloc") } #[inline] -pub fn init() -> Sel { - // SAFETY: Must have NUL byte - __sel_inner!("init\0", "init") +pub fn init_sel() -> Sel { + __sel_inner!(__sel_data!(init), "init") } #[inline] -pub fn new() -> Sel { - // SAFETY: Must have NUL byte - __sel_inner!("new\0", "new") +pub fn new_sel() -> Sel { + __sel_inner!(__sel_data!(new), "new") } /// Helper for specifying the retain semantics for a given selector family. @@ -72,22 +69,33 @@ pub fn new() -> Sel { /// means it can't be a class method! /// /// -pub struct RetainSemantics< - // `new` family - const NEW: bool, - // `alloc` family - const ALLOC: bool, - // `init` family - const INIT: bool, - // `copy` or `mutableCopy` family - const COPY_OR_MUT_COPY: bool, -> {} - -type New = RetainSemantics; -type Alloc = RetainSemantics; -type Init = RetainSemantics; -type CopyOrMutCopy = RetainSemantics; -type Other = RetainSemantics; +// TODO: Use an enum instead of u8 here when stable +pub struct RetainSemantics {} + +pub type New = RetainSemantics<1>; +pub type Alloc = RetainSemantics<2>; +pub type Init = RetainSemantics<3>; +pub type CopyOrMutCopy = RetainSemantics<4>; +pub type Other = RetainSemantics<5>; + +pub const fn retain_semantics(selector: &str) -> u8 { + let selector = selector.as_bytes(); + match ( + in_selector_family(selector, b"new"), + in_selector_family(selector, b"alloc"), + in_selector_family(selector, b"init"), + in_selector_family(selector, b"copy"), + in_selector_family(selector, b"mutableCopy"), + ) { + (true, false, false, false, false) => 1, + (false, true, false, false, false) => 2, + (false, false, true, false, false) => 3, + (false, false, false, true, false) => 4, + (false, false, false, false, true) => 4, + (false, false, false, false, false) => 5, + _ => unreachable!(), + } +} pub trait MsgSendId { unsafe fn send_message_id>( @@ -266,7 +274,7 @@ impl<'a> MsgSendIdFailed<'a> for New { if let Some(obj) = obj { let cls = obj.class(); if cls.is_metaclass() { - if sel == new() { + if sel == new_sel() { panic!("failed creating new instance of {:?}", cls) } else { panic!("failed creating new instance using +[{:?} {:?}]", cls, sel) @@ -286,7 +294,7 @@ impl<'a> MsgSendIdFailed<'a> for Alloc { #[cold] #[track_caller] fn failed((cls, sel): Self::Args) -> ! { - if sel == alloc() { + if sel == alloc_sel() { panic!("failed allocating {:?}", cls) } else { panic!("failed allocating with +[{:?} {:?}]", cls, sel) @@ -305,7 +313,7 @@ impl MsgSendIdFailed<'_> for Init { } else { // We can't really display a more descriptive message here since the // object is consumed by `init` and may not be valid any more. - if sel == init() { + if sel == init_sel() { panic!("failed initializing object") } else { panic!("failed initializing object with -{:?}", sel) @@ -347,7 +355,7 @@ impl<'a> MsgSendIdFailed<'a> for Other { /// Checks whether a given selector is said to be in a given selector family. /// /// -pub const fn in_selector_family(mut selector: &[u8], mut family: &[u8]) -> bool { +const fn in_selector_family(mut selector: &[u8], mut family: &[u8]) -> bool { // Skip leading underscores from selector loop { selector = match selector { diff --git a/objc2/src/macros.rs b/objc2/src/macros.rs index 5428da0e4..368efb0a8 100644 --- a/objc2/src/macros.rs +++ b/objc2/src/macros.rs @@ -185,21 +185,34 @@ macro_rules! __class_inner { #[macro_export] macro_rules! sel { (alloc) => ({ - $crate::__macro_helpers::alloc() + $crate::__macro_helpers::alloc_sel() }); (init) => ({ - $crate::__macro_helpers::init() + $crate::__macro_helpers::init_sel() }); (new) => ({ - $crate::__macro_helpers::new() + $crate::__macro_helpers::new_sel() }); ($first:ident $(: $($rest:ident :)*)?) => ({ - use $crate::__macro_helpers::{concat, stringify, str}; - const SELECTOR_DATA: &str = concat!(stringify!($first), $(':', $(stringify!($rest), ':',)*)? '\0'); - $crate::__sel_inner!(SELECTOR_DATA, $crate::__hash_idents!($first $($($rest)*)?)) + $crate::__sel_inner!( + $crate::__sel_data!($first $(: $($rest :)*)?), + $crate::__hash_idents!($first $($($rest)*)?) + ) }); } +#[doc(hidden)] +#[macro_export] +macro_rules! __sel_data { + ($first:ident $(: $($rest:ident :)*)?) => { + $crate::__macro_helpers::concat!( + $crate::__macro_helpers::stringify!($first), + $(':', $($crate::__macro_helpers::stringify!($rest), ':',)*)? + '\0', + ) + }; +} + #[doc(hidden)] #[macro_export] #[cfg(not(feature = "unstable-static-sel"))] @@ -961,22 +974,42 @@ macro_rules! msg_send_bool { /// ``` #[macro_export] macro_rules! msg_send_id { + [$obj:expr, new $(,)?] => ({ + let sel = $crate::sel!(new); + let result; + result = <$crate::__macro_helpers::New as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ()); + result + }); + [$obj:expr, alloc $(,)?] => ({ + let sel = $crate::sel!(alloc); + let result; + result = <$crate::__macro_helpers::Alloc as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ()); + result + }); + [$obj:expr, init $(,)?] => ({ + let sel = $crate::sel!(init); + let result; + result = <$crate::__macro_helpers::Init as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ()); + result + }); [$obj:expr, $selector:ident $(,)?] => ({ $crate::__msg_send_id_helper!(@verify $selector); let sel = $crate::sel!($selector); - const NAME: &[$crate::__macro_helpers::u8] = $crate::__macro_helpers::stringify!($selector).as_bytes(); - $crate::__msg_send_id_helper!(@get_assert_consts NAME); + const NAME: &$crate::__macro_helpers::str = $crate::__macro_helpers::stringify!($selector); let result; - result = >::send_message_id($obj, sel, ()); + result = <$crate::__macro_helpers::RetainSemantics<{ + $crate::__macro_helpers::retain_semantics(NAME) + }> as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ()); result }); [$obj:expr, $($selector:ident : $argument:expr),+ $(,)?] => ({ let sel = $crate::sel!($($selector:)+); - const NAME: &[$crate::__macro_helpers::u8] = - $crate::__macro_helpers::concat!($($crate::__macro_helpers::stringify!($selector), ':'),+).as_bytes(); - $crate::__msg_send_id_helper!(@get_assert_consts NAME); + const NAME: &$crate::__macro_helpers::str = + $crate::__macro_helpers::concat!($($crate::__macro_helpers::stringify!($selector), ':'),+); let result; - result = >::send_message_id($obj, sel, ($($argument,)+)); + result = <$crate::__macro_helpers::RetainSemantics<{ + $crate::__macro_helpers::retain_semantics(NAME) + }> as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ($($argument,)+)); result }); } @@ -1001,14 +1034,4 @@ macro_rules! __msg_send_id_helper { ) }}; (@verify $selector:ident) => {{}}; - (@get_assert_consts $selector:ident) => { - use $crate::__macro_helpers::{bool, in_selector_family, RetainSemantics}; - const NEW: bool = in_selector_family($selector, b"new"); - const ALLOC: bool = in_selector_family($selector, b"alloc"); - const INIT: bool = in_selector_family($selector, b"init"); - const COPY_OR_MUT_COPY: bool = { - in_selector_family($selector, b"copy") || in_selector_family($selector, b"mutableCopy") - }; - type RS = RetainSemantics; - }; } diff --git a/test-assembly/crates/test_msg_send_id/expected/apple-aarch64.s b/test-assembly/crates/test_msg_send_id/expected/apple-aarch64.s index dae34f462..c022fc6cc 100644 --- a/test-assembly/crates/test_msg_send_id/expected/apple-aarch64.s +++ b/test-assembly/crates/test_msg_send_id/expected/apple-aarch64.s @@ -24,7 +24,7 @@ Lloh1: add x2, x2, l_anon.[ID].1@PAGEOFF mov x0, x20 mov x1, x19 - bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .loh AdrpAdd Lloh0, Lloh1 .globl _handle_alloc @@ -52,7 +52,7 @@ Lloh3: add x2, x2, l_anon.[ID].2@PAGEOFF mov x0, x20 mov x1, x19 - bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 1) + bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .loh AdrpAdd Lloh2, Lloh3 .globl _handle_init @@ -80,7 +80,7 @@ Lloh5: add x2, x2, l_anon.[ID].3@PAGEOFF mov x0, x20 mov x1, x19 - bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 2) + bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .loh AdrpAdd Lloh4, Lloh5 .globl _handle_alloc_init @@ -138,7 +138,7 @@ Lloh6: adrp x0, l_anon.[ID].4@PAGE Lloh7: add x0, x0, l_anon.[ID].4@PAGEOFF - bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 3) + bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .loh AdrpAdd Lloh6, Lloh7 .globl _handle_autoreleased @@ -177,7 +177,7 @@ Lloh9: add x2, x2, l_anon.[ID].5@PAGEOFF mov x0, x20 mov x1, x19 - bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 4) + bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .loh AdrpAdd Lloh8, Lloh9 .section __TEXT,__const @@ -203,11 +203,11 @@ l_anon.[ID].3: .p2align 3 l_anon.[ID].4: .quad l_anon.[ID].0 - .asciz ",\000\000\000\000\000\000\000@\000\000\000\005\000\000" + .asciz ",\000\000\000\000\000\000\000>\000\000\000\005\000\000" .p2align 3 l_anon.[ID].5: .quad l_anon.[ID].0 - .asciz ",\000\000\000\000\000\000\000J\000\000\000\005\000\000" + .asciz ",\000\000\000\000\000\000\000H\000\000\000\005\000\000" .subsections_via_symbols diff --git a/test-assembly/crates/test_msg_send_id/expected/apple-armv7.s b/test-assembly/crates/test_msg_send_id/expected/apple-armv7.s index afd168946..7ee02f261 100644 --- a/test-assembly/crates/test_msg_send_id/expected/apple-armv7.s +++ b/test-assembly/crates/test_msg_send_id/expected/apple-armv7.s @@ -25,7 +25,7 @@ LBB1_1: LPC1_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc .p2align 2 @@ -52,7 +52,7 @@ LBB3_1: LPC3_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 1) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_init .p2align 2 @@ -79,7 +79,7 @@ LBB5_1: LPC5_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 2) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc_init .p2align 2 @@ -137,7 +137,7 @@ LBB10_1: LPC10_0: add r0, pc, r0 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 3) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_autoreleased .p2align 2 @@ -175,7 +175,7 @@ LBB12_1: LPC12_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 4) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .section __TEXT,__const l_anon.[ID].0: @@ -200,11 +200,11 @@ l_anon.[ID].3: .p2align 2 l_anon.[ID].4: .long l_anon.[ID].0 - .asciz ",\000\000\000@\000\000\000\005\000\000" + .asciz ",\000\000\000>\000\000\000\005\000\000" .p2align 2 l_anon.[ID].5: .long l_anon.[ID].0 - .asciz ",\000\000\000J\000\000\000\005\000\000" + .asciz ",\000\000\000H\000\000\000\005\000\000" .subsections_via_symbols diff --git a/test-assembly/crates/test_msg_send_id/expected/apple-armv7s.s b/test-assembly/crates/test_msg_send_id/expected/apple-armv7s.s index ceab8205b..a43e76a47 100644 --- a/test-assembly/crates/test_msg_send_id/expected/apple-armv7s.s +++ b/test-assembly/crates/test_msg_send_id/expected/apple-armv7s.s @@ -28,7 +28,7 @@ LBB1_1: LPC1_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc .p2align 2 @@ -58,7 +58,7 @@ LBB3_1: LPC3_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 1) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_init .p2align 2 @@ -88,7 +88,7 @@ LBB5_1: LPC5_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 2) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc_init .p2align 2 @@ -149,7 +149,7 @@ LBB10_1: LPC10_0: add r0, pc, r0 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 3) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_autoreleased .p2align 2 @@ -187,7 +187,7 @@ LBB12_1: LPC12_0: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 4) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .section __TEXT,__const l_anon.[ID].0: @@ -212,11 +212,11 @@ l_anon.[ID].3: .p2align 2 l_anon.[ID].4: .long l_anon.[ID].0 - .asciz ",\000\000\000@\000\000\000\005\000\000" + .asciz ",\000\000\000>\000\000\000\005\000\000" .p2align 2 l_anon.[ID].5: .long l_anon.[ID].0 - .asciz ",\000\000\000J\000\000\000\005\000\000" + .asciz ",\000\000\000H\000\000\000\005\000\000" .subsections_via_symbols diff --git a/test-assembly/crates/test_msg_send_id/expected/apple-x86.s b/test-assembly/crates/test_msg_send_id/expected/apple-x86.s index 072ee31a5..e603e4833 100644 --- a/test-assembly/crates/test_msg_send_id/expected/apple-x86.s +++ b/test-assembly/crates/test_msg_send_id/expected/apple-x86.s @@ -41,7 +41,7 @@ LBB1_2: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc .p2align 4, 0x90 @@ -84,7 +84,7 @@ LBB3_2: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 1) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_init .p2align 4, 0x90 @@ -127,7 +127,7 @@ LBB5_2: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 2) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc_init .p2align 4, 0x90 @@ -222,7 +222,7 @@ L10$pb: LBB10_2: lea eax, [esi + l_anon.[ID].4-L10$pb] mov dword ptr [esp], eax - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 3) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_autoreleased .p2align 4, 0x90 @@ -288,7 +288,7 @@ LBB12_2: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 4) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .section __TEXT,__const l_anon.[ID].0: @@ -313,11 +313,11 @@ l_anon.[ID].3: .p2align 2 l_anon.[ID].4: .long l_anon.[ID].0 - .asciz ",\000\000\000@\000\000\000\005\000\000" + .asciz ",\000\000\000>\000\000\000\005\000\000" .p2align 2 l_anon.[ID].5: .long l_anon.[ID].0 - .asciz ",\000\000\000J\000\000\000\005\000\000" + .asciz ",\000\000\000H\000\000\000\005\000\000" .subsections_via_symbols diff --git a/test-assembly/crates/test_msg_send_id/expected/apple-x86_64.s b/test-assembly/crates/test_msg_send_id/expected/apple-x86_64.s index 5bb36f4f9..62145fe0a 100644 --- a/test-assembly/crates/test_msg_send_id/expected/apple-x86_64.s +++ b/test-assembly/crates/test_msg_send_id/expected/apple-x86_64.s @@ -28,7 +28,7 @@ LBB1_2: lea rdx, [rip + l_anon.[ID].1] mov rdi, rbx mov rsi, r14 - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc .p2align 4, 0x90 @@ -58,7 +58,7 @@ LBB3_2: lea rdx, [rip + l_anon.[ID].2] mov rdi, rbx mov rsi, r14 - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 1) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_init .p2align 4, 0x90 @@ -88,7 +88,7 @@ LBB5_2: lea rdx, [rip + l_anon.[ID].3] mov rdi, rbx mov rsi, r14 - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 2) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_alloc_init .p2align 4, 0x90 @@ -154,7 +154,7 @@ _handle_copy_fallible: ret LBB10_2: lea rdi, [rip + l_anon.[ID].4] - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 3) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _handle_autoreleased .p2align 4, 0x90 @@ -199,7 +199,7 @@ LBB12_2: lea rdx, [rip + l_anon.[ID].5] mov rdi, rbx mov rsi, r14 - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 4) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .section __TEXT,__const l_anon.[ID].0: @@ -224,11 +224,11 @@ l_anon.[ID].3: .p2align 3 l_anon.[ID].4: .quad l_anon.[ID].0 - .asciz ",\000\000\000\000\000\000\000@\000\000\000\005\000\000" + .asciz ",\000\000\000\000\000\000\000>\000\000\000\005\000\000" .p2align 3 l_anon.[ID].5: .quad l_anon.[ID].0 - .asciz ",\000\000\000\000\000\000\000J\000\000\000\005\000\000" + .asciz ",\000\000\000\000\000\000\000H\000\000\000\005\000\000" .subsections_via_symbols diff --git a/test-assembly/crates/test_msg_send_id/expected/gnustep-x86.s b/test-assembly/crates/test_msg_send_id/expected/gnustep-x86.s index 9dae7bdf0..03988629b 100644 --- a/test-assembly/crates/test_msg_send_id/expected/gnustep-x86.s +++ b/test-assembly/crates/test_msg_send_id/expected/gnustep-x86.s @@ -67,7 +67,7 @@ handle_new_fallible: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@PLT + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@PLT add esp, 16 ud2 .Lfunc_end1: @@ -140,7 +140,7 @@ handle_alloc_fallible: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 1)@PLT + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@PLT add esp, 16 ud2 .Lfunc_end3: @@ -223,7 +223,7 @@ handle_init_fallible: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 2)@PLT + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@PLT add esp, 16 ud2 .Lfunc_end5: @@ -420,7 +420,7 @@ handle_copy_fallible: .LBB10_1: lea eax, [ebx + .Lanon.[ID].4@GOTOFF] mov dword ptr [esp], eax - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 3)@PLT + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@PLT ud2 .Lfunc_end10: .size handle_copy_fallible, .Lfunc_end10-handle_copy_fallible @@ -496,7 +496,7 @@ handle_autoreleased_fallible: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 4)@PLT + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@PLT add esp, 16 ud2 .Lfunc_end12: @@ -537,7 +537,7 @@ handle_autoreleased_fallible: .p2align 2 .Lanon.[ID].4: .long .Lanon.[ID].0 - .asciz ",\000\000\000@\000\000\000\005\000\000" + .asciz ",\000\000\000>\000\000\000\005\000\000" .size .Lanon.[ID].4, 16 .type .Lanon.[ID].5,@object @@ -545,7 +545,7 @@ handle_autoreleased_fallible: .p2align 2 .Lanon.[ID].5: .long .Lanon.[ID].0 - .asciz ",\000\000\000J\000\000\000\005\000\000" + .asciz ",\000\000\000H\000\000\000\005\000\000" .size .Lanon.[ID].5, 16 .section ".note.GNU-stack","",@progbits diff --git a/test-assembly/crates/test_msg_send_id/expected/gnustep-x86_64.s b/test-assembly/crates/test_msg_send_id/expected/gnustep-x86_64.s index 084dcb6f2..9bbc515c6 100644 --- a/test-assembly/crates/test_msg_send_id/expected/gnustep-x86_64.s +++ b/test-assembly/crates/test_msg_send_id/expected/gnustep-x86_64.s @@ -44,7 +44,7 @@ handle_new_fallible: lea rdx, [rip + .Lanon.[ID].1] mov rdi, rbx mov rsi, r14 - call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@GOTPCREL] + call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@GOTPCREL] ud2 .Lfunc_end1: .size handle_new_fallible, .Lfunc_end1-handle_new_fallible @@ -93,7 +93,7 @@ handle_alloc_fallible: lea rdx, [rip + .Lanon.[ID].2] mov rdi, rbx mov rsi, r14 - call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 1)@GOTPCREL] + call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@GOTPCREL] ud2 .Lfunc_end3: .size handle_alloc_fallible, .Lfunc_end3-handle_alloc_fallible @@ -151,7 +151,7 @@ handle_init_fallible: lea rdx, [rip + .Lanon.[ID].3] mov rdi, rbx mov rsi, r14 - call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 2)@GOTPCREL] + call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@GOTPCREL] ud2 .Lfunc_end5: .size handle_init_fallible, .Lfunc_end5-handle_init_fallible @@ -294,7 +294,7 @@ handle_copy_fallible: ret .LBB10_1: lea rdi, [rip + .Lanon.[ID].4] - call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 3)@GOTPCREL] + call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@GOTPCREL] ud2 .Lfunc_end10: .size handle_copy_fallible, .Lfunc_end10-handle_copy_fallible @@ -347,7 +347,7 @@ handle_autoreleased_fallible: lea rdx, [rip + .Lanon.[ID].5] mov rdi, rbx mov rsi, r14 - call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 4)@GOTPCREL] + call qword ptr [rip + SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0)@GOTPCREL] ud2 .Lfunc_end12: .size handle_autoreleased_fallible, .Lfunc_end12-handle_autoreleased_fallible @@ -387,7 +387,7 @@ handle_autoreleased_fallible: .p2align 3 .Lanon.[ID].4: .quad .Lanon.[ID].0 - .asciz ",\000\000\000\000\000\000\000@\000\000\000\005\000\000" + .asciz ",\000\000\000\000\000\000\000>\000\000\000\005\000\000" .size .Lanon.[ID].4, 24 .type .Lanon.[ID].5,@object @@ -395,7 +395,7 @@ handle_autoreleased_fallible: .p2align 3 .Lanon.[ID].5: .quad .Lanon.[ID].0 - .asciz ",\000\000\000\000\000\000\000J\000\000\000\005\000\000" + .asciz ",\000\000\000\000\000\000\000H\000\000\000\005\000\000" .size .Lanon.[ID].5, 24 .section ".note.GNU-stack","",@progbits diff --git a/test-assembly/crates/test_msg_send_id/lib.rs b/test-assembly/crates/test_msg_send_id/lib.rs index 24946b07e..929218caf 100644 --- a/test-assembly/crates/test_msg_send_id/lib.rs +++ b/test-assembly/crates/test_msg_send_id/lib.rs @@ -1,75 +1,73 @@ //! Test assembly output of `msg_send_id!` internals. -use objc2::__macro_helpers::{MsgSendId, RetainSemantics}; +use objc2::__macro_helpers::{Alloc, CopyOrMutCopy, Init, MsgSendId, New, Other}; use objc2::rc::{Allocated, Id, Shared}; use objc2::runtime::{Class, Object, Sel}; #[no_mangle] unsafe fn handle_new(cls: &Class, sel: Sel) -> Option> { - >::send_message_id(cls, sel, ()) + New::send_message_id(cls, sel, ()) } #[no_mangle] unsafe fn handle_new_fallible(cls: &Class, sel: Sel) -> Id { - >::send_message_id(cls, sel, ()) + New::send_message_id(cls, sel, ()) } #[no_mangle] unsafe fn handle_alloc(cls: &Class, sel: Sel) -> Option> { - >::send_message_id(cls, sel, ()) + Alloc::send_message_id(cls, sel, ()) } #[no_mangle] unsafe fn handle_alloc_fallible(cls: &Class, sel: Sel) -> Allocated { - >::send_message_id(cls, sel, ()) + Alloc::send_message_id(cls, sel, ()) } #[no_mangle] unsafe fn handle_init(obj: Option>, sel: Sel) -> Option> { - >::send_message_id(obj, sel, ()) + Init::send_message_id(obj, sel, ()) } #[no_mangle] unsafe fn handle_init_fallible(obj: Option>, sel: Sel) -> Id { - >::send_message_id(obj, sel, ()) + Init::send_message_id(obj, sel, ()) } #[no_mangle] unsafe fn handle_alloc_init(cls: &Class, sel1: Sel, sel2: Sel) -> Option> { - let obj = >::send_message_id(cls, sel1, ()); - >::send_message_id(obj, sel2, ()) + let obj = Alloc::send_message_id(cls, sel1, ()); + Init::send_message_id(obj, sel2, ()) } #[no_mangle] unsafe fn handle_alloc_release(cls: &Class, sel: Sel) { - let obj: Option> = - >::send_message_id(cls, sel, ()); + let obj: Option> = Alloc::send_message_id(cls, sel, ()); let _obj = obj.unwrap_unchecked(); } #[no_mangle] unsafe fn handle_alloc_init_release(cls: &Class, sel1: Sel, sel2: Sel) { - let obj = >::send_message_id(cls, sel1, ()); - let obj: Option> = - >::send_message_id(obj, sel2, ()); + let obj = Alloc::send_message_id(cls, sel1, ()); + let obj: Option> = Init::send_message_id(obj, sel2, ()); let _obj = obj.unwrap_unchecked(); } #[no_mangle] unsafe fn handle_copy(obj: &Object, sel: Sel) -> Option> { - >::send_message_id(obj, sel, ()) + CopyOrMutCopy::send_message_id(obj, sel, ()) } #[no_mangle] unsafe fn handle_copy_fallible(obj: &Object, sel: Sel) -> Id { - >::send_message_id(obj, sel, ()) + CopyOrMutCopy::send_message_id(obj, sel, ()) } #[no_mangle] unsafe fn handle_autoreleased(obj: &Object, sel: Sel) -> Option> { - >::send_message_id(obj, sel, ()) + Other::send_message_id(obj, sel, ()) } #[no_mangle] unsafe fn handle_autoreleased_fallible(obj: &Object, sel: Sel) -> Id { - >::send_message_id(obj, sel, ()) + Other::send_message_id(obj, sel, ()) } diff --git a/test-assembly/crates/test_msg_send_static_sel/expected/apple-aarch64.s b/test-assembly/crates/test_msg_send_static_sel/expected/apple-aarch64.s index 5c3c579f0..945c1eafe 100644 --- a/test-assembly/crates/test_msg_send_static_sel/expected/apple-aarch64.s +++ b/test-assembly/crates/test_msg_send_static_sel/expected/apple-aarch64.s @@ -42,7 +42,7 @@ Lloh9: add x2, x2, l_anon.[ID].1@PAGEOFF mov x0, x20 mov x1, x19 - bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + bl SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .loh AdrpLdrGotLdr Lloh5, Lloh6, Lloh7 .loh AdrpLdrGotLdr Lloh2, Lloh3, Lloh4 .loh AdrpAdd Lloh8, Lloh9 diff --git a/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7.s b/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7.s index ae48e6d7b..7bf1f0327 100644 --- a/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7.s +++ b/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7.s @@ -40,7 +40,7 @@ LBB1_1: LPC1_2: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _use_generic .p2align 2 diff --git a/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7s.s b/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7s.s index e9025bf8d..23b6b9c41 100644 --- a/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7s.s +++ b/test-assembly/crates/test_msg_send_static_sel/expected/apple-armv7s.s @@ -43,7 +43,7 @@ LBB1_1: LPC1_2: add r2, pc, r2 mov lr, pc - b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + b SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _use_generic .p2align 2 diff --git a/test-assembly/crates/test_msg_send_static_sel/expected/apple-old-x86.s b/test-assembly/crates/test_msg_send_static_sel/expected/apple-old-x86.s index 218d65363..0f40559b4 100644 --- a/test-assembly/crates/test_msg_send_static_sel/expected/apple-old-x86.s +++ b/test-assembly/crates/test_msg_send_static_sel/expected/apple-old-x86.s @@ -57,7 +57,7 @@ LBB1_2: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _use_generic .p2align 4, 0x90 diff --git a/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86.s b/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86.s index 47cbec54a..a60a68436 100644 --- a/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86.s +++ b/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86.s @@ -57,7 +57,7 @@ LBB1_2: push eax push edi push esi - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _use_generic .p2align 4, 0x90 diff --git a/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86_64.s b/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86_64.s index ef96130b6..9b18f7bb1 100644 --- a/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86_64.s +++ b/test-assembly/crates/test_msg_send_static_sel/expected/apple-x86_64.s @@ -35,7 +35,7 @@ LBB1_2: lea rdx, [rip + l_anon.[ID].1] mov rdi, rbx mov rsi, r14 - call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) + call SYM( as objc2::__macro_helpers::MsgSendIdFailed>::failed::GENERATED_ID, 0) .globl _use_generic .p2align 4, 0x90 diff --git a/test-ui/ui/msg_send_id_invalid_receiver.stderr b/test-ui/ui/msg_send_id_invalid_receiver.stderr index 826448e12..2f98bd39c 100644 --- a/test-ui/ui/msg_send_id_invalid_receiver.stderr +++ b/test-ui/ui/msg_send_id_invalid_receiver.stderr @@ -95,7 +95,7 @@ error[E0277]: the trait bound `Id: MessageReceiv = help: the following other types implement trait `MessageReceiver`: &'a Id &'a mut Id - = note: required for `RetainSemantics` to implement `MsgSendId, Id<_, _>>` + = note: required for `RetainSemantics<1>` to implement `MsgSendId, Id<_, _>>` error[E0277]: the trait bound `Id: MessageReceiver` is not satisfied --> ui/msg_send_id_invalid_receiver.rs @@ -109,4 +109,4 @@ error[E0277]: the trait bound `Id: MessageReceiv = help: the following other types implement trait `MessageReceiver`: &'a Id &'a mut Id - = note: required for `RetainSemantics` to implement `MsgSendId, Id<_, _>>` + = note: required for `RetainSemantics<4>` to implement `MsgSendId, Id<_, _>>` diff --git a/test-ui/ui/msg_send_id_invalid_return.stderr b/test-ui/ui/msg_send_id_invalid_return.stderr index f5746cdbc..3eb7d0e29 100644 --- a/test-ui/ui/msg_send_id_invalid_return.stderr +++ b/test-ui/ui/msg_send_id_invalid_return.stderr @@ -35,7 +35,7 @@ error[E0277]: the trait bound `objc2::runtime::Class: Message` is not satisfied NSError NSException and $N others - = note: required for `RetainSemantics` to implement `MsgSendId<&objc2::runtime::Class, Id>` + = note: required for `RetainSemantics<1>` to implement `MsgSendId<&objc2::runtime::Class, Id>` error[E0277]: the trait bound `objc2::runtime::Class: Message` is not satisfied --> ui/msg_send_id_invalid_return.rs @@ -56,7 +56,7 @@ error[E0277]: the trait bound `objc2::runtime::Class: Message` is not satisfied NSError NSException and $N others - = note: required for `RetainSemantics` to implement `MsgSendId<&objc2::runtime::Class, Id>` + = note: required for `RetainSemantics<1>` to implement `MsgSendId<&objc2::runtime::Class, Id>` error[E0277]: the trait bound `&objc2::runtime::Object: MaybeUnwrap` is not satisfied --> ui/msg_send_id_invalid_return.rs @@ -95,7 +95,7 @@ error[E0277]: the trait bound `objc2::runtime::Class: Message` is not satisfied NSError NSException and $N others - = note: required for `RetainSemantics` to implement `MsgSendId<&objc2::runtime::Class, Allocated>` + = note: required for `RetainSemantics<2>` to implement `MsgSendId<&objc2::runtime::Class, Allocated>` error[E0271]: type mismatch resolving ` as MaybeUnwrap>::Input == Allocated<_>` --> ui/msg_send_id_invalid_return.rs From be6c78fc7fbbe65419d633f5d7fd71fade2d60f6 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Thu, 27 Oct 2022 16:21:37 +0200 Subject: [PATCH 3/5] Add ability to nicely handle NSError** parameters --- objc2/CHANGELOG.md | 8 +- objc2/src/__macro_helpers.rs | 202 ++++++++---- objc2/src/macros.rs | 231 ++++++++++---- objc2/src/macros/__msg_send_parse.rs | 100 ++++++ objc2/src/macros/extern_methods.rs | 41 ++- objc2/src/message/mod.rs | 104 +++++- objc2/src/rc/id.rs | 2 +- objc2/src/rc/test_object.rs | 295 +++++++++++++++++- objc2/tests/use_macros.rs | 2 - test-ui/ui/extern_methods_wrong_arguments.rs | 7 + .../ui/extern_methods_wrong_arguments.stderr | 60 +++- test-ui/ui/invalid_msg_send.rs | 4 + test-ui/ui/invalid_msg_send.stderr | 35 ++- test-ui/ui/msg_send_id_invalid_return.stderr | 6 +- test-ui/ui/msg_send_id_underspecified.stderr | 2 +- test-ui/ui/msg_send_invalid_error.rs | 21 ++ test-ui/ui/msg_send_invalid_error.stderr | 97 ++++++ test-ui/ui/msg_send_mutable.stderr | 8 +- test-ui/ui/msg_send_no_return_type.stderr | 2 +- test-ui/ui/msg_send_not_encode.stderr | 4 +- test-ui/ui/msg_send_underspecified_error.rs | 13 + .../ui/msg_send_underspecified_error.stderr | 84 +++++ test-ui/ui/msg_send_underspecified_error2.rs | 9 + .../ui/msg_send_underspecified_error2.stderr | 7 + 24 files changed, 1169 insertions(+), 175 deletions(-) create mode 100644 objc2/src/macros/__msg_send_parse.rs create mode 100644 test-ui/ui/msg_send_invalid_error.rs create mode 100644 test-ui/ui/msg_send_invalid_error.stderr create mode 100644 test-ui/ui/msg_send_underspecified_error.rs create mode 100644 test-ui/ui/msg_send_underspecified_error.stderr create mode 100644 test-ui/ui/msg_send_underspecified_error2.rs create mode 100644 test-ui/ui/msg_send_underspecified_error2.stderr diff --git a/objc2/CHANGELOG.md b/objc2/CHANGELOG.md index eb3cc22b9..d3f02903e 100644 --- a/objc2/CHANGELOG.md +++ b/objc2/CHANGELOG.md @@ -23,7 +23,13 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). msg_send_id![NSObject::alloc(), init] }; ``` -* Add `Class::class_method`. +* Added `Class::class_method`. +* Added the ability to specify `error: _`, `somethingReturningError: _` and + so on at the end of `msg_send!`/`msg_send_id!`, and have it automatically + return a `Result<..., Id>`. +* Added the ability to specify an extra parameter at the end of the selector + in methods declared with `extern_methods!`, and let that be the `NSError**` + parameter. ### Changed * Allow other types than `&Class` as the receiver in `msg_send_id!` methods diff --git a/objc2/src/__macro_helpers.rs b/objc2/src/__macro_helpers.rs index 3de840cb4..3de982292 100644 --- a/objc2/src/__macro_helpers.rs +++ b/objc2/src/__macro_helpers.rs @@ -1,12 +1,15 @@ #[cfg(feature = "verify_message")] use alloc::vec::Vec; +use core::ptr; #[cfg(feature = "verify_message")] use std::collections::HashSet; use crate::declare::ClassBuilder; #[cfg(feature = "verify_message")] use crate::declare::MethodImplementation; -use crate::rc::{Allocated, Id, Ownership}; +use crate::encode::Encode; +use crate::message::__TupleExtender; +use crate::rc::{Allocated, Id, Ownership, Shared}; #[cfg(feature = "verify_message")] use crate::runtime::MethodDescription; use crate::runtime::{Class, Object, Protocol, Sel}; @@ -103,6 +106,56 @@ pub trait MsgSendId { sel: Sel, args: A, ) -> R; + + /// Add an extra error argument to the argument list, call + /// `send_message_id` with that, and return an error if one occurred. + #[inline] + #[track_caller] + unsafe fn send_message_id_error(obj: T, sel: Sel, args: A) -> Result> + where + *mut *mut E: Encode, + A: __TupleExtender<*mut *mut E>, + >::PlusOneArgument: MessageArguments, + E: Message, + Option: MaybeUnwrap, + { + let mut err: *mut E = ptr::null_mut(); + let args = args.add_argument(&mut err); + let res: Option = unsafe { Self::send_message_id(obj, sel, args) }; + // As per the Cocoa documentation: + // > Success or failure is indicated by the return value of the + // > method. Although Cocoa methods that indirectly return error + // > objects in the Cocoa error domain are guaranteed to return such + // > objects if the method indicates failure by directly returning + // > `nil` or `NO`, you should always check that the return value is + // > `nil` or `NO` before attempting to do anything with the `NSError` + // > object. + if let Some(res) = res { + // In this case, the error is likely not created. If it is, it is + // autoreleased anyhow, so it would be a waste to retain and + // release it. + Ok(res) + } else { + // In this case, the error has very likely been created, but has + // been autoreleased (as is common for "out parameters"). Hence we + // need to retain it if we want it to live across autorelease + // pools. + // + // SAFETY: The closure `f` is guaranteed to populate the error + // object, or leave it as NULL. The error is shared, and all + // holders of the error know this, so is safe to retain. + Err(unsafe { encountered_error(err) }) + } + } +} + +// Marked `cold` to tell the optimizer that errors are comparatively rare. +// And intentionally not inlined, for much the same reason. +#[cold] +#[track_caller] +unsafe fn encountered_error(err: *mut E) -> Id { + // SAFETY: Ensured by caller + unsafe { Id::retain(err) }.expect("error parameter should be set if the method returns NULL") } impl MsgSendId> for New { @@ -555,6 +608,7 @@ impl Drop for ClassProtocolMethodsBuilder<'_, '_> { mod tests { use super::*; + use alloc::string::ToString; use alloc::vec; use core::ptr; @@ -777,80 +831,92 @@ mod tests { #[test] fn test_in_selector_family() { + fn assert_in_family(selector: &str, family: &str) { + assert!(in_selector_family(selector.as_bytes(), family.as_bytes())); + let selector = selector.to_string() + "\0"; + assert!(in_selector_family(selector.as_bytes(), family.as_bytes())); + } + + fn assert_not_in_family(selector: &str, family: &str) { + assert!(!in_selector_family(selector.as_bytes(), family.as_bytes())); + let selector = selector.to_string() + "\0"; + assert!(!in_selector_family(selector.as_bytes(), family.as_bytes())); + } + // Common cases - assert!(in_selector_family(b"alloc", b"alloc")); - assert!(in_selector_family(b"allocWithZone:", b"alloc")); - assert!(!in_selector_family(b"dealloc", b"alloc")); - assert!(!in_selector_family(b"initialize", b"init")); - assert!(!in_selector_family(b"decimalNumberWithDecimal:", b"init")); - assert!(in_selector_family(b"initWithCapacity:", b"init")); - assert!(in_selector_family(b"_initButPrivate:withParam:", b"init")); - assert!(!in_selector_family(b"description", b"init")); - assert!(!in_selector_family(b"inIT", b"init")); - - assert!(!in_selector_family(b"init", b"copy")); - assert!(!in_selector_family(b"copyingStuff:", b"copy")); - assert!(in_selector_family(b"copyWithZone:", b"copy")); - assert!(!in_selector_family(b"initWithArray:copyItems:", b"copy")); - assert!(in_selector_family(b"copyItemAtURL:toURL:error:", b"copy")); - - assert!(!in_selector_family(b"mutableCopying", b"mutableCopy")); - assert!(in_selector_family(b"mutableCopyWithZone:", b"mutableCopy")); - assert!(in_selector_family(b"mutableCopyWithZone:", b"mutableCopy")); - - assert!(in_selector_family( - b"newScriptingObjectOfClass:forValueForKey:withContentsValue:properties:", - b"new" - )); - assert!(in_selector_family( - b"newScriptingObjectOfClass:forValueForKey:withContentsValue:properties:", - b"new" - )); - assert!(!in_selector_family(b"newsstandAssetDownload", b"new")); + assert_in_family("alloc", "alloc"); + assert_in_family("allocWithZone:", "alloc"); + assert_not_in_family("dealloc", "alloc"); + assert_not_in_family("initialize", "init"); + assert_not_in_family("decimalNumberWithDecimal:", "init"); + assert_in_family("initWithCapacity:", "init"); + assert_in_family("_initButPrivate:withParam:", "init"); + assert_not_in_family("description", "init"); + assert_not_in_family("inIT", "init"); + + assert_not_in_family("init", "copy"); + assert_not_in_family("copyingStuff:", "copy"); + assert_in_family("copyWithZone:", "copy"); + assert_not_in_family("initWithArray:copyItems:", "copy"); + assert_in_family("copyItemAtURL:toURL:error:", "copy"); + + assert_not_in_family("mutableCopying", "mutableCopy"); + assert_in_family("mutableCopyWithZone:", "mutableCopy"); + assert_in_family("mutableCopyWithZone:", "mutableCopy"); + + assert_in_family( + "newScriptingObjectOfClass:forValueForKey:withContentsValue:properties:", + "new", + ); + assert_in_family( + "newScriptingObjectOfClass:forValueForKey:withContentsValue:properties:", + "new", + ); + assert_not_in_family("newsstandAssetDownload", "new"); // Trying to weed out edge-cases: - assert!(in_selector_family(b"__abcDef", b"abc")); - assert!(in_selector_family(b"_abcDef", b"abc")); - assert!(in_selector_family(b"abcDef", b"abc")); - assert!(in_selector_family(b"___a", b"a")); - assert!(in_selector_family(b"__a", b"a")); - assert!(in_selector_family(b"_a", b"a")); - assert!(in_selector_family(b"a", b"a")); - - assert!(!in_selector_family(b"_abcdef", b"abc")); - assert!(!in_selector_family(b"_abcdef", b"def")); - assert!(!in_selector_family(b"_bcdef", b"abc")); - assert!(!in_selector_family(b"a_bc", b"abc")); - assert!(!in_selector_family(b"abcdef", b"abc")); - assert!(!in_selector_family(b"abcdef", b"def")); - assert!(!in_selector_family(b"abcdef", b"abb")); - assert!(!in_selector_family(b"___", b"a")); - assert!(!in_selector_family(b"_", b"a")); - assert!(!in_selector_family(b"", b"a")); - - assert!(in_selector_family(b"copy", b"copy")); - assert!(in_selector_family(b"copy:", b"copy")); - assert!(in_selector_family(b"copyMe", b"copy")); - assert!(in_selector_family(b"_copy", b"copy")); - assert!(in_selector_family(b"_copy:", b"copy")); - assert!(in_selector_family(b"_copyMe", b"copy")); - assert!(!in_selector_family(b"copying", b"copy")); - assert!(!in_selector_family(b"copying:", b"copy")); - assert!(!in_selector_family(b"_copying", b"copy")); - assert!(!in_selector_family(b"Copy", b"copy")); - assert!(!in_selector_family(b"COPY", b"copy")); + assert_in_family("__abcDef", "abc"); + assert_in_family("_abcDef", "abc"); + assert_in_family("abcDef", "abc"); + assert_in_family("___a", "a"); + assert_in_family("__a", "a"); + assert_in_family("_a", "a"); + assert_in_family("a", "a"); + + assert_not_in_family("_abcdef", "abc"); + assert_not_in_family("_abcdef", "def"); + assert_not_in_family("_bcdef", "abc"); + assert_not_in_family("a_bc", "abc"); + assert_not_in_family("abcdef", "abc"); + assert_not_in_family("abcdef", "def"); + assert_not_in_family("abcdef", "abb"); + assert_not_in_family("___", "a"); + assert_not_in_family("_", "a"); + assert_not_in_family("", "a"); + + assert_in_family("copy", "copy"); + assert_in_family("copy:", "copy"); + assert_in_family("copyMe", "copy"); + assert_in_family("_copy", "copy"); + assert_in_family("_copy:", "copy"); + assert_in_family("_copyMe", "copy"); + assert_not_in_family("copying", "copy"); + assert_not_in_family("copying:", "copy"); + assert_not_in_family("_copying", "copy"); + assert_not_in_family("Copy", "copy"); + assert_not_in_family("COPY", "copy"); // Empty family (not supported) - assert!(in_selector_family(b"___", b"")); - assert!(in_selector_family(b"__", b"")); - assert!(in_selector_family(b"_", b"")); - assert!(in_selector_family(b"", b"")); - assert!(!in_selector_family(b"_a", b"")); - assert!(!in_selector_family(b"a", b"")); - assert!(in_selector_family(b"_A", b"")); - assert!(in_selector_family(b"A", b"")); + assert_in_family("___", ""); + assert_in_family("__", ""); + assert_in_family("_", ""); + assert_in_family("", ""); + assert_not_in_family("_a", ""); + assert_not_in_family("a", ""); + assert_in_family("_A", ""); + assert_in_family("A", ""); } mod test_trait_disambugated { diff --git a/objc2/src/macros.rs b/objc2/src/macros.rs index 368efb0a8..b01ff1963 100644 --- a/objc2/src/macros.rs +++ b/objc2/src/macros.rs @@ -1,3 +1,4 @@ +mod __msg_send_parse; mod __rewrite_self_arg; mod declare_class; mod extern_class; @@ -641,6 +642,9 @@ macro_rules! __class_inner { /// /// All arguments, and the return type, must implement [`Encode`]. /// +/// If the last argument is the special marker `_`, the macro will return a +/// `Result<(), Id>`, see below. +/// /// This macro translates into a call to [`sel!`], and afterwards a fully /// qualified call to [`MessageReceiver::send_message`]. Note that this means /// that auto-dereferencing of the receiver is not supported, and that the @@ -675,6 +679,33 @@ macro_rules! __class_inner { /// [`runtime::Bool`]: crate::runtime::Bool /// /// +/// # Errors +/// +/// Many methods take an `NSError**` as their last parameter, which is used to +/// communicate errors to the caller, see [Error Handling Programming Guide +/// For Cocoa][cocoa-error]. +/// +/// Similar to Swift's [importing of error parameters][swift-error], this +/// macro supports transforming methods whose last parameter is `NSError**` +/// and returns `BOOL`, into the Rust equivalent, the [`Result`] type. +/// +/// In particular, you can make the last argument the special marker `_`, and +/// then the macro will return a `Result<(), Id>` (where you must +/// specify `E` yourself, usually you'd use [`foundation::NSError`]). +/// +/// At runtime, a temporary error variable is created on the stack and sent as +/// the last parameter. If the message send then returns `NO`/`false` (or in +/// the case of `msg_send_id!`, `NULL`), the error variable is loaded and +/// returned in [`Err`]. +/// +/// Do beware that this is only valid on methods that return `BOOL`, see +/// [`msg_send_id!`] for methods that return instance types. +/// +/// [cocoa-error]: https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/ErrorHandlingCocoa/ErrorHandling/ErrorHandling.html +/// [swift-error]: https://developer.apple.com/documentation/swift/about-imported-cocoa-error-parameters +/// [`foundation::NSError`]: crate::foundation::NSError +/// +/// /// # Panics /// /// Panics if the `"catch-all"` feature is enabled and the Objective-C method @@ -685,6 +716,9 @@ macro_rules! __class_inner { /// method's argument's encoding does not match the encoding of the given /// arguments. This is highly recommended to enable while testing! /// +/// And panics if the `NSError**` handling functionality described above is +/// used, and the error object was unexpectedly `NULL`. +/// /// [RFC-2945]: https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html /// /// @@ -770,45 +804,79 @@ macro_rules! __class_inner { /// # superclass = class!(NSObject); /// let arg3: u32 = unsafe { msg_send![super(obj, superclass), getArg3] }; /// ``` +/// +/// Sending a message with automatic error handling. +/// +/// ```no_run +/// use objc2::msg_send; +/// use objc2::runtime::Object; +/// use objc2::rc::{Id, Shared}; +/// +/// let obj: *mut Object; // Let's assume an instance of `NSBundle` +/// # obj = 0 as *mut Object; +/// // The `_` tells the macro that the return type should be `Result`. +/// let res: Result<(), Id> = unsafe { +/// msg_send![obj, preflightAndReturnError: _] +/// }; +/// ``` #[macro_export] macro_rules! msg_send { - [super($obj:expr), $selector:ident $(,)?] => ({ - let sel = $crate::sel!($selector); - let result; - // Note: `sel` and `result` can be accessed from the `obj` and - // `superclass` expressions - we won't (yet) bother with preventing - // that though. - result = $crate::MessageReceiver::__send_super_message_static($obj, sel, ()); - result - }); - [super($obj:expr), $($selector:ident : $argument:expr),+ $(,)?] => ({ - let sel = $crate::sel!($($selector :)+); - let result; - result = $crate::MessageReceiver::__send_super_message_static($obj, sel, ($($argument,)+)); - result - }); - [super($obj:expr, $superclass:expr), $selector:ident $(,)?] => ({ - let sel = $crate::sel!($selector); - let result; - result = $crate::MessageReceiver::send_super_message($obj, $superclass, sel, ()); - result - }); - [super($obj:expr, $superclass:expr), $($selector:ident : $argument:expr $(,)?)+] => ({ - let sel = $crate::sel!($($selector :)+); - let result; - result = $crate::MessageReceiver::send_super_message($obj, $superclass, sel, ($($argument,)+)); - result - }); - [$obj:expr, $selector:ident $(,)?] => ({ - let sel = $crate::sel!($selector); - let result; - result = $crate::MessageReceiver::send_message($obj, sel, ()); - result - }); - [$obj:expr, $($selector:ident : $argument:expr $(,)?)+] => ({ - let sel = $crate::sel!($($selector :)+); + [super($obj:expr), $($selector_and_arguments:tt)+] => { + $crate::__msg_send_parse! { + ($crate::__msg_send_helper) + @(__send_super_message_static_error) + @() + @() + @($($selector_and_arguments)+) + @(__send_super_message_static) + + @($obj) + } + }; + [super($obj:expr, $superclass:expr), $($selector_and_arguments:tt)+] => { + $crate::__msg_send_parse! { + ($crate::__msg_send_helper) + @(__send_super_message_error) + @() + @() + @($($selector_and_arguments)+) + @(send_super_message) + + @($obj, $superclass) + } + }; + [$obj:expr, $($selector_and_arguments:tt)+] => { + $crate::__msg_send_parse! { + ($crate::__msg_send_helper) + @(__send_message_error) + @() + @() + @($($selector_and_arguments)+) + @(send_message) + + @($obj) + } + }; +} + +#[doc(hidden)] +#[macro_export] +macro_rules! __msg_send_helper { + { + @($fn:ident) + @($($fn_args:tt)+) + @($($selector:tt)*) + @($($argument:expr,)*) + } => ({ + // Assign to intermediary variable for better UI, and to prevent + // miscompilation on older Rust versions. + // + // Note: This can be accessed from any expression in `fn_args` and + // `arguments` - we won't (yet) bother with preventing that though. let result; - result = $crate::MessageReceiver::send_message($obj, sel, ($($argument,)+)); + // Always add trailing comma after each argument, so that we get a + // 1-tuple if there is only one. + result = $crate::MessageReceiver::$fn($($fn_args)+, $crate::sel!($($selector)*), ($($argument,)*)); result }); } @@ -910,6 +978,9 @@ macro_rules! msg_send_bool { /// `Id / Allocated` this macro will automatically unwrap the object, or panic /// with an error message if it couldn't be retrieved. /// +/// Though as a special case, if the last argument is the marker `_`, the +/// macro will return a `Result, Id>`, see below. +/// /// This macro doesn't support super methods yet, see [#173]. /// The `retain`, `release` and `autorelease` selectors are not supported, use /// [`Id::retain`], [`Id::drop`] and [`Id::autorelease`] for that. @@ -924,6 +995,19 @@ macro_rules! msg_send_bool { /// [`Id::autorelease`]: crate::rc::Id::autorelease /// /// +/// # Errors +/// +/// Very similarly to [`msg_send!`], this macro supports transforming the +/// return type of methods whose last parameter is `NSError**` into the Rust +/// equivalent, the [`Result`] type. +/// +/// In particular, you can make the last argument the special marker `_`, and +/// then the macro will return a `Result, Id>` (where you +/// must specify `E` yourself, usually you'd use [`foundation::NSError`]). +/// +/// [`foundation::NSError`]: crate::foundation::NSError +/// +/// /// # Panics /// /// Panics if the return type is specified as `Id<_, _>` and the method @@ -992,46 +1076,73 @@ macro_rules! msg_send_id { result = <$crate::__macro_helpers::Init as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ()); result }); - [$obj:expr, $selector:ident $(,)?] => ({ - $crate::__msg_send_id_helper!(@verify $selector); - let sel = $crate::sel!($selector); - const NAME: &$crate::__macro_helpers::str = $crate::__macro_helpers::stringify!($selector); - let result; - result = <$crate::__macro_helpers::RetainSemantics<{ - $crate::__macro_helpers::retain_semantics(NAME) - }> as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ()); - result - }); - [$obj:expr, $($selector:ident : $argument:expr),+ $(,)?] => ({ - let sel = $crate::sel!($($selector:)+); - const NAME: &$crate::__macro_helpers::str = - $crate::__macro_helpers::concat!($($crate::__macro_helpers::stringify!($selector), ':'),+); - let result; - result = <$crate::__macro_helpers::RetainSemantics<{ - $crate::__macro_helpers::retain_semantics(NAME) - }> as $crate::__macro_helpers::MsgSendId<_, _>>::send_message_id($obj, sel, ($($argument,)+)); - result - }); + [$obj:expr, $($selector_and_arguments:tt)+] => { + $crate::__msg_send_parse! { + ($crate::__msg_send_id_helper) + @(send_message_id_error) + @() + @() + @($($selector_and_arguments)+) + @(send_message_id) + + @($obj) + } + }; } /// Helper macro to avoid exposing these in the docs for [`msg_send_id!`]. #[doc(hidden)] #[macro_export] macro_rules! __msg_send_id_helper { - (@verify retain) => {{ + { + @($fn:ident) + @($obj:expr) + @(retain) + @() + } => {{ $crate::__macro_helpers::compile_error!( "msg_send_id![obj, retain] is not supported. Use `Id::retain` instead" ) }}; - (@verify release) => {{ + { + @($fn:ident) + @($obj:expr) + @(release) + @() + } => {{ $crate::__macro_helpers::compile_error!( "msg_send_id![obj, release] is not supported. Drop an `Id` instead" ) }}; - (@verify autorelease) => {{ + { + @($fn:ident) + @($obj:expr) + @(autorelease) + @() + } => {{ $crate::__macro_helpers::compile_error!( "msg_send_id![obj, autorelease] is not supported. Use `Id::autorelease`" ) }}; - (@verify $selector:ident) => {{}}; + { + @($fn:ident) + @($obj:expr) + @($sel_first:ident $(: $($sel_rest:ident :)*)?) + @($($argument:expr,)*) + } => ({ + // Don't use `sel!`, otherwise we'd end up with defining this data twice. + const __SELECTOR_DATA: &$crate::__macro_helpers::str = $crate::__sel_data!($sel_first $(: $($sel_rest :)*)?); + let result; + result = <$crate::__macro_helpers::RetainSemantics<{ + $crate::__macro_helpers::retain_semantics(__SELECTOR_DATA) + }> as $crate::__macro_helpers::MsgSendId<_, _>>::$fn( + $obj, + $crate::__sel_inner!( + __SELECTOR_DATA, + $crate::__hash_idents!($sel_first $($($sel_rest)*)?) + ), + ($($argument,)*), + ); + result + }); } diff --git a/objc2/src/macros/__msg_send_parse.rs b/objc2/src/macros/__msg_send_parse.rs new file mode 100644 index 000000000..7b4657159 --- /dev/null +++ b/objc2/src/macros/__msg_send_parse.rs @@ -0,0 +1,100 @@ +#[doc(hidden)] +#[macro_export] +macro_rules! __msg_send_parse { + // No arguments + { + ($out_macro:path) + @($error_fn:ident) + // Intentionally empty + @() + @() + @($selector:ident $(,)?) + $($macro_args:tt)* + } => { + $crate::__msg_send_parse! { + ($out_macro) + @($error_fn) + @($selector) + @() + @() + $($macro_args)* + } + }; + + // tt-munch remaining `selector: argument` pairs, looking for a pattern + // that ends with `sel: _`. + { + ($out_macro:path) + @($_error_fn:ident) + @($($selector_output:tt)*) + @($($argument_output:tt)*) + @() + $($macro_args:tt)* + } => ({ + $out_macro! { + $($macro_args)* + @($($selector_output)*) + @($($argument_output)*) + } + }); + { + ($out_macro:path) + @($error_fn:ident) + @($($selector_output:tt)*) + @($($argument_output:tt)*) + @($selector:ident: _ $(,)?) + @($fn:ident) + $($macro_args:tt)* + } => { + $crate::__msg_send_parse! { + ($out_macro) + @($error_fn) + @($($selector_output)* $selector:) + // Don't pass an argument + @($($argument_output)*) + @() + + // Instead, we change the called function to the error function. + @($error_fn) + $($macro_args)* + } + }; + { + ($out_macro:path) + @($error_fn:ident) + @($($selector_output:tt)*) + @($($argument_output:tt)*) + @($selector:ident : $argument:expr $(, $($rest:tt)*)?) + $($macro_args:tt)* + } => { + $crate::__msg_send_parse! { + ($out_macro) + @($error_fn) + @($($selector_output)* $selector:) + @($($argument_output)* $argument,) + @($($($rest)*)?) + $($macro_args)* + } + }; + + // Handle calls without comma between `selector: argument` pair. + // TODO: Deprecate this + { + ($out_macro:path) + @($error_fn:ident) + // Intentionally empty + @() + @() + @($($selector:ident : $argument:expr)*) + $($macro_args:tt)* + } => { + $crate::__msg_send_parse! { + ($out_macro) + @($error_fn) + @() + @() + @($($selector : $argument),*) + $($macro_args)* + } + }; +} diff --git a/objc2/src/macros/extern_methods.rs b/objc2/src/macros/extern_methods.rs index 45c32050f..07249fdb8 100644 --- a/objc2/src/macros/extern_methods.rs +++ b/objc2/src/macros/extern_methods.rs @@ -37,8 +37,9 @@ /// [`NSCalendar`]: https://developer.apple.com/documentation/foundation/nscalendar?language=objc /// /// ``` -/// use objc2::foundation::{NSObject, NSRange, NSString, NSUInteger}; +/// use objc2::foundation::{NSError, NSObject, NSRange, NSString, NSUInteger}; /// use objc2::rc::{Id, Shared}; +/// use objc2::runtime::Object; /// use objc2::{extern_class, extern_methods, msg_send_id, Encode, Encoding, ClassType}; /// # /// # #[cfg(feature = "gnustep-1-7")] @@ -103,6 +104,16 @@ /// /// #[sel(maximumRangeOfUnit:)] /// pub fn max_range(&self, unit: NSCalendarUnit) -> NSRange; +/// +/// // From `NSKeyValueCoding` +/// #[sel(validateValue:forKey:error:)] +/// pub unsafe fn validate_value_for_key( +/// &self, +/// value: &mut *mut Object, +/// key: &NSString, +/// // Since the selector specifies one more argument than we +/// // have, the return type is assumed to be `Result`. +/// ) -> Result<(), Id>; /// } /// ); /// ``` @@ -110,8 +121,9 @@ /// The `extern_methods!` declaration then becomes: /// /// ``` -/// # use objc2::foundation::{NSObject, NSRange, NSString, NSUInteger}; +/// # use objc2::foundation::{NSError, NSObject, NSRange, NSString, NSUInteger}; /// # use objc2::rc::{Id, Shared}; +/// # use objc2::runtime::Object; /// # use objc2::{extern_class, extern_methods, msg_send_id, Encode, Encoding, ClassType}; /// # /// # #[cfg(feature = "gnustep-1-7")] @@ -174,6 +186,14 @@ /// pub fn max_range(&self, unit: NSCalendarUnit) -> NSRange { /// unsafe { msg_send![self, maximumRangeOfUnit: unit] } /// } +/// +/// pub unsafe fn validate_value_for_key( +/// &self, +/// value: &mut *mut Object, +/// key: &NSString, +/// ) -> Result<(), Id> { +/// unsafe { msg_send![self, validateValue: value, forKey: key, error: _] } +/// } /// } /// ``` #[macro_export] @@ -370,6 +390,23 @@ macro_rules! __collect_msg_send { $macro![$obj, $($output)+] }}; + // Allow trailing `sel:` without a corresponding argument (for errors) + ( + $macro:path; + $obj:expr; + ($sel:ident:); + ($(,)?); + $($output:tt)* + ) => { + $crate::__collect_msg_send! { + $macro; + $obj; + (); + (); + $($output)* $sel: _, + } + }; + // tt-munch each argument ( $macro:path; diff --git a/objc2/src/message/mod.rs b/objc2/src/message/mod.rs index 7753300a6..768af743f 100644 --- a/objc2/src/message/mod.rs +++ b/objc2/src/message/mod.rs @@ -1,9 +1,9 @@ use core::mem; use core::mem::ManuallyDrop; -use core::ptr::NonNull; +use core::ptr::{self, NonNull}; use crate::encode::{Encode, EncodeArguments, EncodeConvert, RefEncode}; -use crate::rc::{Id, Owned, Ownership}; +use crate::rc::{Id, Owned, Ownership, Shared}; use crate::runtime::{Class, Imp, Object, Sel}; use crate::ClassType; @@ -249,6 +249,90 @@ pub unsafe trait MessageReceiver: private::Sealed + Sized { { unsafe { self.send_super_message(::Super::class(), sel, args) } } + + // Error functions below. See MsgSendId::send_message_id_error for further + // details. + // + // Some of this could be abstracted away using closures, but that would + // interfere with `#[track_caller]`, so we avoid doing that. + + #[inline] + #[track_caller] + #[doc(hidden)] + unsafe fn __send_message_error(self, sel: Sel, args: A) -> Result<(), Id> + where + *mut *mut E: Encode, + A: __TupleExtender<*mut *mut E>, + >::PlusOneArgument: MessageArguments, + E: Message, + { + let mut err: *mut E = ptr::null_mut(); + let args = args.add_argument(&mut err); + let res: bool = unsafe { self.send_message(sel, args) }; + if res { + Ok(()) + } else { + Err(unsafe { encountered_error(err) }) + } + } + + #[inline] + #[track_caller] + #[doc(hidden)] + unsafe fn __send_super_message_error( + self, + superclass: &Class, + sel: Sel, + args: A, + ) -> Result<(), Id> + where + *mut *mut E: Encode, + A: __TupleExtender<*mut *mut E>, + >::PlusOneArgument: MessageArguments, + E: Message, + { + let mut err: *mut E = ptr::null_mut(); + let args = args.add_argument(&mut err); + let res: bool = unsafe { self.send_super_message(superclass, sel, args) }; + if res { + Ok(()) + } else { + Err(unsafe { encountered_error(err) }) + } + } + + #[inline] + #[track_caller] + #[doc(hidden)] + unsafe fn __send_super_message_static_error( + self, + sel: Sel, + args: A, + ) -> Result<(), Id> + where + Self::__Inner: ClassType, + ::Super: ClassType, + *mut *mut E: Encode, + A: __TupleExtender<*mut *mut E>, + >::PlusOneArgument: MessageArguments, + E: Message, + { + let mut err: *mut E = ptr::null_mut(); + let args = args.add_argument(&mut err); + let res: bool = unsafe { self.__send_super_message_static(sel, args) }; + if res { + Ok(()) + } else { + Err(unsafe { encountered_error(err) }) + } + } +} + +#[cold] +#[track_caller] +unsafe fn encountered_error(err: *mut E) -> Id { + // SAFETY: Ensured by caller + unsafe { Id::retain(err) }.expect("error parameter should be set if the method returns NO") } // Note that we implement MessageReceiver for unsized types as well, this is @@ -367,6 +451,13 @@ pub unsafe trait MessageArguments: EncodeArguments { unsafe fn __invoke(imp: Imp, obj: *mut Object, sel: Sel, args: Self) -> R; } +pub trait __TupleExtender { + #[doc(hidden)] + type PlusOneArgument; + #[doc(hidden)] + fn add_argument(self, arg: T) -> Self::PlusOneArgument; +} + macro_rules! message_args_impl { ($($a:ident: $t:ident),*) => ( unsafe impl<$($t: EncodeConvert),*> MessageArguments for ($($t,)*) { @@ -390,6 +481,15 @@ macro_rules! message_args_impl { unsafe { imp(obj, sel $(, EncodeConvert::__into_inner($a))*) } } } + + impl<$($t,)* T> __TupleExtender for ($($t,)*) { + type PlusOneArgument = ($($t,)* T,); + + fn add_argument(self, arg: T) -> Self::PlusOneArgument { + let ($($a,)*) = self; + ($($a,)* arg,) + } + } ); } diff --git a/objc2/src/rc/id.rs b/objc2/src/rc/id.rs index 244489df0..6b3fc3b02 100644 --- a/objc2/src/rc/id.rs +++ b/objc2/src/rc/id.rs @@ -438,7 +438,7 @@ impl Id { } #[inline] - fn autorelease_inner(self) -> *mut T { + pub(super) fn autorelease_inner(self) -> *mut T { // Note that this (and the actual `autorelease`) is not an associated // function. This breaks the guideline that smart pointers shouldn't // add inherent methods, but since autoreleasing only works on already diff --git a/objc2/src/rc/test_object.rs b/objc2/src/rc/test_object.rs index 859db74a5..c71b09b19 100644 --- a/objc2/src/rc/test_object.rs +++ b/objc2/src/rc/test_object.rs @@ -4,7 +4,7 @@ use core::ptr; use super::{Id, Owned}; use crate::foundation::{NSObject, NSZone}; -use crate::{declare_class, msg_send, ClassType}; +use crate::{declare_class, msg_send, msg_send_id, ClassType}; #[derive(Debug, Clone, Default, PartialEq, Eq)] pub(crate) struct ThreadTestData { @@ -74,7 +74,7 @@ declare_class!( } #[sel(alloc)] - fn alloc() -> *mut Self { + fn alloc_() -> *mut Self { TEST_DATA.with(|data| data.borrow_mut().alloc += 1); let superclass = NSObject::class().metaclass(); let zone: *const NSZone = ptr::null(); @@ -161,6 +161,106 @@ declare_class!( fn method_returning_null(&self) -> *const Self { ptr::null() } + + #[sel(boolAndShouldError:error:)] + fn class_error_bool(should_error: bool, err: Option<&mut *mut RcTestObject>) -> bool { + if should_error { + if let Some(err) = err { + *err = RcTestObject::new().autorelease_inner(); + } + false + } else { + true + } + } + + #[sel(boolAndShouldError:error:)] + fn instance_error_bool( + &self, + should_error: bool, + err: Option<&mut *mut RcTestObject>, + ) -> bool { + if should_error { + if let Some(err) = err { + *err = RcTestObject::new().autorelease_inner(); + } + false + } else { + true + } + } + + #[sel(idAndShouldError:error:)] + fn class_error_id( + should_error: bool, + err: Option<&mut *mut RcTestObject>, + ) -> *mut RcTestObject { + if should_error { + if let Some(err) = err { + *err = RcTestObject::new().autorelease_inner(); + } + ptr::null_mut() + } else { + RcTestObject::new().autorelease_return() + } + } + + #[sel(idAndShouldError:error:)] + fn instance_error_id( + &self, + should_error: bool, + err: Option<&mut *mut RcTestObject>, + ) -> *mut RcTestObject { + if should_error { + if let Some(err) = err { + *err = RcTestObject::new().autorelease_inner(); + } + ptr::null_mut() + } else { + RcTestObject::new().autorelease_return() + } + } + + #[sel(newAndShouldError:error:)] + fn new_error(should_error: bool, err: Option<&mut *mut RcTestObject>) -> *mut Self { + if should_error { + if let Some(err) = err { + *err = RcTestObject::new().autorelease_inner(); + } + ptr::null_mut() + } else { + unsafe { msg_send![Self::class(), new] } + } + } + + #[sel(allocAndShouldError:error:)] + fn alloc_error(should_error: bool, err: Option<&mut *mut RcTestObject>) -> *mut Self { + if should_error { + if let Some(err) = err { + *err = RcTestObject::new().autorelease_inner(); + } + ptr::null_mut() + } else { + unsafe { msg_send![Self::class(), alloc] } + } + } + + #[sel(initAndShouldError:error:)] + fn init_error( + &mut self, + should_error: bool, + err: Option<&mut *mut RcTestObject>, + ) -> *mut Self { + if should_error { + if let Some(err) = err { + *err = RcTestObject::new().autorelease_inner(); + } + let _: () = unsafe { msg_send![self, release] }; + ptr::null_mut() + } else { + unsafe { msg_send![self, init] } + } + } } ); @@ -174,12 +274,203 @@ impl RcTestObject { } } +declare_class!( + #[derive(Debug, PartialEq)] + pub(crate) struct RcTestObjectSubclass {} + + unsafe impl ClassType for RcTestObjectSubclass { + #[inherits(NSObject)] + type Super = RcTestObject; + } +); + +impl RcTestObjectSubclass { + pub(crate) fn new() -> Id { + unsafe { msg_send_id![Self::class(), new] } + } +} + #[cfg(test)] mod tests { use super::*; + use crate::rc::{autoreleasepool, Allocated, Shared}; #[test] fn ensure_declared_name() { assert_eq!(RcTestObject::class().name(), RcTestObject::NAME); } + + macro_rules! test_error_bool { + ($expected:expr, $($obj:tt)*) => { + // Succeeds + let res: Result<(), Id> = unsafe { + msg_send![$($obj)*, boolAndShouldError: false, error: _] + }; + assert_eq!(res, Ok(())); + $expected.assert_current(); + + // Errors + let res = autoreleasepool(|_pool| { + // `Ok` type is inferred to be `()` + let res: Id = unsafe { + msg_send![$($obj)*, boolAndShouldError: true, error: _] + }.expect_err("not err"); + $expected.alloc += 1; + $expected.init += 1; + $expected.autorelease += 1; + $expected.retain += 1; + $expected.assert_current(); + res + }); + $expected.release += 1; + $expected.assert_current(); + + drop(res); + $expected.release += 1; + $expected.dealloc += 1; + $expected.assert_current(); + } + } + + #[test] + fn test_error_bool() { + let mut expected = ThreadTestData::current(); + + let cls = RcTestObject::class(); + test_error_bool!(expected, cls); + + let obj = RcTestObject::new(); + expected.alloc += 1; + expected.init += 1; + test_error_bool!(expected, &obj); + + let obj = RcTestObjectSubclass::new(); + expected.alloc += 1; + expected.init += 1; + test_error_bool!(expected, &obj); + test_error_bool!(expected, super(&obj)); + test_error_bool!(expected, super(&obj, RcTestObjectSubclass::class())); + test_error_bool!(expected, super(&obj, RcTestObject::class())); + } + + // This is imperfect, but will do for now. + // See also `tests/id_retain_autoreleased.rs`. + // + // Work around https://github.com/rust-lang/rust-clippy/issues/9737: + const IF_AUTORELEASE_NOT_SKIPPED: usize = if cfg!(feature = "gnustep-1-7") { + 1 + } else if cfg!(any( + debug_assertions, + feature = "exception", + feature = "verify_message" + )) { + 2 + } else { + 1 + } - 1; + + macro_rules! test_error_id { + ($expected:expr, $if_autorelease_not_skipped:expr, $sel:ident, $($obj:tt)*) => { + // Succeeds + let res = autoreleasepool(|_pool| { + let res: Result, Id> = unsafe { + msg_send_id![$($obj)*, $sel: false, error: _] + }; + let res = res.expect("not ok"); + $expected.alloc += 1; + $expected.init += 1; + $expected.autorelease += $if_autorelease_not_skipped; + $expected.retain += $if_autorelease_not_skipped; + $expected.assert_current(); + res + }); + $expected.release += $if_autorelease_not_skipped; + $expected.assert_current(); + + drop(res); + $expected.release += 1; + $expected.dealloc += 1; + $expected.assert_current(); + + // Errors + let res = autoreleasepool(|_pool| { + let res: Result, Id> = unsafe { + msg_send_id![$($obj)*, $sel: true, error: _] + }; + $expected.alloc += 1; + $expected.init += 1; + $expected.autorelease += 1; + $expected.retain += 1; + $expected.assert_current(); + res.expect_err("not err") + }); + $expected.release += 1; + $expected.assert_current(); + + drop(res); + $expected.release += 1; + $expected.dealloc += 1; + $expected.assert_current(); + } + } + + #[test] + fn test_error_id() { + let mut expected = ThreadTestData::current(); + + let cls = RcTestObject::class(); + test_error_id!(expected, IF_AUTORELEASE_NOT_SKIPPED, idAndShouldError, cls); + test_error_id!(expected, 0, newAndShouldError, cls); + + let obj = RcTestObject::new(); + expected.alloc += 1; + expected.init += 1; + test_error_id!(expected, IF_AUTORELEASE_NOT_SKIPPED, idAndShouldError, &obj); + + expected.alloc -= 1; + expected.release -= 1; + expected.dealloc -= 1; + test_error_id!(expected, 0, initAndShouldError, { + expected.alloc += 1; + expected.release += 1; + expected.dealloc += 1; + RcTestObject::alloc() + }); + } + + #[test] + fn test_error_alloc() { + let mut expected = ThreadTestData::current(); + + // Succeeds + let res: Result, Id> = + unsafe { msg_send_id![RcTestObject::class(), allocAndShouldError: false, error: _] }; + let res = res.expect("not ok"); + expected.alloc += 1; + expected.assert_current(); + + drop(res); + expected.release += 1; + expected.dealloc += 1; + expected.assert_current(); + + // Errors + let res = autoreleasepool(|_pool| { + let res: Result, Id> = + unsafe { msg_send_id![RcTestObject::class(), allocAndShouldError: true, error: _] }; + expected.alloc += 1; + expected.init += 1; + expected.autorelease += 1; + expected.retain += 1; + expected.assert_current(); + res.expect_err("not err") + }); + expected.release += 1; + expected.assert_current(); + + drop(res); + expected.release += 1; + expected.dealloc += 1; + expected.assert_current(); + } } diff --git a/objc2/tests/use_macros.rs b/objc2/tests/use_macros.rs index 3f7c6418d..cb1fdf6f3 100644 --- a/objc2/tests/use_macros.rs +++ b/objc2/tests/use_macros.rs @@ -32,7 +32,6 @@ fn test_msg_send_comma_handling(obj: &NSString, superclass: &Class) { let _: () = msg_send![obj, a: 32i32]; let _: () = msg_send![obj, a: 32i32,]; let _: () = msg_send![obj, a: 32i32 b: 32i32]; - let _: () = msg_send![obj, a: 32i32 b: 32i32,]; let _: () = msg_send![obj, a: 32i32, b: 32i32]; let _: () = msg_send![obj, a: 32i32, b: 32i32,]; } @@ -43,7 +42,6 @@ fn test_msg_send_comma_handling(obj: &NSString, superclass: &Class) { let _: () = msg_send![super(obj, superclass), a: 32i32]; let _: () = msg_send![super(obj, superclass), a: 32i32,]; let _: () = msg_send![super(obj, superclass), a: 32i32 b: 32i32]; - let _: () = msg_send![super(obj, superclass), a: 32i32 b: 32i32,]; let _: () = msg_send![super(obj, superclass), a: 32i32, b: 32i32]; let _: () = msg_send![super(obj, superclass), a: 32i32, b: 32i32,]; } diff --git a/test-ui/ui/extern_methods_wrong_arguments.rs b/test-ui/ui/extern_methods_wrong_arguments.rs index 39c1df128..50379a8a6 100644 --- a/test-ui/ui/extern_methods_wrong_arguments.rs +++ b/test-ui/ui/extern_methods_wrong_arguments.rs @@ -30,6 +30,13 @@ extern_methods!( } ); +extern_methods!( + unsafe impl MyObject { + #[sel(f:g:)] + fn f(); + } +); + extern_methods!( unsafe impl MyObject { #[sel(x:)] diff --git a/test-ui/ui/extern_methods_wrong_arguments.stderr b/test-ui/ui/extern_methods_wrong_arguments.stderr index 48b8121b8..47339e0a8 100644 --- a/test-ui/ui/extern_methods_wrong_arguments.stderr +++ b/test-ui/ui/extern_methods_wrong_arguments.stderr @@ -3,8 +3,8 @@ error: Number of arguments in function and selector did not match! | | / extern_methods!( | | unsafe impl MyObject { - | | #[sel(a:)] - | | fn a(); + | | #[sel(b)] + | | fn b(arg: i32); | | } | | ); | |_^ @@ -16,8 +16,8 @@ error: Number of arguments in function and selector did not match! | | / extern_methods!( | | unsafe impl MyObject { - | | #[sel(b)] - | | fn b(arg: i32); + | | #[sel(f:g:)] + | | fn f(); | | } | | ); | |_^ @@ -29,36 +29,64 @@ error: Number of arguments in function and selector did not match! | | / extern_methods!( | | unsafe impl MyObject { - | | #[sel(c:d:e:)] - | | fn c(arg1: i32, arg2: u32); + | | #[sel(y)] + | | fn y(&self, arg: i32); | | } | | ); | |_^ | = note: this error originates in the macro `$crate::__collect_msg_send` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info) -error: Number of arguments in function and selector did not match! +error[E0308]: mismatched types --> ui/extern_methods_wrong_arguments.rs | | / extern_methods!( | | unsafe impl MyObject { - | | #[sel(x:)] - | | fn x(&self); + | | #[sel(a:)] + | | fn a(); | | } | | ); - | |_^ + | | ^ + | | | + | |_expected `()`, found enum `Result` + | expected `()` because of default return type | - = note: this error originates in the macro `$crate::__collect_msg_send` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: expected unit type `()` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info) -error: Number of arguments in function and selector did not match! +error[E0308]: mismatched types --> ui/extern_methods_wrong_arguments.rs | | / extern_methods!( | | unsafe impl MyObject { - | | #[sel(y)] - | | fn y(&self, arg: i32); + | | #[sel(c:d:e:)] + | | fn c(arg1: i32, arg2: u32); | | } | | ); - | |_^ + | | ^ + | | | + | |_expected `()`, found enum `Result` + | expected `()` because of default return type | - = note: this error originates in the macro `$crate::__collect_msg_send` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: expected unit type `()` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0308]: mismatched types + --> ui/extern_methods_wrong_arguments.rs + | + | / extern_methods!( + | | unsafe impl MyObject { + | | #[sel(x:)] + | | fn x(&self); + | | } + | | ); + | | ^ + | | | + | |_expected `()`, found enum `Result` + | expected `()` because of default return type + | + = note: expected unit type `()` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info) diff --git a/test-ui/ui/invalid_msg_send.rs b/test-ui/ui/invalid_msg_send.rs index 21e493c3c..7cb244783 100644 --- a/test-ui/ui/invalid_msg_send.rs +++ b/test-ui/ui/invalid_msg_send.rs @@ -1,5 +1,6 @@ //! Test invalid msg_send syntax use objc2::msg_send; +use objc2::rc::{Id, Shared}; use objc2::runtime::Object; fn main() { @@ -13,4 +14,7 @@ fn main() { let _: () = unsafe { msg_send![obj, a: b: c] }; let _: () = unsafe { msg_send![obj, a: b, c d] }; let _: () = unsafe { msg_send![obj, a: b: c] }; + let _: () = unsafe { msg_send![obj, a: b c: d,] }; + + let _: Result<(), Id> = unsafe { msg_send![obj, a: _, b: _] }; } diff --git a/test-ui/ui/invalid_msg_send.stderr b/test-ui/ui/invalid_msg_send.stderr index 3a9e1b22e..d038741f9 100644 --- a/test-ui/ui/invalid_msg_send.stderr +++ b/test-ui/ui/invalid_msg_send.stderr @@ -10,23 +10,41 @@ error: unexpected end of macro invocation | let _: () = unsafe { msg_send![obj,] }; | ^ missing tokens in macro arguments -error: unexpected end of macro invocation +error: no rules expected the token `)` --> ui/invalid_msg_send.rs | | let _: () = unsafe { msg_send![obj, a:] }; - | ^ missing tokens in macro arguments + | ^^^^^^^^^^^^^^^^^^ no rules expected this token in macro call + | + = note: this error originates in the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) -error: unexpected end of macro invocation +error: no rules expected the token `)` --> ui/invalid_msg_send.rs | | let _: () = unsafe { msg_send![obj, a: b c] }; - | ^ missing tokens in macro arguments + | ^^^^^^^^^^^^^^^^^^^^^^ no rules expected this token in macro call + | + = note: this error originates in the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) -error: no rules expected the token `d` +error: no rules expected the token `a` --> ui/invalid_msg_send.rs | | let _: () = unsafe { msg_send![obj, a: b, c d] }; - | ^ no rules expected this token in macro call + | ^^^^^^^^^^^^^^^^^^^^^^^^^ no rules expected this token in macro call + | + = note: this error originates in the macro `$crate::__msg_send_parse` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error: no rules expected the token `,` + --> ui/invalid_msg_send.rs + | + | let _: () = unsafe { msg_send![obj, a: b c: d,] }; + | ^ no rules expected this token in macro call + +error: no rules expected the token `b` + --> ui/invalid_msg_send.rs + | + | let _: Result<(), Id> = unsafe { msg_send![obj, a: _, b: _] }; + | ^ no rules expected this token in macro call error[E0412]: cannot find type `c` in this scope --> ui/invalid_msg_send.rs @@ -41,4 +59,7 @@ error[E0412]: cannot find type `c` in this scope --> ui/invalid_msg_send.rs | | let _: () = unsafe { msg_send![obj, a: b: c] }; - | ^ expecting a type here because of type ascription + | ^ + | | + | not found in this scope + | help: maybe you meant to write an assignment here: `let c` diff --git a/test-ui/ui/msg_send_id_invalid_return.stderr b/test-ui/ui/msg_send_id_invalid_return.stderr index 3eb7d0e29..e5bc0c6c9 100644 --- a/test-ui/ui/msg_send_id_invalid_return.stderr +++ b/test-ui/ui/msg_send_id_invalid_return.stderr @@ -191,7 +191,7 @@ note: required by a bound in `send_message_id` | | unsafe fn send_message_id>( | ^^^^^^^^^^^^^^^^^^^^^^ required by this bound in `send_message_id` - = note: this error originates in the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) error[E0277]: the trait bound `&objc2::runtime::Object: MaybeUnwrap` is not satisfied --> ui/msg_send_id_invalid_return.rs @@ -209,7 +209,7 @@ note: required by a bound in `send_message_id` | | unsafe fn send_message_id>( | ^^^^^^^^^^^^^^^^^^^^^^ required by this bound in `send_message_id` - = note: this error originates in the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) error[E0277]: the trait bound `Option<&objc2::runtime::Object>: MaybeUnwrap` is not satisfied --> ui/msg_send_id_invalid_return.rs @@ -225,4 +225,4 @@ note: required by a bound in `send_message_id` | | unsafe fn send_message_id>( | ^^^^^^^^^^^^^^^^^^^^^^ required by this bound in `send_message_id` - = note: this error originates in the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) diff --git a/test-ui/ui/msg_send_id_underspecified.stderr b/test-ui/ui/msg_send_id_underspecified.stderr index c1fa4056a..bbbdd20da 100644 --- a/test-ui/ui/msg_send_id_underspecified.stderr +++ b/test-ui/ui/msg_send_id_underspecified.stderr @@ -4,7 +4,7 @@ error[E0282]: type annotations needed | let _: &Object = &*unsafe { msg_send_id![obj, description] }; | ----------^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^-- type must be known at this point | - = note: this error originates in the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) help: consider giving `result` an explicit type --> $WORKSPACE/objc2/src/macros.rs | diff --git a/test-ui/ui/msg_send_invalid_error.rs b/test-ui/ui/msg_send_invalid_error.rs new file mode 100644 index 000000000..0ee6b0470 --- /dev/null +++ b/test-ui/ui/msg_send_invalid_error.rs @@ -0,0 +1,21 @@ +//! Test that msg_send! error handling works correctly. +use objc2::{msg_send, msg_send_id}; +use objc2::ClassType; +use objc2::rc::{Id, Shared}; +use objc2::foundation::NSString; + +fn main() { + let obj: &NSString; + + // Wrong type + let _: () = unsafe { msg_send![obj, a: _] }; + let _: Result = unsafe { msg_send![obj, b: _] }; + let _: Result<(), i32> = unsafe { msg_send![obj, c: _] }; + let _: Result<(), Id> = unsafe { msg_send![obj, d: _] }; + + // Different calls + let _: () = unsafe { msg_send![obj, e: obj, f: _] }; + let _: () = unsafe { msg_send![super(obj), g: _] }; + let _: () = unsafe { msg_send![super(obj, NSString::class()), h: _] }; + let _: () = unsafe { msg_send_id![obj, i: _] }; +} diff --git a/test-ui/ui/msg_send_invalid_error.stderr b/test-ui/ui/msg_send_invalid_error.stderr new file mode 100644 index 000000000..440f170f7 --- /dev/null +++ b/test-ui/ui/msg_send_invalid_error.stderr @@ -0,0 +1,97 @@ +error[E0308]: mismatched types + --> ui/msg_send_invalid_error.rs + | + | let _: () = unsafe { msg_send![obj, a: _] }; + | ^^^^^^^^^^^^^^^^^^^^ expected `()`, found enum `Result` + | + = note: expected unit type `()` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0308]: mismatched types + --> ui/msg_send_invalid_error.rs + | + | let _: Result = unsafe { msg_send![obj, b: _] }; + | ^^^^^^^^^^^^^^^^^^^^ expected `i32`, found `()` + | + = note: expected enum `Result` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) +help: try wrapping the expression in `Err` + --> $WORKSPACE/objc2/src/macros.rs + | + | Err(result) + | ++++ + + +error[E0308]: mismatched types + --> ui/msg_send_invalid_error.rs + | + | let _: Result<(), i32> = unsafe { msg_send![obj, c: _] }; + | ^^^^^^^^^^^^^^^^^^^^ expected `i32`, found struct `Id` + | + = note: expected enum `Result<_, i32>` + found enum `Result<_, Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0277]: the trait bound `i32: Message` is not satisfied + --> ui/msg_send_invalid_error.rs + | + | let _: Result<(), Id> = unsafe { msg_send![obj, d: _] }; + | ^^^^^^^^^^^^^^^^^^^^ the trait `Message` is not implemented for `i32` + | + = help: the following other types implement trait `Message`: + Exception + NSArray + NSAttributedString + NSBundle + NSData + NSDictionary + NSError + NSException + and $N others +note: required by a bound in `__send_message_error` + --> $WORKSPACE/objc2/src/message/mod.rs + | + | E: Message, + | ^^^^^^^ required by this bound in `__send_message_error` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0308]: mismatched types + --> ui/msg_send_invalid_error.rs + | + | let _: () = unsafe { msg_send![obj, e: obj, f: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected `()`, found enum `Result` + | + = note: expected unit type `()` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0308]: mismatched types + --> ui/msg_send_invalid_error.rs + | + | let _: () = unsafe { msg_send![super(obj), g: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected `()`, found enum `Result` + | + = note: expected unit type `()` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0308]: mismatched types + --> ui/msg_send_invalid_error.rs + | + | let _: () = unsafe { msg_send![super(obj, NSString::class()), h: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected `()`, found enum `Result` + | + = note: expected unit type `()` + found enum `Result<(), Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0308]: mismatched types + --> ui/msg_send_invalid_error.rs + | + | let _: () = unsafe { msg_send_id![obj, i: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^ expected `()`, found enum `Result` + | + = note: expected unit type `()` + found enum `Result, Id<_, Shared>>` + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) diff --git a/test-ui/ui/msg_send_mutable.stderr b/test-ui/ui/msg_send_mutable.stderr index cab18e9fb..21930eaf7 100644 --- a/test-ui/ui/msg_send_mutable.stderr +++ b/test-ui/ui/msg_send_mutable.stderr @@ -5,13 +5,7 @@ error[E0382]: use of moved value: `obj` | --- move occurs because `obj` has type `&mut objc2::runtime::Object`, which does not implement the `Copy` trait | | let _: () = unsafe { msg_send![obj, selector] }; - | ------------------------ `obj` moved due to this method call + | --- value moved here | // Could be solved with a reborrow | let _: () = unsafe { msg_send![obj, selector] }; | ^^^ value used here after move - | -note: this function takes ownership of the receiver `self`, which moves `obj` - --> $WORKSPACE/objc2/src/message/mod.rs - | - | unsafe fn send_message(self, sel: Sel, args: A) -> R - | ^^^^ diff --git a/test-ui/ui/msg_send_no_return_type.stderr b/test-ui/ui/msg_send_no_return_type.stderr index ffe6ed567..9bcd9a14b 100644 --- a/test-ui/ui/msg_send_no_return_type.stderr +++ b/test-ui/ui/msg_send_no_return_type.stderr @@ -4,7 +4,7 @@ error[E0282]: type annotations needed | msg_send![cls, new]; | ^^^^^^^^^^^^^^^^^^^ | - = note: this error originates in the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) help: consider giving `result` an explicit type --> $WORKSPACE/objc2/src/macros.rs | diff --git a/test-ui/ui/msg_send_not_encode.stderr b/test-ui/ui/msg_send_not_encode.stderr index c549e346b..9b4e497f4 100644 --- a/test-ui/ui/msg_send_not_encode.stderr +++ b/test-ui/ui/msg_send_not_encode.stderr @@ -20,7 +20,7 @@ note: required by a bound in `send_message` | | R: EncodeConvert, | ^^^^^^^^^^^^^ required by this bound in `send_message` - = note: this error originates in the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) error[E0277]: the trait bound `Vec: Encode` is not satisfied --> ui/msg_send_not_encode.rs @@ -48,4 +48,4 @@ note: required by a bound in `send_message` | | A: MessageArguments, | ^^^^^^^^^^^^^^^^ required by this bound in `send_message` - = note: this error originates in the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) diff --git a/test-ui/ui/msg_send_underspecified_error.rs b/test-ui/ui/msg_send_underspecified_error.rs new file mode 100644 index 000000000..0af2a1045 --- /dev/null +++ b/test-ui/ui/msg_send_underspecified_error.rs @@ -0,0 +1,13 @@ +//! Test underspecified msg_send! errors. +use objc2::{msg_send, msg_send_id}; +use objc2::rc::{Id, Shared}; +use objc2::foundation::NSString; + +fn main() { + let obj: &NSString; + let _: Result<(), _> = unsafe { msg_send![obj, a: _] }; + + let _: Result<_, _> = unsafe { msg_send_id![obj, b: _] }; + let _: Result, _> = unsafe { msg_send_id![obj, c: _] }; + let _: Result, Id<_, Shared>> = unsafe { msg_send_id![obj, d: _] }; +} diff --git a/test-ui/ui/msg_send_underspecified_error.stderr b/test-ui/ui/msg_send_underspecified_error.stderr new file mode 100644 index 000000000..e852982ca --- /dev/null +++ b/test-ui/ui/msg_send_underspecified_error.stderr @@ -0,0 +1,84 @@ +error[E0282]: type annotations needed + --> ui/msg_send_underspecified_error.rs + | + | let _: Result<(), _> = unsafe { msg_send![obj, a: _] }; + | ^^^^^^^^^^^^^^^^^^^^ cannot infer type of the type parameter `E` declared on the associated function `__send_message_error` + | + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) +help: consider specifying the generic arguments + --> $WORKSPACE/objc2/src/macros.rs + | + | @(__send_message_error::<(), E>) + | +++++++++ + +error[E0283]: type annotations needed + --> ui/msg_send_underspecified_error.rs + | + | let _: Result<(), _> = unsafe { msg_send![obj, a: _] }; + | ^^^^^^^^^^^^^^^^^^^^ cannot infer type for raw pointer `*mut _` + | + = note: multiple `impl`s satisfying `*mut _: RefEncode` found in the `objc2_encode` crate: + - impl RefEncode for *mut c_void; + - impl RefEncode for *mut T + where T: RefEncode, T: ?Sized; + = note: required for `*mut *mut _` to implement `Encode` +note: required by a bound in `__send_message_error` + --> $WORKSPACE/objc2/src/message/mod.rs + | + | *mut *mut E: Encode, + | ^^^^^^ required by this bound in `__send_message_error` + = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0283]: type annotations needed + --> ui/msg_send_underspecified_error.rs + | + | let _: Result<_, _> = unsafe { msg_send_id![obj, b: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^ cannot infer type for raw pointer `*mut _` + | + = note: multiple `impl`s satisfying `*mut _: RefEncode` found in the `objc2_encode` crate: + - impl RefEncode for *mut c_void; + - impl RefEncode for *mut T + where T: RefEncode, T: ?Sized; + = note: required for `*mut *mut _` to implement `Encode` +note: required by a bound in `send_message_id_error` + --> $WORKSPACE/objc2/src/__macro_helpers.rs + | + | *mut *mut E: Encode, + | ^^^^^^ required by this bound in `send_message_id_error` + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0283]: type annotations needed + --> ui/msg_send_underspecified_error.rs + | + | let _: Result, _> = unsafe { msg_send_id![obj, c: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^ cannot infer type for raw pointer `*mut _` + | + = note: multiple `impl`s satisfying `*mut _: RefEncode` found in the `objc2_encode` crate: + - impl RefEncode for *mut c_void; + - impl RefEncode for *mut T + where T: RefEncode, T: ?Sized; + = note: required for `*mut *mut _` to implement `Encode` +note: required by a bound in `send_message_id_error` + --> $WORKSPACE/objc2/src/__macro_helpers.rs + | + | *mut *mut E: Encode, + | ^^^^^^ required by this bound in `send_message_id_error` + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) + +error[E0283]: type annotations needed + --> ui/msg_send_underspecified_error.rs + | + | let _: Result, Id<_, Shared>> = unsafe { msg_send_id![obj, d: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^ cannot infer type for raw pointer `*mut _` + | + = note: multiple `impl`s satisfying `*mut _: RefEncode` found in the `objc2_encode` crate: + - impl RefEncode for *mut c_void; + - impl RefEncode for *mut T + where T: RefEncode, T: ?Sized; + = note: required for `*mut *mut _` to implement `Encode` +note: required by a bound in `send_message_id_error` + --> $WORKSPACE/objc2/src/__macro_helpers.rs + | + | *mut *mut E: Encode, + | ^^^^^^ required by this bound in `send_message_id_error` + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) diff --git a/test-ui/ui/msg_send_underspecified_error2.rs b/test-ui/ui/msg_send_underspecified_error2.rs new file mode 100644 index 000000000..ab9548232 --- /dev/null +++ b/test-ui/ui/msg_send_underspecified_error2.rs @@ -0,0 +1,9 @@ +//! Test underspecified msg_send! errors. +use objc2::msg_send_id; +use objc2::rc::{Id, Shared}; +use objc2::foundation::{NSError, NSString}; + +fn main() { + let obj: &NSString; + let _: Result, Id> = unsafe { msg_send_id![obj, a: _] }; +} diff --git a/test-ui/ui/msg_send_underspecified_error2.stderr b/test-ui/ui/msg_send_underspecified_error2.stderr new file mode 100644 index 000000000..5f0134a80 --- /dev/null +++ b/test-ui/ui/msg_send_underspecified_error2.stderr @@ -0,0 +1,7 @@ +error[E0282]: type annotations needed + --> ui/msg_send_underspecified_error2.rs + | + | let _: Result, Id> = unsafe { msg_send_id![obj, a: _] }; + | ^^^^^^^^^^^^^^^^^^^^^^^ cannot infer type of the type parameter `T` declared on the trait `MsgSendId` + | + = note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `msg_send_id` (in Nightly builds, run with -Z macro-backtrace for more info) From a3faaeaa19cb263b63869b69209c8b71e091dd07 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Thu, 27 Oct 2022 14:33:51 +0200 Subject: [PATCH 4/5] Add NSString::write_to_file --- objc2/CHANGELOG_FOUNDATION.md | 1 + objc2/src/foundation/string.rs | 11 ++++++++++- 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/objc2/CHANGELOG_FOUNDATION.md b/objc2/CHANGELOG_FOUNDATION.md index 7f6d52269..2068eb094 100644 --- a/objc2/CHANGELOG_FOUNDATION.md +++ b/objc2/CHANGELOG_FOUNDATION.md @@ -21,6 +21,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). * Added `NSString::concat` and `NSString::join_path`. * Added `CGSize`, `CGPoint` and `CGRect` (just aliases to equivalent `NS`-types, but helps readability). +* Added `NSString::write_to_file`. ### Changed * **BREAKING**: `NSSize::new` no longer requires it's arguments to be diff --git a/objc2/src/foundation/string.rs b/objc2/src/foundation/string.rs index 0c0b4d982..d9b83e323 100644 --- a/objc2/src/foundation/string.rs +++ b/objc2/src/foundation/string.rs @@ -9,7 +9,7 @@ use core::slice; use core::str; use std::os::raw::c_char; -use super::{NSComparisonResult, NSCopying, NSMutableCopying, NSMutableString, NSObject}; +use super::{NSComparisonResult, NSCopying, NSError, NSMutableCopying, NSMutableString, NSObject}; use crate::rc::{autoreleasepool, AutoreleasePool, DefaultId, Id, Shared}; use crate::runtime::{Class, Object}; use crate::{extern_class, extern_methods, msg_send, msg_send_id, ClassType}; @@ -272,6 +272,15 @@ extern_methods!( // pub fn from_nsrange(range: NSRange) -> Id // https://developer.apple.com/documentation/foundation/1415155-nsstringfromrange?language=objc + + // TODO: Safety + #[sel(writeToFile:atomically:encoding:error:)] + pub unsafe fn write_to_file( + &self, + path: &NSString, + atomically: bool, + encoding: usize, + ) -> Result<(), Id>; } ); From 88d1b836cfea46e1064b3be7124c8987f9e559c5 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Fri, 28 Oct 2022 15:54:28 +0200 Subject: [PATCH 5/5] Slightly improve msg_send! error UI --- objc2/src/macros.rs | 6 ++++-- test-ui/ui/msg_send_underspecified_error.stderr | 5 ----- 2 files changed, 4 insertions(+), 7 deletions(-) diff --git a/objc2/src/macros.rs b/objc2/src/macros.rs index b01ff1963..0f81c0cc9 100644 --- a/objc2/src/macros.rs +++ b/objc2/src/macros.rs @@ -876,7 +876,9 @@ macro_rules! __msg_send_helper { let result; // Always add trailing comma after each argument, so that we get a // 1-tuple if there is only one. - result = $crate::MessageReceiver::$fn($($fn_args)+, $crate::sel!($($selector)*), ($($argument,)*)); + // + // And use `::<_, _>` for better UI + result = $crate::MessageReceiver::$fn::<_, _>($($fn_args)+, $crate::sel!($($selector)*), ($($argument,)*)); result }); } @@ -1135,7 +1137,7 @@ macro_rules! __msg_send_id_helper { let result; result = <$crate::__macro_helpers::RetainSemantics<{ $crate::__macro_helpers::retain_semantics(__SELECTOR_DATA) - }> as $crate::__macro_helpers::MsgSendId<_, _>>::$fn( + }> as $crate::__macro_helpers::MsgSendId<_, _>>::$fn::<_, _>( $obj, $crate::__sel_inner!( __SELECTOR_DATA, diff --git a/test-ui/ui/msg_send_underspecified_error.stderr b/test-ui/ui/msg_send_underspecified_error.stderr index e852982ca..cd33ae828 100644 --- a/test-ui/ui/msg_send_underspecified_error.stderr +++ b/test-ui/ui/msg_send_underspecified_error.stderr @@ -5,11 +5,6 @@ error[E0282]: type annotations needed | ^^^^^^^^^^^^^^^^^^^^ cannot infer type of the type parameter `E` declared on the associated function `__send_message_error` | = note: this error originates in the macro `$crate::__msg_send_helper` which comes from the expansion of the macro `msg_send` (in Nightly builds, run with -Z macro-backtrace for more info) -help: consider specifying the generic arguments - --> $WORKSPACE/objc2/src/macros.rs - | - | @(__send_message_error::<(), E>) - | +++++++++ error[E0283]: type annotations needed --> ui/msg_send_underspecified_error.rs