Skip to content

Commit 7520d05

Browse files
committed
Revise after splitting rust-lang#8 for match arm attributes (now landed).
1 parent fed1f9f commit 7520d05

File tree

1 file changed

+59
-87
lines changed

1 file changed

+59
-87
lines changed

active/0000-more-attributes.md

+59-87
Original file line numberDiff line numberDiff line change
@@ -4,15 +4,14 @@
44

55
# Summary
66

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.
99

1010
# Motivation
1111

1212
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.
1615

1716
For the lints, one can currently only activate lints at the level of
1817
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
2423
```
2524

2625
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.
7529

7630
The sort of things one could do with other arbitrary annotations are
7731

@@ -85,59 +39,77 @@ and then have an external tool that checks that that `unsafe` block's
8539
only unsafe actions are FFI, or a tool that lists blocks that have
8640
been changed since the last audit or haven't been audited ever.
8741

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.
8845

8946
# Detailed design
9047

91-
Normal attribute syntax:
48+
Normal attribute syntax on `let` statements and blocks.
9249

9350
```rust
9451
fn foo() {
95-
#[attr]
52+
#[attr1]
9653
let x = 1;
9754

98-
#[attr]
99-
foo();
55+
#[attr2]
56+
{
57+
// code
58+
}
10059

101-
#[attr]
102-
match x {
103-
#[attr]
104-
Thing => {}
60+
#[attr3]
61+
unsafe {
62+
// code
10563
}
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();
10676

107-
#[attr]
108-
if foo {
109-
} else {
77+
#[attr5] if cond {
78+
bar()
79+
} else #[attr6] {
80+
baz()
11081
}
82+
83+
let x = #[attr7] 1;
84+
85+
qux(3 + #[attr8] 2);
86+
87+
foo(x, #[attr9] y, z);
11188
}
11289
```
11390

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`.
11598

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)`.
120103

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
123108
general `unsafe` block).
124109

125110
# Unresolved questions
126111

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?
137114

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

Comments
 (0)