2019-02-18 16:42:15 +00:00
|
|
|
/*
|
|
|
|
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
|
|
|
* See the copyright notice in the ACK home directory, in the file "Copyright".
|
|
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
/* DECLARATION SYNTAX PARSER */
|
|
|
|
|
|
|
|
{
|
|
|
|
#include "parameters.h"
|
|
|
|
#include <alloc.h>
|
|
|
|
#include <flt_arith.h>
|
|
|
|
#include "idf.h"
|
|
|
|
#include "arith.h"
|
|
|
|
#include "LLlex.h"
|
|
|
|
#include "label.h"
|
|
|
|
#include "code.h"
|
|
|
|
#include "type.h"
|
|
|
|
#include "proto.h"
|
|
|
|
#include "struct.h"
|
|
|
|
#include "field.h"
|
|
|
|
#include "decspecs.h"
|
|
|
|
#include "declarator.h"
|
|
|
|
#include "def.h"
|
|
|
|
#include "declar.h"
|
|
|
|
#include "label.h"
|
|
|
|
#include "expr.h"
|
|
|
|
#include "sizes.h"
|
|
|
|
#include "level.h"
|
|
|
|
#include "error.h"
|
|
|
|
#include "stab.h"
|
|
|
|
|
|
|
|
#ifdef LINT
|
|
|
|
#include "l_lint.h"
|
|
|
|
#endif /* LINT */
|
|
|
|
|
|
|
|
extern char options[];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 3.5 */
|
|
|
|
declaration
|
|
|
|
{struct decspecs Ds;}
|
|
|
|
:
|
|
|
|
{Ds = null_decspecs;}
|
|
|
|
decl_specifiers(&Ds)
|
|
|
|
init_declarator_list(&Ds)?
|
|
|
|
';'
|
|
|
|
;
|
|
|
|
|
|
|
|
/* A `decl_specifiers' describes a sequence of a storage_class_specifier,
|
|
|
|
an unsigned_specifier, a size_specifier and a simple type_specifier,
|
|
|
|
which may occur in arbitrary order and each of which may be absent;
|
|
|
|
at least one of them must be present, however, since the totally
|
|
|
|
empty case has already be dealt with in `external_definition'.
|
|
|
|
This means that something like:
|
|
|
|
unsigned extern int short xx;
|
|
|
|
is perfectly legal C.
|
|
|
|
|
|
|
|
On top of that, multiple occurrences of storage_class_specifiers,
|
|
|
|
unsigned_specifiers and size_specifiers are errors, but a second
|
|
|
|
type_specifier should end the decl_specifiers and be treated as
|
|
|
|
the name to be declared.
|
|
|
|
Such a language is not easily expressed in a grammar; enumeration
|
|
|
|
of the permutations is unattractive. We solve the problem by
|
|
|
|
having a regular grammar for the "soft" items, handling the single
|
|
|
|
occurrence of the type_specifier in the grammar (we have no choice),
|
|
|
|
collecting all data in a `struct decspecs' and turning that data
|
|
|
|
structure into what we want.
|
|
|
|
|
|
|
|
The existence of declarations like
|
|
|
|
short typedef yepp;
|
|
|
|
makes all hope of writing a specific grammar for typedefs illusory.
|
|
|
|
*/
|
|
|
|
/* Accept a single declaration specifier. Then accept zero or more
|
|
|
|
declaration specifiers. There can be a conflict on both
|
|
|
|
TYPE_IDENTIFIER and IDENTIFIER.
|
|
|
|
The following rule is used:
|
|
|
|
When we see a TYPE_IDENTIFIER, we accept it if no type-specifier was
|
|
|
|
given, and it is not directly followed by an identifier. If a
|
|
|
|
type-specifier was given, it is taken as the identifier being
|
|
|
|
declared. If it is followed by an identifier, we assume that an
|
|
|
|
error has been made, (e.g. unsigned typedeffed_int x;) and that
|
|
|
|
this will be detected later on.
|
|
|
|
When we see an IDENTIFIER, directly followed by another IDENTIFIER,
|
|
|
|
we assume that a typing mistake has been made, and we accept it as
|
|
|
|
an erroneous type-identifier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
decl_specifiers /* non-empty */ (register struct decspecs *ds;)
|
|
|
|
/* Reads a non-empty decl_specifiers and fills the struct
|
|
|
|
decspecs *ds.
|
|
|
|
*/
|
|
|
|
:
|
|
|
|
single_decl_specifier(ds)
|
|
|
|
[ %while( (DOT==TYPE_IDENTIFIER
|
|
|
|
&& ds->ds_size == 0
|
|
|
|
&& ds->ds_unsigned == 0
|
|
|
|
&& ds->ds_type == (struct type *)0)
|
|
|
|
|| AHEAD == IDENTIFIER) /* always an error */
|
|
|
|
single_decl_specifier(ds)
|
|
|
|
]*
|
|
|
|
{do_decspecs(ds);}
|
|
|
|
;
|
|
|
|
|
|
|
|
single_decl_specifier /* non_empty */ (register struct decspecs *ds;)
|
|
|
|
:
|
|
|
|
[ AUTO | STATIC | EXTERN | TYPEDEF | REGISTER ]
|
|
|
|
{ if (ds->ds_sc_given)
|
|
|
|
error("repeated storage class specifier");
|
|
|
|
ds->ds_sc_given = 1;
|
|
|
|
ds->ds_sc = DOT;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
VOLATILE
|
|
|
|
{ if (ds->ds_typequal & TQ_VOLATILE)
|
|
|
|
error("repeated type qualifier");
|
|
|
|
ds->ds_typequal |= TQ_VOLATILE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
CONST
|
|
|
|
{ if (ds->ds_typequal & TQ_CONST)
|
|
|
|
error("repeated type qualifier");
|
|
|
|
ds->ds_typequal |= TQ_CONST;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
[ SHORT | LONG ]
|
Begin to add `long long` to C compiler for linux386.
Add long long type, but without literals; you can't say '123LL' yet.
You can try constant operations, like `(long long)123 + 1`, but the
compiler's `arith` type might not be wide enough. Conversions,
shifts, and some other operations don't work in i386 ncg; I am using a
union instead of conversions:
union q {
long long ll;
unsigned long long ull;
int i[2];
};
Hack plat/linux386/descr to enable long long (size 8, alignment 4)
only for this platform. The default for other platforms is to disable
long long (size -1).
In lang/cem/cemcom.ansi,
- BigPars, SmallPars: Add default size, alignment of long long.
- align.h: Add lnglng_align.
- arith.c: Convert arithmetic operands to long long or unsigned long
long when necessary; avoid conversion from long long to long.
Allow long long as an arithmetic, integral, or logical operand.
- ch3.c: Handle long long like int and long when erroneously applying
a selector, like `long long ll; ll.member` or `ll->member`. Add
long long to integral and arithmetic types.
- code.c: Add long long to type stabs for debugging.
- conversion.c: Add long long to integral conversions.
- cstoper.c: Write masks up to full_mask[8]. Add FIXME comment.
- declar.g: Parse `long long` in code.
- decspecs.c: Understand long long in type declarations.
- eval.c: Add long long to operations, to generate code like `adi 8`.
Don't use `ldc` with constant over 4 bytes.
- ival.g: Allow long long in initializations.
- main.c: Set lnglng_type and related values.
- options.c: Add option like `-Vq8.4` to set long long to size 8,
alignment 4. I chose 'q', because Perl's pack and Ruby's
Array#pack use 'q' for 64-bit or long long values; it might be a
reference to BSD's old quad_t alias for long long.
- sizes.h: Add lnglng_size.
- stab.c: Allow long long when writing the type stab for debugging.
Switch from calculating the ranges to hardcoding them in strings;
add 8-byte ranges as a special case. This also hardcodes the
unsigned 4-byte range as "0;-1". Before it was either "0;-1" or
"0;4294967295", depending on sizeof(long) in the compiler.
- struct.c: Try long long bitfield, but it will probably give the
error, "bit field type long long does not fit in a word".
- switch.c: Update comment.
- tokenname.c: Define LNGLNG (long long) like LNGDBL (long double).
- type.c, type.str: Add lnglng_type and ulnglng_type. Add function
no_long_long() to check if long long is disabled.
2019-09-02 15:24:44 +00:00
|
|
|
{ if (ds->ds_size == LONG && DOT == LONG)
|
|
|
|
ds->ds_size = LNGLNG;
|
|
|
|
else {
|
|
|
|
if (ds->ds_size)
|
|
|
|
error("repeated size specifier");
|
|
|
|
ds->ds_size = DOT;
|
|
|
|
}
|
2019-02-18 16:42:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
[ SIGNED | UNSIGNED ]
|
|
|
|
{ if (ds->ds_unsigned != 0)
|
|
|
|
error("repeated sign specifier");
|
|
|
|
ds->ds_unsigned = DOT;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
[ VOID | CHAR | INT | FLOAT | DOUBLE ]
|
|
|
|
{
|
|
|
|
idf2type(dot.tk_idf, &ds->ds_type);
|
|
|
|
ds->ds_typedef = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
%default TYPE_IDENTIFIER
|
|
|
|
{
|
|
|
|
idf2type(dot.tk_idf, &ds->ds_type);
|
|
|
|
ds->ds_typedef = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
%erroneous
|
|
|
|
IDENTIFIER
|
|
|
|
{
|
|
|
|
error("%s is not a type identifier", dot.tk_idf->id_text);
|
|
|
|
ds->ds_type = error_type;
|
|
|
|
if (dot.tk_idf->id_def) {
|
|
|
|
dot.tk_idf->id_def->df_type = error_type;
|
|
|
|
dot.tk_idf->id_def->df_sc = TYPEDEF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
%illegal
|
|
|
|
IDENTIFIER
|
|
|
|
|
|
|
|
|
struct_or_union_specifier(&ds->ds_type)
|
|
|
|
|
|
|
|
|
enum_specifier(&ds->ds_type)
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.5.2 */
|
|
|
|
type_specifier(struct type **tpp;)
|
|
|
|
/* Used in struct/union declarations and in casts; only the
|
|
|
|
type is relevant.
|
|
|
|
*/
|
|
|
|
{struct decspecs Ds; Ds = null_decspecs;}
|
|
|
|
:
|
|
|
|
decl_specifiers(&Ds)
|
|
|
|
{
|
|
|
|
if (Ds.ds_sc_given)
|
|
|
|
error("storage class ignored");
|
|
|
|
if (Ds.ds_sc == REGISTER)
|
|
|
|
error("register ignored");
|
|
|
|
}
|
|
|
|
{*tpp = Ds.ds_type;}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.5 */
|
|
|
|
init_declarator_list(struct decspecs *ds;):
|
|
|
|
init_declarator(ds)
|
|
|
|
[ ',' init_declarator(ds) ]*
|
|
|
|
;
|
|
|
|
|
|
|
|
init_declarator(register struct decspecs *ds;)
|
|
|
|
{
|
|
|
|
struct declarator Dc;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
{
|
|
|
|
Dc = null_declarator;
|
|
|
|
}
|
|
|
|
[
|
|
|
|
declarator(&Dc)
|
|
|
|
{
|
|
|
|
reject_params(&Dc);
|
|
|
|
declare_idf(ds, &Dc, level);
|
|
|
|
#ifdef LINT
|
|
|
|
lint_declare_idf(Dc.dc_idf, ds->ds_sc);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
[
|
|
|
|
initializer(Dc.dc_idf, ds->ds_sc)
|
|
|
|
|
|
|
|
|
{ code_declaration(Dc.dc_idf, (struct expr *) 0, level, ds->ds_sc); }
|
|
|
|
]
|
|
|
|
]
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
add_auto(Dc.dc_idf);
|
|
|
|
#endif /* LINT */
|
|
|
|
remove_declarator(&Dc);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.5.7: initializer */
|
|
|
|
initializer(struct idf *idf; int sc;)
|
|
|
|
{
|
|
|
|
struct expr *expr = (struct expr *) 0;
|
|
|
|
int fund = idf->id_def->df_type->tp_fund;
|
|
|
|
int autoagg = (level >= L_LOCAL
|
|
|
|
&& sc != STATIC
|
|
|
|
&& ( fund == STRUCT
|
|
|
|
|| fund == UNION
|
|
|
|
|| fund == ARRAY));
|
|
|
|
int globalflag = level == L_GLOBAL
|
|
|
|
|| (level >= L_LOCAL && sc == STATIC);
|
|
|
|
}
|
|
|
|
:
|
|
|
|
{ if (idf->id_def->df_type->tp_fund == FUNCTION) {
|
|
|
|
error("illegal initialization of function");
|
|
|
|
idf->id_def->df_type->tp_fund = ERRONEOUS;
|
|
|
|
}
|
|
|
|
if (level == L_FORMAL2)
|
|
|
|
error("illegal initialization of formal parameter");
|
|
|
|
}
|
|
|
|
'='
|
|
|
|
{
|
|
|
|
if (AHEAD != '{' && AHEAD != STRING ) autoagg = 0;
|
|
|
|
#ifdef LINT
|
|
|
|
lint_statement();
|
|
|
|
#endif /* LINT */
|
|
|
|
if (globalflag) {
|
|
|
|
struct expr ex;
|
|
|
|
code_declaration(idf, &ex, level, sc);
|
|
|
|
}
|
|
|
|
else if (autoagg)
|
|
|
|
loc_init((struct expr *) 0, idf);
|
|
|
|
}
|
|
|
|
initial_value((globalflag || autoagg) ?
|
|
|
|
&(idf->id_def->df_type)
|
|
|
|
: (struct type **)0,
|
|
|
|
&expr)
|
|
|
|
{ if (! globalflag) {
|
|
|
|
if (idf->id_def->df_type->tp_fund == FUNCTION) {
|
|
|
|
free_expression(expr);
|
|
|
|
expr = 0;
|
|
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
|
|
print_expr("initializer-expression", expr);
|
|
|
|
#endif /* DEBUG */
|
|
|
|
#ifdef LINT
|
|
|
|
change_state(idf, SET);
|
|
|
|
#endif /* LINT */
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
if (options['g'] && level >= L_LOCAL && expr) {
|
|
|
|
db_line(expr->ex_file, (unsigned) expr->ex_line);
|
|
|
|
}
|
|
|
|
#endif /* DBSYMTAB */
|
|
|
|
if (autoagg)
|
|
|
|
loc_init((struct expr *) 0, idf);
|
|
|
|
else code_declaration(idf, expr, level, sc);
|
|
|
|
}
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
if (options['g'] && globalflag) {
|
|
|
|
stb_string(idf->id_def, sc, idf->id_text);
|
|
|
|
}
|
|
|
|
#endif /* DBSYMTAB */
|
|
|
|
idf_initialized(idf);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/*
|
|
|
|
Functions yielding pointers to functions must be declared as, e.g.,
|
|
|
|
int (*hehe(par1, par2))() char *par1, *par2; {}
|
|
|
|
Since the function heading is read as a normal declarator,
|
|
|
|
we just include the (formal) parameter list in the declarator
|
|
|
|
description list dc.
|
|
|
|
*/
|
|
|
|
/* 3.5.4 */
|
|
|
|
declarator(register struct declarator *dc;)
|
|
|
|
{ struct formal *fm = NO_PARAMS;
|
|
|
|
struct proto *pl = NO_PROTO;
|
|
|
|
arith count;
|
|
|
|
int qual;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
primary_declarator(dc)
|
|
|
|
[/*%while(1)*/
|
|
|
|
'('
|
|
|
|
[ %if (DOT != IDENTIFIER)
|
|
|
|
parameter_type_list(&pl)
|
|
|
|
|
|
|
|
|
formal_list(&fm)
|
|
|
|
|
|
|
|
|
/* empty */
|
|
|
|
]
|
|
|
|
')'
|
|
|
|
{ add_decl_unary(dc, FUNCTION, 0, (arith)0, fm, pl);
|
|
|
|
fm = NO_PARAMS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
arrayer(&count)
|
|
|
|
{add_decl_unary(dc, ARRAY, 0, count, NO_PARAMS, NO_PROTO);}
|
|
|
|
]*
|
|
|
|
|
|
|
|
|
pointer(&qual) declarator(dc)
|
|
|
|
{add_decl_unary(dc, POINTER, qual, (arith)0, NO_PARAMS, NO_PROTO);}
|
|
|
|
;
|
|
|
|
|
|
|
|
primary_declarator(register struct declarator *dc;) :
|
|
|
|
identifier(&dc->dc_idf)
|
|
|
|
|
|
|
|
|
'(' declarator(dc) ')'
|
|
|
|
;
|
|
|
|
|
|
|
|
arrayer(arith *sizep;)
|
|
|
|
{ struct expr *expr; }
|
|
|
|
:
|
|
|
|
'['
|
|
|
|
{ *sizep = (arith)-1; }
|
|
|
|
[
|
|
|
|
constant_expression(&expr)
|
|
|
|
{
|
|
|
|
check_array_subscript(expr);
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
*sizep = (arith)expr->VL_VALUE;
|
2019-02-18 16:42:15 +00:00
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
]?
|
|
|
|
']'
|
|
|
|
;
|
|
|
|
|
|
|
|
formal_list (struct formal **fmp;)
|
|
|
|
:
|
|
|
|
formal(fmp) [ %persistent ',' formal(fmp) ]*
|
|
|
|
;
|
|
|
|
|
|
|
|
formal(struct formal **fmp;)
|
|
|
|
{struct idf *idf; }
|
|
|
|
:
|
|
|
|
identifier(&idf)
|
|
|
|
{
|
|
|
|
register struct formal *new = new_formal();
|
|
|
|
|
|
|
|
new->fm_idf = idf;
|
|
|
|
new->next = *fmp;
|
|
|
|
*fmp = new;
|
|
|
|
if (idf->id_def && idf->id_def->df_sc == TYPEDEF) {
|
|
|
|
error("typedef name %s may not be redeclared as a parameter", idf->id_text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* Change 2 */
|
|
|
|
enum_specifier(register struct type **tpp;)
|
|
|
|
{
|
|
|
|
struct idf *idf;
|
|
|
|
arith l = (arith)0;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
{if (*tpp) error("multiple types in declaration");}
|
|
|
|
ENUM
|
|
|
|
[
|
|
|
|
{declare_struct(ENUM, (struct idf *) 0, tpp);}
|
|
|
|
enumerator_pack(*tpp, &l)
|
|
|
|
|
|
|
|
|
identifier(&idf)
|
|
|
|
[
|
|
|
|
{declare_struct(ENUM, idf, tpp);}
|
|
|
|
enumerator_pack(*tpp, &l)
|
|
|
|
{
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
if (options['g']) {
|
|
|
|
stb_tag(idf->id_tag, idf->id_text);
|
|
|
|
}
|
|
|
|
#endif /*DBSYMTAB */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
{apply_struct(ENUM, idf, tpp);}
|
|
|
|
/* empty */
|
|
|
|
]
|
|
|
|
]
|
|
|
|
;
|
|
|
|
|
|
|
|
enumerator_pack(register struct type *tp; arith *lp;) :
|
|
|
|
'{'
|
|
|
|
enumerator(tp, lp)
|
|
|
|
[%while (AHEAD != '}')
|
|
|
|
','
|
|
|
|
enumerator(tp, lp)
|
|
|
|
]*
|
|
|
|
[
|
|
|
|
',' {warning("unexpected trailing comma in enumerator pack");}
|
|
|
|
]?
|
|
|
|
'}'
|
|
|
|
{tp->tp_size = int_size;}
|
|
|
|
/* fancy implementations that put small enums in 1 byte
|
|
|
|
or so should start here.
|
|
|
|
*/
|
|
|
|
;
|
|
|
|
|
|
|
|
enumerator(struct type *tp; arith *lp;)
|
|
|
|
{
|
|
|
|
struct idf *idf;
|
|
|
|
struct expr *expr;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
identifier(&idf)
|
|
|
|
[
|
|
|
|
'='
|
|
|
|
constant_expression(&expr)
|
|
|
|
{
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
*lp = (arith)expr->VL_VALUE;
|
2019-02-18 16:42:15 +00:00
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
]?
|
|
|
|
{declare_enum(tp, idf, (*lp)++);}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 8.5 */
|
|
|
|
struct_or_union_specifier(register struct type **tpp;)
|
|
|
|
{
|
|
|
|
int fund;
|
|
|
|
struct idf *idfX;
|
|
|
|
register struct idf *idf;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
{if (*tpp) error("multiple types in declaration");}
|
|
|
|
[ STRUCT | UNION ]
|
|
|
|
{fund = DOT;}
|
|
|
|
[
|
|
|
|
{
|
|
|
|
declare_struct(fund, (struct idf *)0, tpp);
|
|
|
|
}
|
|
|
|
struct_declaration_pack(*tpp)
|
|
|
|
|
|
|
|
|
identifier(&idfX) { idf = idfX; }
|
|
|
|
[
|
|
|
|
{
|
|
|
|
declare_struct(fund, idf, tpp);
|
|
|
|
(idf->id_tag->tg_busy)++;
|
|
|
|
}
|
|
|
|
struct_declaration_pack(*tpp)
|
|
|
|
{
|
|
|
|
(idf->id_tag->tg_busy)--;
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
if (options['g']) {
|
|
|
|
stb_tag(idf->id_tag, idf->id_text);
|
|
|
|
}
|
|
|
|
#endif /*DBSYMTAB */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
{
|
|
|
|
/* a ';' means an empty declaration (probably)
|
|
|
|
* this means that we have to declare a new
|
|
|
|
* structure. (yegh)
|
|
|
|
*/
|
|
|
|
if (DOT == ';' &&
|
|
|
|
( !idf->id_tag ||
|
|
|
|
idf->id_tag->tg_level != level ||
|
|
|
|
idf->id_tag->tg_type->tp_size < 0
|
|
|
|
)) declare_struct(fund, idf, tpp);
|
|
|
|
else apply_struct(fund, idf, tpp);
|
|
|
|
}
|
|
|
|
/* empty */
|
|
|
|
]
|
|
|
|
]
|
|
|
|
;
|
|
|
|
|
|
|
|
struct_declaration_pack(register struct type *stp;)
|
|
|
|
{
|
|
|
|
struct sdef **sdefp = &stp->tp_sdef;
|
|
|
|
arith size = (arith)0;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
/* The size is only filled in after the whole struct has
|
|
|
|
been read, to prevent recursive definitions.
|
|
|
|
*/
|
|
|
|
'{'
|
|
|
|
struct_declaration(stp, &sdefp, &size)+
|
|
|
|
'}'
|
|
|
|
{stp->tp_size = align(size, stp->tp_align);
|
|
|
|
completed(stp);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
struct_declaration(struct type *stp; struct sdef ***sdefpp; arith *szp;)
|
|
|
|
{struct type *tp;}
|
|
|
|
:
|
|
|
|
type_specifier(&tp) struct_declarator_list(tp, stp, sdefpp, szp) ';'
|
|
|
|
;
|
|
|
|
|
|
|
|
struct_declarator_list(struct type *tp; struct type *stp;
|
|
|
|
struct sdef ***sdefpp; arith *szp;)
|
|
|
|
:
|
|
|
|
struct_declarator(tp, stp, sdefpp, szp)
|
|
|
|
[ ',' struct_declarator(tp, stp, sdefpp, szp) ]*
|
|
|
|
;
|
|
|
|
|
|
|
|
struct_declarator(struct type *tp; struct type *stp;
|
|
|
|
struct sdef ***sdefpp; arith *szp;)
|
|
|
|
{
|
|
|
|
struct declarator Dc;
|
|
|
|
struct field *fd = 0;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
{
|
|
|
|
Dc = null_declarator;
|
|
|
|
}
|
|
|
|
[
|
|
|
|
declarator(&Dc)
|
|
|
|
{reject_params(&Dc);}
|
|
|
|
bit_expression(&fd)?
|
|
|
|
|
|
|
|
|
{Dc.dc_idf = gen_idf();}
|
|
|
|
bit_expression(&fd)
|
|
|
|
]
|
|
|
|
{add_sel(stp, declare_type(tp, &Dc), Dc.dc_idf, sdefpp, szp, fd);}
|
|
|
|
{remove_declarator(&Dc);}
|
|
|
|
;
|
|
|
|
|
|
|
|
bit_expression(struct field **fd;)
|
|
|
|
{ struct expr *expr; }
|
|
|
|
:
|
|
|
|
{
|
|
|
|
*fd = new_field();
|
|
|
|
}
|
|
|
|
':'
|
|
|
|
constant_expression(&expr)
|
|
|
|
{
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
(*fd)->fd_width = (arith)expr->VL_VALUE;
|
2019-02-18 16:42:15 +00:00
|
|
|
free_expression(expr);
|
|
|
|
#ifdef NOBITFIELD
|
|
|
|
error("bitfields are not implemented");
|
|
|
|
#endif /* NOBITFIELD */
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 8.7 */
|
|
|
|
cast(struct type **tpp;)
|
|
|
|
{struct declarator Dc;}
|
|
|
|
:
|
|
|
|
{Dc = null_declarator;}
|
|
|
|
'('
|
|
|
|
type_specifier(tpp)
|
|
|
|
abstract_declarator(&Dc)
|
|
|
|
')'
|
|
|
|
{*tpp = declare_type(*tpp, &Dc);}
|
|
|
|
{remove_declarator(&Dc);}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* This code is an abject copy of that of 'declarator', for lack of
|
|
|
|
a two-level grammar.
|
|
|
|
*/
|
|
|
|
abstract_declarator(register struct declarator *dc;)
|
|
|
|
{ struct proto *pl = NO_PROTO;
|
|
|
|
arith count;
|
|
|
|
int qual;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
primary_abstract_declarator(dc)
|
|
|
|
[
|
|
|
|
'('
|
|
|
|
[
|
|
|
|
parameter_type_list(&pl)
|
|
|
|
|
|
|
|
|
/* empty */
|
|
|
|
]
|
|
|
|
')'
|
|
|
|
{add_decl_unary(dc, FUNCTION, 0, (arith)0, NO_PARAMS, pl);
|
|
|
|
if (pl) remove_proto_idfs(pl);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
arrayer(&count)
|
|
|
|
{add_decl_unary(dc, ARRAY, 0, count, NO_PARAMS, NO_PROTO);}
|
|
|
|
]*
|
|
|
|
|
|
|
|
|
pointer(&qual) abstract_declarator(dc)
|
|
|
|
{add_decl_unary(dc, POINTER, qual, (arith)0, NO_PARAMS, NO_PROTO);}
|
|
|
|
;
|
|
|
|
|
|
|
|
%first first_of_parameter_type_list, parameter_type_list;
|
|
|
|
|
|
|
|
primary_abstract_declarator(struct declarator *dc;)
|
|
|
|
:
|
|
|
|
[%if (AHEAD == ')' || first_of_parameter_type_list(AHEAD))
|
|
|
|
/* empty */
|
|
|
|
|
|
|
|
|
'(' abstract_declarator(dc) ')'
|
|
|
|
]
|
|
|
|
;
|
|
|
|
|
|
|
|
parameter_type_list(struct proto **plp;)
|
|
|
|
{ int save_level; }
|
|
|
|
:
|
|
|
|
{ if (level > L_PROTO) {
|
|
|
|
save_level = level;
|
|
|
|
level = L_PROTO;
|
|
|
|
} else level--;
|
|
|
|
}
|
|
|
|
parameter_decl_list(plp)
|
|
|
|
[
|
|
|
|
',' ELLIPSIS
|
|
|
|
{ register struct proto *new = new_proto();
|
|
|
|
|
|
|
|
new->next = *plp;
|
|
|
|
new->pl_flag = PL_ELLIPSIS;
|
|
|
|
*plp = new;
|
|
|
|
}
|
|
|
|
|
|
|
|
]?
|
|
|
|
{ check_for_void(*plp);
|
|
|
|
if (level == L_PROTO)
|
|
|
|
level = save_level;
|
|
|
|
else level++;
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
parameter_decl_list(struct proto **plp;)
|
|
|
|
:
|
|
|
|
parameter_decl(plp)
|
|
|
|
[ %while (AHEAD != ELLIPSIS)
|
|
|
|
%persistent
|
|
|
|
',' parameter_decl(plp)
|
|
|
|
]*
|
|
|
|
;
|
|
|
|
|
|
|
|
parameter_decl(struct proto **plp;)
|
|
|
|
{ register struct proto *new = new_proto();
|
|
|
|
struct declarator Dc;
|
|
|
|
struct decspecs Ds;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
{ Dc = null_declarator;
|
|
|
|
Ds = null_decspecs;
|
|
|
|
}
|
|
|
|
decl_specifiers(&Ds)
|
|
|
|
parameter_declarator(&Dc)
|
|
|
|
{ add_proto(new, &Ds, &Dc, level);
|
|
|
|
new->next = *plp;
|
|
|
|
*plp = new;
|
|
|
|
remove_declarator(&Dc);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* This is weird. Due to the LR structure of the ANSI C grammar
|
|
|
|
we have to duplicate the actions of 'declarator' and
|
|
|
|
'abstract_declarator'. Calling these separately, as in
|
|
|
|
|
|
|
|
parameter_decl:
|
|
|
|
decl_specifiers
|
|
|
|
[
|
|
|
|
declarator
|
|
|
|
|
|
|
|
|
abstract_declarator
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
gives us a conflict on the terminals '(' and '*'. E.i. on
|
|
|
|
some input, it is impossible to decide which rule we take.
|
|
|
|
Combining the two declarators into one common declarator
|
|
|
|
is out of the question, since this results in an empty
|
|
|
|
string for the non-terminal 'declarator'.
|
|
|
|
So we combine the two only for the use of parameter_decl,
|
|
|
|
since this is the only place where they don't give
|
|
|
|
conflicts. However, this makes the grammar messy.
|
|
|
|
*/
|
|
|
|
parameter_declarator(register struct declarator *dc;)
|
|
|
|
{ struct formal *fm = NO_PARAMS;
|
|
|
|
struct proto *pl = NO_PROTO;
|
|
|
|
arith count;
|
|
|
|
int qual;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
primary_parameter_declarator(dc)
|
|
|
|
[
|
|
|
|
'('
|
|
|
|
[ %if (DOT != IDENTIFIER)
|
|
|
|
parameter_type_list(&pl)
|
|
|
|
|
|
|
|
|
formal_list(&fm)
|
|
|
|
|
|
|
|
|
/* empty */
|
|
|
|
]
|
|
|
|
')'
|
|
|
|
{ add_decl_unary(dc, FUNCTION, 0, (arith)0, fm, pl);
|
|
|
|
reject_params(dc);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
arrayer(&count)
|
|
|
|
{add_decl_unary(dc, ARRAY, 0, count, NO_PARAMS, NO_PROTO);}
|
|
|
|
]*
|
|
|
|
|
|
|
|
|
pointer(&qual) parameter_declarator(dc)
|
|
|
|
{add_decl_unary(dc, POINTER, qual, (arith)0, NO_PARAMS, NO_PROTO);}
|
|
|
|
;
|
|
|
|
|
|
|
|
primary_parameter_declarator(register struct declarator *dc;)
|
|
|
|
:
|
|
|
|
[%if (AHEAD == ')' || first_of_parameter_type_list(AHEAD)
|
|
|
|
&& (AHEAD != IDENTIFIER))
|
|
|
|
/* empty */
|
|
|
|
|
|
|
|
|
identifier(&dc->dc_idf)
|
|
|
|
|
|
|
|
|
'(' parameter_declarator(dc) ')'
|
|
|
|
]
|
|
|
|
;
|
|
|
|
|
|
|
|
pointer(int *qual;)
|
|
|
|
:
|
|
|
|
'*' type_qualifier_list(qual)
|
|
|
|
;
|
|
|
|
|
|
|
|
/* Type qualifiers may come in three flavours:
|
|
|
|
volatile, const, const volatile.
|
|
|
|
These all have different semantic properties:
|
|
|
|
|
|
|
|
volatile:
|
|
|
|
means that the object can be modified
|
|
|
|
without prior knowledge of the implementation.
|
|
|
|
|
|
|
|
const:
|
|
|
|
means that the object can not be modified; thus
|
|
|
|
it's illegal to use this as a l-value.
|
|
|
|
|
|
|
|
const volatile:
|
|
|
|
means that the object can be modified without
|
|
|
|
prior knowledge of the implementation, but may
|
|
|
|
not be used as a l-value.
|
|
|
|
*/
|
|
|
|
/* 3.5.4 */
|
|
|
|
type_qualifier_list(int *qual;)
|
|
|
|
:
|
|
|
|
{ *qual = 0; }
|
|
|
|
[
|
|
|
|
VOLATILE
|
|
|
|
{ if (*qual & TQ_VOLATILE)
|
|
|
|
error("repeated type qualifier");
|
|
|
|
*qual |= TQ_VOLATILE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
CONST
|
|
|
|
{ if (*qual & TQ_CONST)
|
|
|
|
error("repeated type qualifier");
|
|
|
|
*qual |= TQ_CONST;
|
|
|
|
}
|
|
|
|
]*
|
|
|
|
;
|
|
|
|
|
|
|
|
empty:
|
|
|
|
;
|