4
4
5
5
# Summary
6
6
7
- Allow attributes on more places inside functions, such as match arms
8
- and statements .
7
+ Allow attributes on more places inside functions, such as statements,
8
+ blocks and (possibly) expressions .
9
9
10
10
# Motivation
11
11
12
12
One sometimes wishes to annotate things inside functions with, for
13
- example, lint ` #[allow] ` s, conditional compilation ` #[cfg] ` s, branch
14
- weight hints and even extra semantic (or otherwise) annotations for
15
- external tools.
13
+ example, lint ` #[allow] ` s, conditional compilation ` #[cfg] ` s, and even
14
+ extra semantic (or otherwise) annotations for external tools.
16
15
17
16
For the lints, one can currently only activate lints at the level of
18
17
the function which is possibly larger than one needs, and so may allow
@@ -24,54 +23,9 @@ let L = List::new(); // lowercase looks like one or capital i
24
23
```
25
24
26
25
For the conditional compilation, the work-around is duplicating the
27
- whole containing function with a ` #[cfg] ` . A case study is
28
- [ sfackler's bindings to OpenSSL] ( https://github.com/sfackler/rust-openssl ) ,
29
- where many distributions remove SSLv2 support, and so that portion of
30
- Rust bindings needs to be conditionally disabled. The obvious way to
31
- support the various different SSL versions is an enum
32
-
33
- ``` rust
34
- pub enum SslMethod {
35
- #[cfg(sslv2)]
36
- /// Only support the SSLv2 protocol
37
- Sslv2 ,
38
- /// Only support the SSLv3 protocol
39
- Sslv3 ,
40
- /// Only support the TLSv1 protocol
41
- Tlsv1 ,
42
- /// Support the SSLv2, SSLv3 and TLSv1 protocols
43
- Sslv23 ,
44
- }
45
- ```
46
-
47
- However, all ` match ` s can only mention ` Sslv2 ` when the ` cfg ` is
48
- active, i.e. the following is invalid:
49
-
50
- ``` rust
51
- fn name (method : SslMethod ) -> & 'static str {
52
- match method {
53
- Sslv2 => " SSLv2" ,
54
- Sslv3 => " SSLv3" ,
55
- _ => " ..."
56
- }
57
- }
58
- ```
59
-
60
- A valid method would be to have two definitions: `#[ cfg(sslv2)] fn
61
- name(...)` and ` #[ cfg(not(sslv2)] fn name(...)`. The former has the
62
- ` Sslv2 ` arm, the latter does not. Clearly, this explodes exponentially
63
- for each additional ` cfg ` 'd variant in an enum.
64
-
65
- Branch weights would allow the careful micro-optimiser to inform the
66
- compiler that, for example, a certain match arm is rarely taken:
67
-
68
- ``` rust
69
- match foo {
70
- Common => {}
71
- #[cold]
72
- Rare => {}
73
- }
74
- ```
26
+ whole containing function with a ` #[cfg] ` , or breaking the conditional
27
+ code into a its own function. This does mean that any variables need
28
+ to be explicitly passed as arguments.
75
29
76
30
The sort of things one could do with other arbitrary annotations are
77
31
@@ -85,59 +39,77 @@ and then have an external tool that checks that that `unsafe` block's
85
39
only unsafe actions are FFI, or a tool that lists blocks that have
86
40
been changed since the last audit or haven't been audited ever.
87
41
42
+ The minimum useful functionality would be supporting attributes on
43
+ blocks and ` let ` statements, since these are flexible enough to allow
44
+ for relatively precise attribute handling.
88
45
89
46
# Detailed design
90
47
91
- Normal attribute syntax:
48
+ Normal attribute syntax on ` let ` statements and blocks.
92
49
93
50
``` rust
94
51
fn foo () {
95
- #[attr ]
52
+ #[attr1 ]
96
53
let x = 1 ;
97
54
98
- #[attr]
99
- foo ();
55
+ #[attr2]
56
+ {
57
+ // code
58
+ }
100
59
101
- #[attr]
102
- match x {
103
- #[attr]
104
- Thing => {}
60
+ #[attr3]
61
+ unsafe {
62
+ // code
105
63
}
64
+ }
65
+ ```
66
+
67
+ ## Extension to arbitrary expressions
68
+
69
+ It would also be theoretically possible to extend this to support
70
+ arbitrary expressions (rather than just blocks, which are themselves
71
+ expressions). This would allow writing
72
+
73
+ ``` rust
74
+ fn foo () {
75
+ #[attr4] foo ();
106
76
107
- #[attr]
108
- if foo {
109
- } else {
77
+ #[attr5] if cond {
78
+ bar ()
79
+ } else #[attr6] {
80
+ baz ()
110
81
}
82
+
83
+ let x = #[attr7] 1 ;
84
+
85
+ qux (3 + #[attr8] 2 );
86
+
87
+ foo (x , #[attr9] y , z );
111
88
}
112
89
```
113
90
114
- # Alternatives
91
+ These last examples indicate a possible difficulty: what happens with
92
+ say ` 1 + #[cfg(foo)] 2 ` ? This should be an error, i.e. ` #[cfg] ` is
93
+ only allowed on the "exterior" of expressions, that is, legal in all
94
+ examples above except ` #[attr7] ` and ` #[attr8] ` . ` #[attr9] ` is
95
+ questionable: if it were a ` cfg ` , then it could reasonably be
96
+ interpreted as meaning ` foo(x, z) ` or ` foo(x, y, z) ` conditional on
97
+ the ` cfg ` .
115
98
116
- There aren't really any general alternatives; one could probably hack
117
- around the conditional-enum-variants & matches with some macros and
118
- helper functions to share as much code as possible; but in general
119
- this won't work .
99
+ Allowing attributes there would also require considering
100
+ precedence. There are two sensible options, ` #[...] ` binds tighter
101
+ than everything else, i.e. ` #[attr] 1 + 2 ` is ` (#[attr] 1) + 2 ` , or it
102
+ is weaker, so ` #[attr] 1 + 2 ` is ` #[attr] (1 + 2) ` .
120
103
121
- The other instances could be approximated with macros and helper
122
- functions, but to an even lesser degree (e.g. how would one annotate a
104
+ # Alternatives
105
+
106
+ These instances could possibly be approximated with macros and helper
107
+ functions, but to a low degree degree (e.g. how would one annotate a
123
108
general ` unsafe ` block).
124
109
125
110
# Unresolved questions
126
111
127
- - Should one be able to annotate the ` else ` branch(es) of an ` if ` ? e.g.
128
-
129
- ``` rust
130
- if foo {
131
- } #[attr] else if bar {
132
- } #[attr] else {
133
- }
134
- ```
135
-
136
- or maybe
112
+ - Are the complications of allowing attributes on arbitrary
113
+ expressions worth the benefits?
137
114
138
- ``` rust
139
- if foo {
140
- } else #[attr] if bar {
141
- } else #[attr] {
142
- }
143
- ```
115
+ - Which precedence should attributes have on arbitrary expressions?
0 commit comments