forked from villevoutilainen/papers
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfeature-macros.html
335 lines (316 loc) · 12.9 KB
/
feature-macros.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
"http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<title>Adopt SD-6 feature macros into the C++20 working draft</title>
<style>
p {text-align:justify}
li {text-align:justify}
blockquote.note
{
background-color:#E0E0E0;
padding-left: 15px;
padding-right: 15px;
padding-top: 1px;
padding-bottom: 1px;
}
ins {color:#00A000}
del {color:#A00000}
table {border: 1px solid black;}
th, td {border-left: 1px solid black}
td {border-top: 1px solid black}
td.heading {border-left-width: 0px}
</style>
</head>
<body>
<address align=right>
Document number: D????
<br/>
Audience: EWG, LEWG
<br/>
<br/>
<a href="mailto:[email protected]">Ville Voutilainen</a><br/>
2017-10-15<br/>
</address>
<hr/>
<h1 align=center>Adopt SD-6 feature macros into the C++20 working draft</h1>
<h2>Abstract</h2>
<p>
This paper proposes to adopt the SD-6 feature macros into the
C++20 working draft. Merging the feature macros removes
vendors' obstacles for implementing them, makes them more
portable and causes no delays in vendors tracking the macros
in their implementations.
This proposal is very much separate from what we should do with
SD-6 as far as C++17 goes. We should publish a revision of SD-6
covering C++17 features as we have published SD-6 thus far. This
proposal should not affect that publication in any way.
</p>
<h2>Goal</h2>
<p>
Feature test macros are so we can adopt the latest C++ features *faster*
and always track the *latest* standard even before all our compilers
support it -- so they are a force for *unifying* (tracking the very
latest C++) and *avoiding* dialects that exist today (where code gets
stuck on older C++ versions as dialects because it must target the lowest
common denominator of supported compilers).
</p>
<p>
We have plenty of users who already do feature detection and conditional
compilation. Feature macros allow doing that portably and more simply
than with the alternative techniques available today.
</p>
<h2>Rumination</h2>
<p>
The general problem we are trying to solve here is feature detection,
and based on detection results, conditional compilation of source
code. There have been suggestions that this should, instead of macros,
be done with a new language facility. The problem there is that
<ul>
<li>feature-testing macros work today, yesterday and tomorrow. I can
write my feature-macro protected code that uses C++20 facilities,
and even a C++03 compiler will happily accept that code without
any problems.</li>
<li>If I want to use such a hypothetical future facility, I need
to detect whether it's available. I either run into a chicken-egg
problem or I need to use feature macros. Sure, perhaps there
would be fewer feature macros, but there still would be some.</li>
<li>I have no reason to believe adding such language facilities
is likely to succeed, or that such facilities are easily adopted
by various tools that read C++ code. The preprocessor is already
there, and can provide feature-testing for every user right now,
without having to wait multiple years.</li>
<li>Standardizing feature-testing macros does not prevent us
from adding a non-macro facility for feature testing and conditional
compilation. Feature-testing macros and a potential non-macro
facility serve slightly different purposes with different trade-offs.</li>
</ul>
</p>
<p>
See the very end of this document for some strawman ideas on
such a language facility.
</p>
<h2>Current implementation status</h2>
<p>
Looking at some implementations, we notice the following:
<ul>
<li>gcc and libstdc++ implement both the language feature
macros and the library feature macros fairly completely.</li>
<li>clang implements the language feature macros, libc++
implements fairly few of the library feature macros.</li>
<li>msvc and its library don't implement feature macros.
</ul>
</p>
<p>
There are some indications that there would be a non-zero chance
that it would be more palatable for Microsoft
to implement the feature macros if they are provided
in an official WG21-sanctioned document. Some of these
indications are mentioned in <a href="http://wiki.edg.com/bin/view/Wg21toronto2017/P0697R0">the Toronto discussion notes on P0697R0</a>.
</p>
<h2>Why put it into the IS? Would not a TS be a better choice?</h2>
<p>
A while ago, I myself thought that putting the macros
into an IS makes no sense, because we publish an IS every
3 years, and can put out TSes much faster. However, I realized
that there's no particular difference; implementations can
track IS working drafts, which we publish after every meeting,
so there's not necessary any delay for implementations of the macros,
and for users being able to use the macros when adopting new
features aggressively.
</p>
<p>
Putting the macros into a 'real' standard document makes them
more portable, and putting them into the IS makes them the
most portable, assuming that implementations continue
to strive for conformance.
</p>
<h2>Why not just continue to publish a separate document and hope for
implementations to implement them?</h2>
<p>
As mentioned before, some vendors can justify implementing
the macros better if they are specified in an actual standard
document.
</p>
<p>
I know of some users (the Qt Base module of Qt in particular)
that use the feature macros correctly (without forking the language
or otherwise forming an actual dialect, and without duplicating
implementations), and seem to prefer a feature macro approach
to testing __cplusplus (which implementations are unlikely
to bump before they are in complete conformance) or to testing
features in an implementation-specific manner. Having the macros
be actually required, such testing remains simple and becomes
(much more) portable, compared to having to check what implementation is used
and which version of it is used.
</p>
<p>
I have fairly little to add why we should have feature macros to begin with. When used correctly, they allow library writers and library users to aggressively opt in to C++ features before an implementation is fully conforming to a new standard version, and to do so in a much simpler manner than detecting features in implementation-specific ways. Implementations that don't want to help early adopters can provide all feature test macros at once, at the same time they bump __cplusplus, and the presence of those macros will still help developers who've written code for implementations that do provide the feature test macros early. The feature macros can most likely be abused in various ways, but the
proper ways of using them are a clear improvement to me.
The next section attempts to
provide a mini-FAQ of the reasons why we should have feature macros
in the IS.
</p>
<h2>Mini-FAQ</h2>
<p>
<blockquote>
<p>
Q: Isn't the frequency of IS publications too low for feature macros in the IS to be useful?
</p>
<p>
A: No, because they can be updated in every working draft.
</p>
<p>
Q: But it doesn't make sense to put the macros into an IS. Conforming implementations will have the same values for the macros and they will implement all features, so just checking __cplusplus is enough.
</p>
<p>
A: In isolation, that's all true. But before __cplusplus is bumped,
users can write code that detects and uses some features, and that
code will continue to work even after __cplusplus is bumped
with no source code changes.
</p>
<p>
Q: Aren't we encouraging subsetting and dialects?
</p>
<p>
A: Not really. What we are encouraging is a way for implementations
to give users a way to adopt features early in a portable way,
in a fashion that continues to work even after __cplusplus is bumped, and of course works before it's bumped.
</p>
<p>
Q: But don't feature macros lead into language splintering and
a proliferation of configurations?
</p>
<p>
A: Based on deployment experience, no. The amount of configurations
is the same. The amount of macro permutations is the same, or
smaller. Yes, that can mean that some builds have a lesser feature
set, but that's a trade-off the code author made when choosing
to use feature macros rather than a vendor-specific mechanism.
The point is that feature macros don't seem to increase
the amount of configurations and permutations.
</p>
<p>
Q: But none of this really works unless implementations provide
feature macros early, tracking the working drafts. The only thing
we guarantee is that a conforming implementation provides the
IS versions of the macros.
</p>
<p>
A: Correct, but as described in the previous answers, some implementations
will track the draft macros, and (correctly-written) code written to use
those will automatically work with implementations that didn't
provide the draft macros but will provide the IS macros. And, again,
that will happen without code changes on the user side. We don't
guarantee that every implementation provides every step of a migration
path, but we do guarantee that all of them provide the last step,
and we know that some will provide the earlier steps, so users
can write code that benefits automatically from the point when
the final macros are enabled, and already reaped the benefits
when more aggressive implementations enabled them prior to the IS.
</p>
<p>
Q: But isn't it possible that feature macros are not enough? An implementation can define a feature macro but have a buggy implementation of the feature, so implementation-specific work-arounds may still be necessary.
</p>
<p>
A: No, that's not what feature macros are for. They are to test whether the feature exists at all, not to detect bugs, which needs to be done much less fequently, and will still require compiler version tests.
Let's illustrate what feature-testing macros are for and what they are not for:
<table>
<tr>
<th style="visibility:hidden"></th>
<th>Feature presence test</th>
<th>Compiler bug detection</th>
<th>Compile once</th>
<th>Build variations</th>
</tr>
<tr>
<td class="heading">Feature-testing macros</td>
<td align="center">Y</td>
<td align="center">N</td>
<td align="center">N</td>
<td align="center">Y</td>
</tr>
<tr>
<td class="heading">Compiler-version macros (__GNUC__, _MSC_VER_)</td>
<td align="center">N</td>
<td align="center">Y</td>
<td align="center">N</td>
<td align="center">Y</td>
</tr>
<tr>
<td class="heading">Platform/target macros (bit width, endianness, architecture)</td>
<td align="center">N</td>
<td align="center">N</td>
<td align="center">N</td>
<td align="center">Y</td>
</tr>
<tr>
<td class="heading">Header guards</td>
<td align="center">N</td>
<td align="center">N</td>
<td align="center">Y</td>
<td align="center">N</td>
</tr>
<tr>
<td class="heading">Modules</td>
<td align="center">N</td>
<td align="center">N</td>
<td align="center">Y</td>
<td align="center">Y</td>
</tr>
</table>
</p>
</blockquote>
</p>
<h2>A new language facility</h2>
<p>
I has been suggested to me that one way to do feature detection
would be to annotate declarations somehow when they are exported
from a module, and to import only those declarations for which
the condition in the annotation evaluates to true.
</p>
<p>
Separately from that, I can envision doing conditional
compilation with a new kind of language block:
<blockquote>
<code>
<pre>
std::compile_if<constant expression goes here> {
// here goes code that uses the language facility
// that the constant expression detected
} else { // the else block is completely optional
// here goes code that doesn't use the aforementioned
// language facility, because it's not available
}
</pre>
</code>
</blockquote>
</p>
<p>
We might also get the desired effect with injection:
<blockquote>
<code>
<pre>
constexpr {
if (constant expression goes here) -> {
// here goes code that uses the language facility
// that the constant expression detected
} else -> { // the else block is completely optional
// here goes code that doesn't use the aforementioned
// language facility, because it's not available
}
}
</pre>
</code>
</blockquote>
</p>
<p>
Whether any experimentation with a module facility has been done,
I don't know. These ideas are not, to my knowledge, very baked.
Having such facilities would certainly be great, but we can't
know when they would appear, and they don't work with a C++03
compiler, or with a C++2a compiler that hasn't yet implemented
all of C++20. Feature macros do.
</p>
</body>
</html>