bon/__/cfg_eval.rs
1/// This is all a big embarrassing workaround, please don't oversee 😳😳😳.
2///
3/// Anyway, if you are curious what the hell is going on here, then here is
4/// an explanation 😸. So... where to start 🤔. Ah! The problem!
5///
6/// ## The problem
7///
8/// Proc macro attributes (like `#[builder]`) see all the `#[cfg(...)]` and `#[cfg_attr(...)]`
9/// attributes unexpanded. For example, if you write smth like this:
10///
11/// ```
12/// #[bon::builder]
13/// fn func(
14/// #[cfg(windows)]
15/// windows_only_param: u32,
16/// ) {}
17///
18/// ```
19///
20/// then the `#[builder]` macro will see the full `#[cfg(...)]` attribute with
21/// the `windows_only_param` it is attached to verbatim. The `#[cfg(...)]` isn't
22/// removed by the time the `#[builder]`'s macro expansion is invoked.
23///
24/// It is a problem because the `#[builder]` macro needs to know the exact list
25/// of members it has to generate setters for. It doesn't know whether
26/// the `windows` predicate evaluates to `true` or `false`, especially if this was
27/// a more complex predicate. So it can't decide whether to generate a setter for
28/// the `windows_only_param` or not.
29///
30/// ## The solution
31///
32/// This macro allows us to evaluate the `cfg` predicates by using a variation of
33/// [the trick] shared by @recatek.
34///
35/// When the `#[builder]` macro finds any usage of `#[cfg(...)]` or `#[cfg_attr(...)]`
36/// it generates a call to this macro with all `cfg` predicates collected from the
37/// item it was placed on. The `#[builder]` macro deduplicates and sorts the `cfg`
38/// predicates and passes them as `$pred` to this macro.
39///
40/// This macro then dispatches to `__eval_cfg_callback_true` or `__eval_cfg_callback_false`
41/// by defining a conditional `use ...` statement for each predicate and collects the
42/// results of the evaluation in the `$results` list.
43///
44/// For the last call to this macro (when no more `$pred` are left) the macro calls back
45/// to the proc macro attribute that called it with the results of the evaluation and
46/// the original parameters and the item which are passed through via the `$rest` macro variable.
47///
48/// [the trick]: https://users.rust-lang.org/t/supporting-or-evaluating-cfg-in-proc-macro-parameters/93240/2
49#[macro_export]
50#[doc(hidden)]
51macro_rules! __eval_cfg_callback {
52 (
53 { $($results:tt)* }
54 ( $pred_id:ident: $($pred:tt)* )
55 $($rest:tt)*
56 ) => {
57 // The `pred_id` is required to be a unique identifier for the current
58 // predicate evaluation so that we can use it in a `use` statement to define
59 // a new unique name for the macro to call.
60 #[cfg($($pred)*)]
61 #[doc(hidden)]
62 #[allow(deprecated)]
63 use $crate::__eval_cfg_callback_true as $pred_id;
64
65 #[cfg(not($($pred)*))]
66 #[doc(hidden)]
67 #[allow(deprecated)]
68 use $crate::__eval_cfg_callback_false as $pred_id;
69
70 // The trick here is that `$pred_id` now resolves either to
71 // `__eval_cfg_callback_true` or `__eval_cfg_callback_false`
72 // depending on the evaluation of the cfg predicate, so by
73 // invoking it as a macro, that macro internally pushes either
74 // `true` or `false` to the `$results` list.
75 $pred_id! {
76 { $($results)* }
77 $($rest)*
78 }
79 };
80
81 // The terminal case for the recursion when there are no more predicates left.
82 // We have collected all the results of the cfg evaluations and now we can
83 // delegate them to the proc macro attribute that called this macro.
84 (
85 // The results of the cfg evaluation
86 { $($results:tt)* }
87
88 // The proc macro attribute to invoke with the results
89 $final_macro:path,
90
91 // The number of times this macro was called recursively from the proc macro
92 $recursion_counter:literal,
93
94 // Parameters to pass to the proc macro attribute after the cfg results
95 ( $($macro_params:tt)* )
96
97 // The item to attach the proc macro attribute to
98 $($item:tt)*
99 ) => {
100 // The special `__cfgs(...)` prefix is parsed by the proc macro attribute
101 // to get the results of the cfg evaluations.
102 #[$final_macro(__cfgs($recursion_counter, $($results)*) $($macro_params)*)]
103 $($item)*
104 };
105}
106
107/// The `cfg` predicate evaluated to `true`, now push that information into
108/// the `$results` list.
109#[macro_export]
110#[doc(hidden)]
111macro_rules! __eval_cfg_callback_true {
112 (
113 { $($results:tt)* }
114 $($tt:tt)*
115 ) => {
116 $crate::__eval_cfg_callback! {
117 { $($results)* true, }
118 $($tt)*
119 }
120 };
121}
122
123/// The `cfg` predicate evaluated to `false`, now push that information into
124/// the `$results` list.
125#[macro_export]
126#[doc(hidden)]
127macro_rules! __eval_cfg_callback_false {
128 (
129 { $($results:tt)* }
130 $($tt:tt)*
131 ) => {
132 $crate::__eval_cfg_callback! {
133 { $($results)* false, }
134 $($tt)*
135 }
136 };
137}