Skip to content

Commit ce6100c

Browse files
authored
Merge pull request #416 from madsmtm/mainthreadmarker-in-extern-methods
Allow using `MainThreadMarker` in `extern_methods!`
2 parents 282188d + 962e7a5 commit ce6100c

8 files changed

+200
-0
lines changed

crates/objc2/CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,9 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
66

77
## Unreleased - YYYY-MM-DD
88

9+
### Added
10+
* Allow using `MainThreadMarker` in `extern_methods!`.
11+
912
### Changed
1013
* Renamed `runtime` types:
1114
- `Object` to `AnyObject`.

crates/objc2/src/macros/__method_msg_send.rs

+50
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,30 @@ macro_rules! __method_msg_send {
2222
}
2323
};
2424

25+
// Skip using `MainThreadMarker` in the message send.
26+
//
27+
// This is a purely textual match, and using e.g.
28+
// `Foundation::MainThreadMarker` would fail - but that would just be
29+
// detected as giving a wrong number of arguments, so it's fine for now.
30+
(
31+
($receiver:expr)
32+
($($sel_rest:tt)*)
33+
($arg:ident: MainThreadMarker $(, $($args_rest:tt)*)?)
34+
35+
($($sel_parsed:tt)*)
36+
($($arg_parsed:tt)*)
37+
) => ({
38+
let _ = $arg;
39+
$crate::__method_msg_send! {
40+
($receiver)
41+
($($sel_rest)*)
42+
($($($args_rest)*)?)
43+
44+
($($sel_parsed)*)
45+
($($arg_parsed)*)
46+
}
47+
});
48+
2549
// Parse each argument-selector pair
2650
(
2751
($receiver:expr)
@@ -173,6 +197,32 @@ macro_rules! __method_msg_send_id {
173197
}
174198
};
175199

200+
// Skip using `MainThreadMarker` in the message send.
201+
//
202+
// This is a purely textual match, and using e.g.
203+
// `Foundation::MainThreadMarker` would fail - but that would just be
204+
// detected as giving a wrong number of arguments, so it's fine for now.
205+
(
206+
($receiver:expr)
207+
($($sel_rest:tt)*)
208+
($arg:ident: MainThreadMarker $(, $($args_rest:tt)*)?)
209+
210+
($($sel_parsed:tt)*)
211+
($($arg_parsed:tt)*)
212+
($($retain_semantics:ident)?)
213+
) => ({
214+
let _ = $arg;
215+
$crate::__method_msg_send_id! {
216+
($receiver)
217+
($($sel_rest)*)
218+
($($($args_rest)*)?)
219+
220+
($($sel_parsed)*)
221+
($($arg_parsed)*)
222+
($($retain_semantics)?)
223+
}
224+
});
225+
176226
// Parse each argument-selector pair
177227
(
178228
($receiver:expr)

crates/objc2/src/macros/extern_methods.rs

+6
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,12 @@
2626
/// [`Result`]. See the error section in [`msg_send!`] and [`msg_send_id!`]
2727
/// for details.
2828
///
29+
/// If you use `icrate::Foundation::MainThreadMarker` as a parameter type, the
30+
/// macro will ignore it, allowing you to neatly specify "this method must be
31+
/// run on the main thread". Note that due to type-system limitations, this is
32+
/// currently a textual match on `MainThreadMarker`; so you must use that
33+
/// exact identifier.
34+
///
2935
/// Putting other attributes on the method such as `cfg`, `allow`, `doc`,
3036
/// `deprecated` and so on is supported. However, note that `cfg_attr` may not
3137
/// work correctly, due to implementation difficulty - if you have a concrete
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
use objc2::rc::Id;
2+
use objc2::runtime::{NSObject, NSObjectProtocol};
3+
use objc2::{declare_class, extern_methods, extern_protocol, mutability, ClassType, ProtocolType};
4+
5+
extern_protocol!(
6+
#[allow(clippy::missing_safety_doc)]
7+
unsafe trait Proto: NSObjectProtocol {
8+
#[method(myMethod:)]
9+
fn protocol_method(mtm: MainThreadMarker, arg: i32) -> i32;
10+
11+
#[method_id(myMethodId:)]
12+
fn protocol_method_id(mtm: MainThreadMarker, arg: &Self) -> Id<Self>;
13+
}
14+
15+
unsafe impl ProtocolType for dyn Proto {
16+
const NAME: &'static str = "MainThreadMarkerTestProtocol";
17+
}
18+
);
19+
20+
declare_class!(
21+
#[derive(PartialEq, Eq, Hash, Debug)]
22+
struct Cls;
23+
24+
unsafe impl ClassType for Cls {
25+
type Super = NSObject;
26+
type Mutability = mutability::InteriorMutable;
27+
const NAME: &'static str = "MainThreadMarkerTest";
28+
}
29+
30+
unsafe impl Proto for Cls {
31+
#[method(myMethod:)]
32+
fn _my_mainthreadonly_method(arg: i32) -> i32 {
33+
arg + 1
34+
}
35+
36+
#[method_id(myMethodId:)]
37+
fn _my_mainthreadonly_method_id(arg: &Self) -> Id<Self> {
38+
unsafe { Id::retain(arg as *const Self as *mut Self).unwrap() }
39+
}
40+
}
41+
);
42+
43+
unsafe impl NSObjectProtocol for Cls {}
44+
45+
// The macro does a textual match; but when users actually use
46+
// `icrate::Foundation::MainThreadMarker` to ensure soundness, they will not
47+
// do this!
48+
#[derive(Clone, Copy)]
49+
struct MainThreadMarker(bool);
50+
51+
extern_methods!(
52+
unsafe impl Cls {
53+
#[method_id(new)]
54+
fn new() -> Id<Self>;
55+
56+
#[method(myMethod:)]
57+
fn method(mtm: MainThreadMarker, arg: i32, mtm2: MainThreadMarker) -> i32;
58+
59+
#[method_id(myMethodId:)]
60+
fn method_id(mtm: MainThreadMarker, arg: &Self, mtm2: MainThreadMarker) -> Id<Self>;
61+
}
62+
);
63+
64+
#[test]
65+
fn call() {
66+
let obj1 = Cls::new();
67+
let mtm = MainThreadMarker(true);
68+
69+
let res = Cls::method(mtm, 2, mtm);
70+
assert_eq!(res, 3);
71+
let res = Cls::protocol_method(mtm, 3);
72+
assert_eq!(res, 4);
73+
74+
let obj2 = Cls::method_id(mtm, &obj1, mtm);
75+
assert_eq!(obj1, obj2);
76+
77+
let obj2 = Cls::protocol_method_id(mtm, &obj1);
78+
assert_eq!(obj1, obj2);
79+
}

crates/test-ui/ui/extern_methods_invalid_type.rs

+8
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use icrate::Foundation::MainThreadMarker;
12
use objc2::rc::Id;
23
use objc2::runtime::NSObject;
34
use objc2::{extern_class, extern_methods, mutability, ClassType};
@@ -46,4 +47,11 @@ extern_methods!(
4647
}
4748
);
4849

