Moving the kernel to modern C
Moving the kernel to modern C
Posted Feb 28, 2022 13:56 UTC (Mon) by jem (subscriber, #24231)In reply to: Moving the kernel to modern C by ianmcc
Parent article: Moving the kernel to modern C
Posted Feb 28, 2022 17:06 UTC (Mon)
by ianmcc (subscriber, #88379)
[Link] (4 responses)
for (int i = ..)
Posted Feb 28, 2022 20:10 UTC (Mon)
by nybble41 (subscriber, #55106)
[Link] (3 responses)
What you say agrees with the C++ standard, but it makes me wonder why the standard authors appear to have been competing to come up with the most Byzantine special cases and exceptions they could think of to integrate into the standard rather than taking the simplest and least surprising route. Syntactically, the body of the for loop is a single statement which may be a compound statement. That part is the same as C. The braces are *not* part of the syntax for the loop. If the scope of the control variable were in fact the loop itself, and the body were treated the same as any other statement, then the compound statement would be an independent scope nested *within* that for-loop scope, and declarations within the compound statement would shadow any declarations scoped to the for loop (as they do in C). Instead the standard pierces the abstraction and treats compound statements in a for loop body differently than compound statements located elsewhere. There is no logic to this that I can see, just a bald statement that "If a name introduced in an init-statement or for-range-declaration is redeclared in the outermost block of the substatement, the program is ill-formed."
Posted Mar 1, 2022 16:27 UTC (Tue)
by ianmcc (subscriber, #88379)
[Link] (2 responses)
After all, who would write such code anyway? Its a strange thing to take issue with.
Posted Mar 1, 2022 21:58 UTC (Tue)
by nybble41 (subscriber, #55106)
[Link] (1 responses)
> In C++ the declaration and the body of the loop are the same scope. The body of the for loop is just *statement*. If the declaration were in that scope it wouldn't survive from one iteration of the loop to the next or be visible in *condition* or *expression*. Declarations in *init-statement* are scoped over the entire for loop, not just the body. Normally a compound statement within *statement* would introduce its own separate block scope *below* the level of *statement*, but in C++ the lines are blurred between the body of the for loop and the *inside* of the compound statement. In other words, I would expect this to be a redeclaration error, because `char i` and `int i` are declared in the same scope (note that all the examples in the standard are of this form): but not this, because `char i` is declared in the new *nested* scope created by the compound statement and not directly in the body of the for loop: Contrast this with the following code which the standard (C++20 draft) claims is "equivalent" to the second example "except that names declared in the init-statement are in the same declarative region as those declared in the condition, and except that a continue in statement (not enclosed in another iteration statement) will execute expression before re-evaluating condition": In the "equivalent" while loop version there is clearly no redeclaration error—the `char i` declaration is within not just one but two levels of compound statements under the while loop and the scope where `int i` was declared. > After all, who would write such code anyway? Its a strange thing to take issue with. Whether you would write that by hand or not, it's an unnecessary (and IMHO completely pointless) complication which moreover breaks compatibility with C. Redeclaration conflicts could appear as a result of macro expansion or other code generation, not just in hand-written code.
Posted Mar 2, 2022 8:56 UTC (Wed)
by ianmcc (subscriber, #88379)
[Link]
The bottom line is that C++ will flag an error in some instances of very dubious code that is most likely a bug anyway (i.e. declaring a variable that shadows the loop control variable) where C99 would allow it. None of the standards committee see it as something worth the bother of fixing. If you really did intend to introduce a shadow declaration, the simple fix is to enclose it in another compound statement.
Posted Feb 28, 2022 17:14 UTC (Mon)
by ianmcc (subscriber, #88379)
[Link]
Moving the kernel to modern C
{
int i = 2; // not valid C++. There is already a variable 'i' declared in this scope
}
Moving the kernel to modern C
Moving the kernel to modern C
Moving the kernel to modern C
for (init-statement condition[opt] ; expression) statement
for (int i = 0; i < N; ++i)
char i = 7;
for (int i = 0; i < N; ++i) {
char i = 7;
}
{
int i = 0; /* init-statement */
while (i < N /* condition */) {
{ char i = 7; } // statement
++i;
}
}
Moving the kernel to modern C
See also http://www.open-std.org/jtc1/sc22/wg14/www/docs/n1865.htm
Moving the kernel to modern C
