-
Notifications
You must be signed in to change notification settings - Fork 19
/
DIALECT
107 lines (66 loc) · 3.45 KB
/
DIALECT
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
The compiler accepts "classic" C as described in K&R (1978), with
a few changes as detailed below.
==========
EXTENSIONS
==========
Most of these were common extensions that made it into the ANSI standard.
* 'unsigned' is extended to all integer types.
K&R only mentions 'unsigned int'. NCC extends this qualifier to all
integral types, including 'char', which is signed by default.
* Separate member namespaces.
'struct' and 'union' types each enclose a unique namespace, so members of
different aggregate types need not have different names.
* Separate tag namespaces.
'struct foo' and 'union foo' refer to different types, even in the same scope.
There's really no reason to have these namespaces overlap, and I suspect it was
merely an artifact of dmr's original C compiler that they do (the same could
be said about shared member namespaces). Shared tag namespaces are a nuisance
to enforce, and so the compiler doesn't bother.
* Arbitrarily-typed bitfields.
The compiler allows bit fields to have any integral type: the size of the type
limits the number of bits that are permitted and sets the alignment requirement
of the word that contains field. Also, the signedness of the field is honored.
=======
CHANGES
=======
These are intentional changes to the syntax and semantics of K&R C. These
are fairly minor but make the language more consistent. (ANSI attempts to
address some of the issues addressed here, but they were more constrained
by backwards-compatibility concerns.)
* Floating-point types
K&R basically treats 'float' as a kind of short 'double': 'float' types are
promoted to 'double' before just about any operation, in the same way that a
'short' is promoted to 'int'. This behavior is suboptimal on AMD64.
NCC doesn't have a 'double' type; its double-precision type is instead called
'long float', and its relationship to 'float' is the same as that between
'long' and 'int'. A 'float' is coerced to a 'long float' only during the
"usual conversions": when one of the operands to a binary operator is 'long
float', the other is promoted to 'long float'. Note that this means that
function arguments and return values of type 'float' are distinct from their
'long float' analogs.
All floating-point constants are assumed to be 'float' unless suffixed with
'L' or 'l', in which case they are 'long float'.
* Integer constants
'long' integer constants must be explicitly suffixed with 'L' or 'l'. This
is for parity with the floating-point types and to avoid inadvertent long
constants. Both K&R and ANSI rules for classifying constants are dependent
on the value and base of the constant, as well as the implementation-specific
ranges of the integral types.
=======
DEFECTS
=======
These are minor issues that could be fixed pretty easily. I'm just lazy.
* 'extern' objects must be declared at file scope.
K&R and ANSI both allow 'extern' objects to appear in local scopes. This
results in a lot of extra bookkeeping in the front end because:
1. all such objects must have compatible declarations, and
2. such declarations can't be exported to the global scope.
This means that "invisible" entries need to hang around the symbol table,
and that just makes a mess, for no good reason that I can determine.
* Aggregate initializer braces can't be elided.
int a[2][2] = { { 1, 2 }, { 3, 4 } };
and
int a[2][2] = { 1, 2, 3, 4 };
are identical by both K&R and ANSI, but NCC only accepts the first.
Charles Youse <[email protected]>
December 27, 2018