50+
extern_methods!(
51+
unsafe impl MyObject {
52+
#[method(mainThreadMarkerAsReturn)]
53+
fn main_thread_marker_as_return() -> MainThreadMarker;
54+
}
55+
);
56+
4957
fn main() {}

crates/test-ui/ui/extern_methods_invalid_type.stderr

+33
Original file line numberDiff line numberDiff line change
@@ -132,3 +132,36 @@ note: required by a bound in `send_message_id`
132132
| unsafe fn send_message_id<A: MessageArguments, R: MaybeUnwrap<Input = U>>(
133133
| ^^^^^^^^^^^^^^^^^^^^^^ required by this bound in `MsgSendId::send_message_id`
134134
= note: this error originates in the macro `$crate::__msg_send_id_helper` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info)
135+
136+
error[E0277]: the trait bound `MainThreadMarker: Encode` is not satisfied
137+
--> ui/extern_methods_invalid_type.rs
138+
|
139+
| / extern_methods!(
140+
| | unsafe impl MyObject {
141+
| | #[method(mainThreadMarkerAsReturn)]
142+
| | fn main_thread_marker_as_return() -> MainThreadMarker;
143+
| | }
144+
| | );
145+
| |_^ the trait `Encode` is not implemented for `MainThreadMarker`
146+
|
147+
= help: the following other types implement trait `Encode`:
148+
&'a T
149+
&'a mut T
150+
*const T
151+
*const c_void
152+
*mut T
153+
*mut c_void
154+
AtomicI16
155+
AtomicI32
156+
and $N others
157+
= note: required for `MainThreadMarker` to implement `EncodeReturn`
158+
= note: required for `MainThreadMarker` to implement `EncodeConvertReturn`
159+
note: required by a bound in `send_message`
160+
--> $WORKSPACE/crates/objc2/src/message/mod.rs
161+
|
162+
| unsafe fn send_message<A, R>(self, sel: Sel, args: A) -> R
163+
| ------------ required by a bound in this associated function
164+
...
165+
| R: EncodeConvertReturn,
166+
| ^^^^^^^^^^^^^^^^^^^ required by this bound in `MessageReceiver::send_message`
167+
= 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)

crates/test-ui/ui/extern_methods_wrong_arguments_error.rs

+8
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use icrate::Foundation::MainThreadMarker;
12
use objc2::rc::Id;
23
use objc2::runtime::NSObject;
34
use objc2::{extern_class, extern_methods, mutability, ClassType};
@@ -39,4 +40,11 @@ extern_methods!(
3940
}
4041
);
4142

43+
extern_methods!(
44+
unsafe impl MyObject {
45+
#[method(tooFew:withMtm:_)]
46+
fn too_few_with_mtm(&self, mtm: MainThreadMarker) -> Result<(), Id<NSObject>>;
47+
}
48+
);
49+
4250
fn main() {}

crates/test-ui/ui/extern_methods_wrong_arguments_error.stderr

+13
Original file line numberDiff line numberDiff line change
@@ -49,3 +49,16 @@ error: number of arguments in function and selector did not match
4949
| |_^
5050
|
5151
= note: this error originates in the macro `$crate::__method_msg_send` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info)
52+
53+
error: number of arguments in function and selector did not match
54+
--> ui/extern_methods_wrong_arguments_error.rs
55+
|
56+
| / extern_methods!(
57+
| | unsafe impl MyObject {
58+
| | #[method(tooFew:withMtm:_)]
59+
| | fn too_few_with_mtm(&self, mtm: MainThreadMarker) -> Result<(), Id<NSObject>>;
60+
| | }
61+
| | );
62+
| |_^
63+
|
64+
= note: this error originates in the macro `$crate::__method_msg_send` which comes from the expansion of the macro `extern_methods` (in Nightly builds, run with -Z macro-backtrace for more info)

0 commit comments

Comments
 (0)