mirror of
https://github.com/byteworksinc/ORCA-C.git
synced 2025-01-04 17:31:49 +00:00
Eliminate global variables for declaration specifiers.
They are now represented in local structures instead. This keeps the representation of declaration specifiers together and eliminates the need for awkward and error-prone code to save and restore the global variables.
This commit is contained in:
parent
05ecf5eef3
commit
1fa3ec8fdd
@ -515,10 +515,6 @@ var
|
|||||||
isConstant: boolean; {is the initializer expression constant?}
|
isConstant: boolean; {is the initializer expression constant?}
|
||||||
expressionIsLongLong: boolean; {is the last constant expression long long?}
|
expressionIsLongLong: boolean; {is the last constant expression long long?}
|
||||||
|
|
||||||
{type specifier results}
|
|
||||||
{----------------------}
|
|
||||||
typeSpec: typePtr; {type specifier}
|
|
||||||
|
|
||||||
{flags}
|
{flags}
|
||||||
{-----}
|
{-----}
|
||||||
codegenStarted: boolean; {have we started the code generator?}
|
codegenStarted: boolean; {have we started the code generator?}
|
||||||
|
@ -250,12 +250,11 @@ procedure Match (kind: tokenEnum; err: integer); extern;
|
|||||||
{ err - error number if the expected token is not found }
|
{ err - error number if the expected token is not found }
|
||||||
|
|
||||||
|
|
||||||
procedure TypeName; extern;
|
function TypeName: typePtr; extern;
|
||||||
|
|
||||||
{ process a type name (used for casts and sizeof/_Alignof) }
|
{ process a type name (used for casts and sizeof/_Alignof) }
|
||||||
{ }
|
{ }
|
||||||
{ outputs: }
|
{ returns: a pointer to the type }
|
||||||
{ typeSpec - pointer to the type }
|
|
||||||
|
|
||||||
|
|
||||||
function MakeFuncIdentifier: identPtr; extern;
|
function MakeFuncIdentifier: identPtr; extern;
|
||||||
@ -756,6 +755,7 @@ var
|
|||||||
opStack: tokenPtr; {operation stack}
|
opStack: tokenPtr; {operation stack}
|
||||||
parenCount: integer; {# of open parenthesis}
|
parenCount: integer; {# of open parenthesis}
|
||||||
stack: tokenPtr; {operand stack}
|
stack: tokenPtr; {operand stack}
|
||||||
|
tType: typePtr; {type for cast/sizeof/etc.}
|
||||||
|
|
||||||
op,sp: tokenPtr; {work pointers}
|
op,sp: tokenPtr; {work pointers}
|
||||||
|
|
||||||
@ -2002,8 +2002,7 @@ var
|
|||||||
while not (token.kind in [colonch,commach,rparench,eofsy]) do
|
while not (token.kind in [colonch,commach,rparench,eofsy]) do
|
||||||
NextToken;
|
NextToken;
|
||||||
end; {if}
|
end; {if}
|
||||||
TypeName; {get the type name}
|
currentType := TypeName; {get the type name}
|
||||||
currentType := typeSpec;
|
|
||||||
if (currentType^.size = 0) or (currentType^.kind = functionType) then
|
if (currentType^.size = 0) or (currentType^.kind = functionType) then
|
||||||
Error(133);
|
Error(133);
|
||||||
tl := typesSeen; {check if it is a duplicate}
|
tl := typesSeen; {check if it is a duplicate}
|
||||||
@ -2214,7 +2213,7 @@ if token.kind in startExpression then begin
|
|||||||
doingSizeof := true
|
doingSizeof := true
|
||||||
else if opStack^.token.kind = _Alignofsy then
|
else if opStack^.token.kind = _Alignofsy then
|
||||||
doingAlignof := true;
|
doingAlignof := true;
|
||||||
TypeName;
|
tType := TypeName;
|
||||||
if doingSizeof or doingAlignof then begin
|
if doingSizeof or doingAlignof then begin
|
||||||
|
|
||||||
{handle a sizeof operator}
|
{handle a sizeof operator}
|
||||||
@ -2229,10 +2228,10 @@ if token.kind in startExpression then begin
|
|||||||
sp^.token.kind := ulongconst;
|
sp^.token.kind := ulongconst;
|
||||||
sp^.token.class := longConstant;
|
sp^.token.class := longConstant;
|
||||||
if doingSizeof then
|
if doingSizeof then
|
||||||
sp^.token.lval := typeSpec^.size
|
sp^.token.lval := tType^.size
|
||||||
else {if doingAlignof then}
|
else {if doingAlignof then}
|
||||||
sp^.token.lval := 1;
|
sp^.token.lval := 1;
|
||||||
with typeSpec^ do
|
with tType^ do
|
||||||
if (size = 0) or ((kind = arrayType) and (elements = 0)) then
|
if (size = 0) or ((kind = arrayType) and (elements = 0)) then
|
||||||
Error(133);
|
Error(133);
|
||||||
sp^.next := stack;
|
sp^.next := stack;
|
||||||
@ -2246,7 +2245,7 @@ if token.kind in startExpression then begin
|
|||||||
op^.left := nil;
|
op^.left := nil;
|
||||||
op^.middle := nil;
|
op^.middle := nil;
|
||||||
op^.right := nil;
|
op^.right := nil;
|
||||||
op^.castType := typeSpec;
|
op^.castType := tType;
|
||||||
op^.token.kind := castoper;
|
op^.token.kind := castoper;
|
||||||
op^.token.class := reservedWord;
|
op^.token.class := reservedWord;
|
||||||
op^.next := opStack;
|
op^.next := opStack;
|
||||||
|
159
Parser.pas
159
Parser.pas
@ -40,12 +40,11 @@ procedure DoStatement;
|
|||||||
{ process a statement from a function }
|
{ process a statement from a function }
|
||||||
|
|
||||||
|
|
||||||
procedure TypeName;
|
function TypeName: typePtr;
|
||||||
|
|
||||||
{ process a type name (used for casts and sizeof/_Alignof) }
|
{ process a type name (used for casts and sizeof/_Alignof) }
|
||||||
{ }
|
{ }
|
||||||
{ outputs: }
|
{ returns: a pointer to the type }
|
||||||
{ typeSpec - pointer to the type }
|
|
||||||
|
|
||||||
|
|
||||||
procedure AutoInit (variable: identPtr; line: integer;
|
procedure AutoInit (variable: identPtr; line: integer;
|
||||||
@ -172,6 +171,16 @@ type
|
|||||||
);
|
);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
{type info for a declaration}
|
||||||
|
{---------------------------}
|
||||||
|
declSpecifiersRecord = record
|
||||||
|
storageClass: tokenEnum; {storage class of the declaration}
|
||||||
|
typeSpec: typePtr; {type specifier}
|
||||||
|
declarationModifiers: tokenSet; {all storage class specifiers, type }
|
||||||
|
{qualifiers, function specifiers, & }
|
||||||
|
{alignment specifiers in declaration}
|
||||||
|
end;
|
||||||
|
|
||||||
var
|
var
|
||||||
firstCompoundStatement: boolean; {are we doing a function level compound statement?}
|
firstCompoundStatement: boolean; {are we doing a function level compound statement?}
|
||||||
fType: typePtr; {return type of the current function}
|
fType: typePtr; {return type of the current function}
|
||||||
@ -198,14 +207,6 @@ var
|
|||||||
protoType: typePtr; {type from a parameter list}
|
protoType: typePtr; {type from a parameter list}
|
||||||
protoVariable: identPtr; {variable from a parameter list}
|
protoVariable: identPtr; {variable from a parameter list}
|
||||||
|
|
||||||
{type info for the current declaration}
|
|
||||||
{-------------------------------------}
|
|
||||||
storageClass: tokenEnum; {storage class of the declaration}
|
|
||||||
{ typeSpec: typePtr; (in CCommon) {type specifier}
|
|
||||||
declarationModifiers: tokenSet; {all storage class specifiers, type }
|
|
||||||
{qualifiers, function specifiers, & }
|
|
||||||
{alignment specifiers in declaration}
|
|
||||||
|
|
||||||
{syntactic classes of tokens}
|
{syntactic classes of tokens}
|
||||||
{---------------------------}
|
{---------------------------}
|
||||||
{ specifierQualifierListElement: tokenSet; (in CCommon)}
|
{ specifierQualifierListElement: tokenSet; (in CCommon)}
|
||||||
@ -1256,13 +1257,13 @@ end; {EndWhileStatement}
|
|||||||
|
|
||||||
{-- Type declarations ------------------------------------------}
|
{-- Type declarations ------------------------------------------}
|
||||||
|
|
||||||
procedure Declarator(tPtr: typePtr; var variable: identPtr; space: spaceType;
|
procedure Declarator(declSpecifiers: declSpecifiersRecord;
|
||||||
doingPrototypes: boolean);
|
var variable: identPtr; space: spaceType; doingPrototypes: boolean);
|
||||||
|
|
||||||
{ handle a declarator }
|
{ handle a declarator }
|
||||||
{ }
|
{ }
|
||||||
{ parameters: }
|
{ parameters: }
|
||||||
{ tPtr - pointer to the type to use }
|
{ declSpecifiers - type/specifiers to use }
|
||||||
{ variable - pointer to variable being defined }
|
{ variable - pointer to variable being defined }
|
||||||
{ space - variable space to use }
|
{ space - variable space to use }
|
||||||
{ doingPrototypes - are we compiling prototype parameter }
|
{ doingPrototypes - are we compiling prototype parameter }
|
||||||
@ -1289,6 +1290,7 @@ var
|
|||||||
newName: stringPtr; {new symbol name}
|
newName: stringPtr; {new symbol name}
|
||||||
parameterStorage: boolean; {is the new symbol in a parm list?}
|
parameterStorage: boolean; {is the new symbol in a parm list?}
|
||||||
state: stateKind; {declaration state of the variable}
|
state: stateKind; {declaration state of the variable}
|
||||||
|
tPtr: typePtr; {type of declaration}
|
||||||
tPtr2: typePtr; {work pointer}
|
tPtr2: typePtr; {work pointer}
|
||||||
tsPtr: typeDefPtr; {work pointer}
|
tsPtr: typeDefPtr; {work pointer}
|
||||||
typeStack: typeDefPtr; {stack of type definitions}
|
typeStack: typeDefPtr; {stack of type definitions}
|
||||||
@ -1328,8 +1330,6 @@ var
|
|||||||
lisFunction: boolean; {local copy of isFunction}
|
lisFunction: boolean; {local copy of isFunction}
|
||||||
lisPascal: boolean; {local copy of isPascal}
|
lisPascal: boolean; {local copy of isPascal}
|
||||||
lLastParameter: identPtr; {next parameter to process}
|
lLastParameter: identPtr; {next parameter to process}
|
||||||
lstorageClass: tokenEnum; {storage class of the declaration}
|
|
||||||
ltypeSpec: typePtr; {type specifier}
|
|
||||||
luseGlobalPool: boolean; {local copy of useGlobalPool}
|
luseGlobalPool: boolean; {local copy of useGlobalPool}
|
||||||
lSuppressMacroExpansions: boolean;{local copy of suppressMacroExpansions}
|
lSuppressMacroExpansions: boolean;{local copy of suppressMacroExpansions}
|
||||||
|
|
||||||
@ -1347,12 +1347,12 @@ var
|
|||||||
variable := FindSymbol(token, space, true, true);
|
variable := FindSymbol(token, space, true, true);
|
||||||
newName := token.name;
|
newName := token.name;
|
||||||
if variable = nil then begin
|
if variable = nil then begin
|
||||||
if storageClass = typedefsy then begin
|
if declSpecifiers.storageClass = typedefsy then begin
|
||||||
tPtr2 := pointer(Calloc(sizeof(typeRecord)));
|
tPtr2 := pointer(Calloc(sizeof(typeRecord)));
|
||||||
{tPtr2^.size := 0;}
|
{tPtr2^.size := 0;}
|
||||||
{tPtr2^.saveDisp := 0;}
|
{tPtr2^.saveDisp := 0;}
|
||||||
tPtr2^.kind := definedType;
|
tPtr2^.kind := definedType;
|
||||||
{tPtr^.qualifiers := [];}
|
{tPtr2^.qualifiers := [];}
|
||||||
tPtr2^.dType := tPtr;
|
tPtr2^.dType := tPtr;
|
||||||
end {if}
|
end {if}
|
||||||
else
|
else
|
||||||
@ -1483,8 +1483,6 @@ var
|
|||||||
useGlobalPool := true;
|
useGlobalPool := true;
|
||||||
done2 := false;
|
done2 := false;
|
||||||
lisFunction := isFunction; {preserve global variables}
|
lisFunction := isFunction; {preserve global variables}
|
||||||
ltypeSpec := typeSpec;
|
|
||||||
lstorageClass := storageClass;
|
|
||||||
with tPtr2^ do begin
|
with tPtr2^ do begin
|
||||||
prototyped := true; {it is prototyped}
|
prototyped := true; {it is prototyped}
|
||||||
repeat {collect the declarations}
|
repeat {collect the declarations}
|
||||||
@ -1533,8 +1531,6 @@ var
|
|||||||
until done2;
|
until done2;
|
||||||
end; {with}
|
end; {with}
|
||||||
isFunction := lisFunction; {restore global variables}
|
isFunction := lisFunction; {restore global variables}
|
||||||
storageClass := lstorageClass;
|
|
||||||
typeSpec := ltypeSpec;
|
|
||||||
useGlobalPool := luseGlobalPool;
|
useGlobalPool := luseGlobalPool;
|
||||||
end {if prototype}
|
end {if prototype}
|
||||||
else if token.kind = ident then begin
|
else if token.kind = ident then begin
|
||||||
@ -1651,9 +1647,10 @@ var
|
|||||||
end; {StackDeclarations}
|
end; {StackDeclarations}
|
||||||
|
|
||||||
begin {Declarator}
|
begin {Declarator}
|
||||||
|
tPtr := declSpecifiers.typeSpec;
|
||||||
newName := nil; {no identifier, yet}
|
newName := nil; {no identifier, yet}
|
||||||
unnamedParm := false; {not an unnamed parameter}
|
unnamedParm := false; {not an unnamed parameter}
|
||||||
if storageClass = externsy then {decide on a storage state}
|
if declSpecifiers.storageClass = externsy then {decide on a storage state}
|
||||||
state := declared
|
state := declared
|
||||||
else
|
else
|
||||||
state := defined;
|
state := defined;
|
||||||
@ -1820,7 +1817,7 @@ if tPtr^.kind = functionType then begin {declare the identifier}
|
|||||||
if tPtr^.kind = functionType then
|
if tPtr^.kind = functionType then
|
||||||
state := declared;
|
state := declared;
|
||||||
if newName <> nil then {declare the variable}
|
if newName <> nil then {declare the variable}
|
||||||
variable := NewSymbol(newName, tPtr, storageClass, space, state)
|
variable := NewSymbol(newName, tPtr, declSpecifiers.storageClass, space, state)
|
||||||
else if unnamedParm then
|
else if unnamedParm then
|
||||||
variable^.itype := tPtr
|
variable^.itype := tPtr
|
||||||
else begin
|
else begin
|
||||||
@ -1842,7 +1839,8 @@ if variable <> nil then begin
|
|||||||
lastWasPointer := false;
|
lastWasPointer := false;
|
||||||
tPtr := tPtr^.pType;
|
tPtr := tPtr^.pType;
|
||||||
end; {while}
|
end; {while}
|
||||||
if ((tPtr <> typeSpec) and (not (tPtr^.kind in [structType,unionType])))
|
if ((tPtr <> declSpecifiers.typeSpec)
|
||||||
|
and (not (tPtr^.kind in [structType,unionType])))
|
||||||
then begin
|
then begin
|
||||||
Error(107);
|
Error(107);
|
||||||
SkipStatement;
|
SkipStatement;
|
||||||
@ -2746,12 +2744,13 @@ Match(semicolonch, 22);
|
|||||||
end; {DoStaticAssert}
|
end; {DoStaticAssert}
|
||||||
|
|
||||||
|
|
||||||
procedure DeclarationSpecifiers (allowedTokens: tokenSet;
|
procedure DeclarationSpecifiers (var declSpecifiers: declSpecifiersRecord;
|
||||||
expectedNext: tokenEnum);
|
allowedTokens: tokenSet; expectedNext: tokenEnum);
|
||||||
|
|
||||||
{ handle declaration specifiers or a specifier-qualifier list }
|
{ handle declaration specifiers or a specifier-qualifier list }
|
||||||
{ }
|
{ }
|
||||||
{ parameters: }
|
{ parameters: }
|
||||||
|
{ declSpecifiers - record to hold result type & specifiers}
|
||||||
{ allowedTokens - specifiers/qualifiers that can be used }
|
{ allowedTokens - specifiers/qualifiers that can be used }
|
||||||
{ expectedNext - token usually expected after declaration }
|
{ expectedNext - token usually expected after declaration }
|
||||||
{ specifiers (used for error messages only) }
|
{ specifiers (used for error messages only) }
|
||||||
@ -2761,10 +2760,6 @@ procedure DeclarationSpecifiers (allowedTokens: tokenSet;
|
|||||||
{ referencing a forward struct/union? }
|
{ referencing a forward struct/union? }
|
||||||
{ skipDeclarator - for enum,struct,union with no }
|
{ skipDeclarator - for enum,struct,union with no }
|
||||||
{ declarator }
|
{ declarator }
|
||||||
{ typespec - type specifier }
|
|
||||||
{ declarationModifiers - all storage class specifiers, }
|
|
||||||
{ type qualifiers, function specifiers, and }
|
|
||||||
{ alignment specifiers in this declaration }
|
|
||||||
|
|
||||||
label 1,2,3;
|
label 1,2,3;
|
||||||
|
|
||||||
@ -2791,6 +2786,7 @@ var
|
|||||||
mySkipDeclarator: boolean; {value of skipDeclarator to generate}
|
mySkipDeclarator: boolean; {value of skipDeclarator to generate}
|
||||||
myTypeSpec: typePtr; {value of typeSpec to generate}
|
myTypeSpec: typePtr; {value of typeSpec to generate}
|
||||||
myDeclarationModifiers: tokenSet; {all modifiers in this declaration}
|
myDeclarationModifiers: tokenSet; {all modifiers in this declaration}
|
||||||
|
myStorageClass: tokenEnum; {storage class}
|
||||||
|
|
||||||
isLongLong: boolean; {is this a "long long" type?}
|
isLongLong: boolean; {is this a "long long" type?}
|
||||||
|
|
||||||
@ -2810,18 +2806,15 @@ var
|
|||||||
fl,tfl,ufl: identPtr; {field list}
|
fl,tfl,ufl: identPtr; {field list}
|
||||||
ldoingParameters: boolean; {local copy of doingParameters}
|
ldoingParameters: boolean; {local copy of doingParameters}
|
||||||
lisForwardDeclared: boolean; {local copy of isForwardDeclared}
|
lisForwardDeclared: boolean; {local copy of isForwardDeclared}
|
||||||
lstorageClass: tokenEnum; {storage class of the declaration}
|
|
||||||
maxDisp: longint; {for determining union sizes}
|
maxDisp: longint; {for determining union sizes}
|
||||||
variable: identPtr; {variable being defined}
|
variable: identPtr; {variable being defined}
|
||||||
didFlexibleArray: boolean; {have we seen a flexible array member?}
|
didFlexibleArray: boolean; {have we seen a flexible array member?}
|
||||||
alignmentSpecified: boolean; {was alignment explicitly specified?}
|
fieldDeclSpecifiers: declSpecifiersRecord; {decl specifiers for field}
|
||||||
|
|
||||||
begin {FieldList}
|
begin {FieldList}
|
||||||
ldoingParameters := doingParameters; {allow fields in K&R dec. area}
|
ldoingParameters := doingParameters; {allow fields in K&R dec. area}
|
||||||
doingParameters := false;
|
doingParameters := false;
|
||||||
lisForwardDeclared := isForwardDeclared; {stack this value}
|
lisForwardDeclared := isForwardDeclared; {stack this value}
|
||||||
lStorageClass := storageClass; {don't allow auto in a struct}
|
|
||||||
storageClass := ident;
|
|
||||||
bitDisp := 0; {start allocation from byte 0}
|
bitDisp := 0; {start allocation from byte 0}
|
||||||
disp := 0;
|
disp := 0;
|
||||||
maxDisp := 0;
|
maxDisp := 0;
|
||||||
@ -2833,15 +2826,14 @@ var
|
|||||||
DoStaticAssert;
|
DoStaticAssert;
|
||||||
goto 1;
|
goto 1;
|
||||||
end; {if}
|
end; {if}
|
||||||
DeclarationSpecifiers(specifierQualifierListElement, ident);
|
DeclarationSpecifiers(fieldDeclSpecifiers, specifierQualifierListElement, ident);
|
||||||
alignmentSpecified := _Alignassy in declarationModifiers;
|
|
||||||
if not skipDeclarator then
|
if not skipDeclarator then
|
||||||
repeat {declare the variables...}
|
repeat {declare the variables...}
|
||||||
if didFlexibleArray then
|
if didFlexibleArray then
|
||||||
Error(118);
|
Error(118);
|
||||||
variable := nil;
|
variable := nil;
|
||||||
if token.kind <> colonch then begin
|
if token.kind <> colonch then begin
|
||||||
Declarator(typeSpec, variable, fieldListSpace, false);
|
Declarator(fieldDeclSpecifiers, variable, fieldListSpace, false);
|
||||||
if variable <> nil then {enter the var in the field list}
|
if variable <> nil then {enter the var in the field list}
|
||||||
begin
|
begin
|
||||||
tfl := fl; {(check for dups)}
|
tfl := fl; {(check for dups)}
|
||||||
@ -2882,7 +2874,7 @@ var
|
|||||||
tPtr := variable^.itype;
|
tPtr := variable^.itype;
|
||||||
end {if}
|
end {if}
|
||||||
else
|
else
|
||||||
tPtr := typeSpec;
|
tPtr := fieldDeclSpecifiers.typeSpec;
|
||||||
bitdisp := bitdisp+long(expressionValue).lsw;
|
bitdisp := bitdisp+long(expressionValue).lsw;
|
||||||
if kind = unionType then
|
if kind = unionType then
|
||||||
if ((bitDisp+7) div 8) > maxDisp then
|
if ((bitDisp+7) div 8) > maxDisp then
|
||||||
@ -2893,7 +2885,7 @@ var
|
|||||||
or (expressionValue > tPtr^.size*8)
|
or (expressionValue > tPtr^.size*8)
|
||||||
or ((expressionValue > 1) and (tPtr^.cType = ctBool)) then
|
or ((expressionValue > 1) and (tPtr^.cType = ctBool)) then
|
||||||
Error(115);
|
Error(115);
|
||||||
if alignmentSpecified then
|
if _Alignassy in fieldDeclSpecifiers.declarationModifiers then
|
||||||
Error(142);
|
Error(142);
|
||||||
end {if}
|
end {if}
|
||||||
else if variable <> nil then begin
|
else if variable <> nil then begin
|
||||||
@ -2922,7 +2914,7 @@ var
|
|||||||
if variable <> nil then {check for a const member}
|
if variable <> nil then {check for a const member}
|
||||||
tPtr := variable^.itype
|
tPtr := variable^.itype
|
||||||
else
|
else
|
||||||
tPtr := typeSpec;
|
tPtr := fieldDeclSpecifiers.typeSpec;
|
||||||
while tPtr^.kind in [definedType,arrayType] do begin
|
while tPtr^.kind in [definedType,arrayType] do begin
|
||||||
if tqConst in tPtr^.qualifiers then
|
if tqConst in tPtr^.qualifiers then
|
||||||
tp^.constMember := true;
|
tp^.constMember := true;
|
||||||
@ -2974,7 +2966,6 @@ var
|
|||||||
end {if}
|
end {if}
|
||||||
else
|
else
|
||||||
Error(26); {error if no named declarations}
|
Error(26); {error if no named declarations}
|
||||||
storageClass := lStorageClass; {restore default storage class}
|
|
||||||
isForwardDeclared := lisForwardDeclared; {restore the forward flag}
|
isForwardDeclared := lisForwardDeclared; {restore the forward flag}
|
||||||
doingParameters := ldoingParameters; {restore the parameters flag}
|
doingParameters := ldoingParameters; {restore the parameters flag}
|
||||||
end; {FieldList}
|
end; {FieldList}
|
||||||
@ -3055,6 +3046,7 @@ myTypeSpec := nil;
|
|||||||
myIsForwardDeclared := false; {not doing a forward reference (yet)}
|
myIsForwardDeclared := false; {not doing a forward reference (yet)}
|
||||||
mySkipDeclarator := false; {declarations are required (so far)}
|
mySkipDeclarator := false; {declarations are required (so far)}
|
||||||
myDeclarationModifiers := [];
|
myDeclarationModifiers := [];
|
||||||
|
myStorageClass := ident;
|
||||||
typeQualifiers := [];
|
typeQualifiers := [];
|
||||||
typeSpecifiers := [];
|
typeSpecifiers := [];
|
||||||
typeDone := false;
|
typeDone := false;
|
||||||
@ -3064,13 +3056,13 @@ while token.kind in allowedTokens do begin
|
|||||||
{storage class specifiers}
|
{storage class specifiers}
|
||||||
autosy,externsy,registersy,staticsy,typedefsy: begin
|
autosy,externsy,registersy,staticsy,typedefsy: begin
|
||||||
myDeclarationModifiers := myDeclarationModifiers + [token.kind];
|
myDeclarationModifiers := myDeclarationModifiers + [token.kind];
|
||||||
if storageClass <> ident then begin
|
if myStorageClass <> ident then begin
|
||||||
if typeDone or (typeSpecifiers <> []) then
|
if typeDone or (typeSpecifiers <> []) then
|
||||||
UnexpectedTokenError(expectedNext)
|
UnexpectedTokenError(expectedNext)
|
||||||
else
|
else
|
||||||
Error(26);
|
Error(26);
|
||||||
end; {if}
|
end; {if}
|
||||||
storageClass := token.kind;
|
myStorageClass := token.kind;
|
||||||
if not doingFunction then
|
if not doingFunction then
|
||||||
if token.kind = autosy then
|
if token.kind = autosy then
|
||||||
Error(62);
|
Error(62);
|
||||||
@ -3078,7 +3070,7 @@ while token.kind in allowedTokens do begin
|
|||||||
if token.kind <> registersy then
|
if token.kind <> registersy then
|
||||||
Error(87);
|
Error(87);
|
||||||
end {if}
|
end {if}
|
||||||
else if storageClass in [staticsy,typedefsy] then begin
|
else if myStorageClass in [staticsy,typedefsy] then begin
|
||||||
{Error if we may have allocated type info in local pool.}
|
{Error if we may have allocated type info in local pool.}
|
||||||
{This should not come up with current use of MM pools. }
|
{This should not come up with current use of MM pools. }
|
||||||
if not useGlobalPool then
|
if not useGlobalPool then
|
||||||
@ -3087,7 +3079,7 @@ while token.kind in allowedTokens do begin
|
|||||||
useGlobalPool := true;
|
useGlobalPool := true;
|
||||||
end; {else if}
|
end; {else if}
|
||||||
if doingForLoopClause1 then
|
if doingForLoopClause1 then
|
||||||
if not (storageClass in [autosy,registersy]) then
|
if not (myStorageClass in [autosy,registersy]) then
|
||||||
Error(127);
|
Error(127);
|
||||||
NextToken;
|
NextToken;
|
||||||
end;
|
end;
|
||||||
@ -3137,8 +3129,7 @@ while token.kind in allowedTokens do begin
|
|||||||
if typeDone or (typeSpecifiers <> []) then
|
if typeDone or (typeSpecifiers <> []) then
|
||||||
UnexpectedTokenError(expectedNext);
|
UnexpectedTokenError(expectedNext);
|
||||||
NextToken;
|
NextToken;
|
||||||
TypeName;
|
myTypeSpec := TypeName;
|
||||||
myTypeSpec := typeSpec;
|
|
||||||
Match(rparench, 12);
|
Match(rparench, 12);
|
||||||
end; {if}
|
end; {if}
|
||||||
typeDone := true;
|
typeDone := true;
|
||||||
@ -3370,8 +3361,8 @@ while token.kind in allowedTokens do begin
|
|||||||
NextToken;
|
NextToken;
|
||||||
Match(lparench, 13);
|
Match(lparench, 13);
|
||||||
if token.kind in specifierQualifierListElement then begin
|
if token.kind in specifierQualifierListElement then begin
|
||||||
TypeName;
|
tPtr := TypeName;
|
||||||
with typeSpec^ do
|
with tPtr^ do
|
||||||
if (size = 0) or ((kind = arrayType) and (elements = 0)) then
|
if (size = 0) or ((kind = arrayType) and (elements = 0)) then
|
||||||
Error(133);
|
Error(133);
|
||||||
end {if}
|
end {if}
|
||||||
@ -3392,13 +3383,15 @@ while token.kind in allowedTokens do begin
|
|||||||
3:
|
3:
|
||||||
isForwardDeclared := myIsForwardDeclared;
|
isForwardDeclared := myIsForwardDeclared;
|
||||||
skipDeclarator := mySkipDeclarator;
|
skipDeclarator := mySkipDeclarator;
|
||||||
declarationModifiers := myDeclarationModifiers;
|
declSpecifiers.declarationModifiers := myDeclarationModifiers;
|
||||||
if myTypeSpec = nil then begin
|
if myTypeSpec = nil then begin
|
||||||
myTypeSpec := intPtr; {under C89, default type is int}
|
myTypeSpec := intPtr; {under C89, default type is int}
|
||||||
if (lint & lintC99Syntax) <> 0 then
|
if (lint & lintC99Syntax) <> 0 then
|
||||||
Error(151);
|
Error(151);
|
||||||
end; {if}
|
end; {if}
|
||||||
typeSpec := MakeQualifiedType(myTypeSpec, typeQualifiers); {apply type qualifiers}
|
declSpecifiers.typeSpec := {apply type qualifiers}
|
||||||
|
MakeQualifiedType(myTypeSpec, typeQualifiers);
|
||||||
|
declSpecifiers.storageClass := myStorageClass;
|
||||||
end; {DeclarationSpecifiers}
|
end; {DeclarationSpecifiers}
|
||||||
|
|
||||||
|
|
||||||
@ -3424,7 +3417,6 @@ var
|
|||||||
lDoingParameters: boolean; {local copy of doingParameters}
|
lDoingParameters: boolean; {local copy of doingParameters}
|
||||||
lisPascal: boolean; {local copy of isPascal}
|
lisPascal: boolean; {local copy of isPascal}
|
||||||
lp,tlp,tlp2: identPtr; {for tracing parameter list}
|
lp,tlp,tlp2: identPtr; {for tracing parameter list}
|
||||||
ltypeSpec: typePtr; {copy of type specifier}
|
|
||||||
lUseGlobalPool: boolean; {local copy of useGlobalPool}
|
lUseGlobalPool: boolean; {local copy of useGlobalPool}
|
||||||
nextPdisp: integer; {for calculating parameter disps}
|
nextPdisp: integer; {for calculating parameter disps}
|
||||||
noFDefinitions: boolean; {are function definitions inhibited?}
|
noFDefinitions: boolean; {are function definitions inhibited?}
|
||||||
@ -3436,6 +3428,7 @@ var
|
|||||||
tk: tokenType; {work token}
|
tk: tokenType; {work token}
|
||||||
typeFound: boolean; {has some type specifier been found?}
|
typeFound: boolean; {has some type specifier been found?}
|
||||||
startLine: integer; {line where this declaration starts}
|
startLine: integer; {line where this declaration starts}
|
||||||
|
declSpecifiers: declSpecifiersRecord; {type & specifiers for the declaration}
|
||||||
|
|
||||||
|
|
||||||
procedure CheckArray (v: identPtr; firstVariable: boolean);
|
procedure CheckArray (v: identPtr; firstVariable: boolean);
|
||||||
@ -3659,19 +3652,18 @@ if not doingFunction then {handle any segment statements}
|
|||||||
SegmentStatement;
|
SegmentStatement;
|
||||||
inhibitHeader := true; {block imbedded includes in headers}
|
inhibitHeader := true; {block imbedded includes in headers}
|
||||||
lUseGlobalPool := useGlobalPool;
|
lUseGlobalPool := useGlobalPool;
|
||||||
storageClass := ident;
|
|
||||||
{handle a TypeSpecifier/declarator}
|
{handle a TypeSpecifier/declarator}
|
||||||
typeFound := token.kind in declarationSpecifiersElement;
|
typeFound := token.kind in declarationSpecifiersElement;
|
||||||
DeclarationSpecifiers(declarationSpecifiersElement, ident);
|
DeclarationSpecifiers(declSpecifiers, declarationSpecifiersElement, ident);
|
||||||
isPascal := pascalsy in declarationModifiers;
|
isPascal := pascalsy in declSpecifiers.declarationModifiers;
|
||||||
isAsm := asmsy in declarationModifiers;
|
isAsm := asmsy in declSpecifiers.declarationModifiers;
|
||||||
isInline := inlinesy in declarationModifiers;
|
isInline := inlinesy in declSpecifiers.declarationModifiers;
|
||||||
isNoreturn := _Noreturnsy in declarationModifiers;
|
isNoreturn := _Noreturnsy in declSpecifiers.declarationModifiers;
|
||||||
alignmentSpecified := _Alignassy in declarationModifiers;
|
alignmentSpecified := _Alignassy in declSpecifiers.declarationModifiers;
|
||||||
lisPascal := isPascal;
|
lisPascal := isPascal;
|
||||||
if not skipDeclarator then begin
|
if not skipDeclarator then begin
|
||||||
variable := nil;
|
variable := nil;
|
||||||
Declarator(typeSpec, variable, variableSpace, doingPrototypes);
|
Declarator(declSpecifiers, variable, variableSpace, doingPrototypes);
|
||||||
if variable = nil then begin
|
if variable = nil then begin
|
||||||
inhibitHeader := false;
|
inhibitHeader := false;
|
||||||
if token.kind = semicolonch then
|
if token.kind = semicolonch then
|
||||||
@ -3715,7 +3707,7 @@ if isFunction then begin
|
|||||||
goto 1;
|
goto 1;
|
||||||
end; {if}
|
end; {if}
|
||||||
if isInline then
|
if isInline then
|
||||||
if storageClass <> staticsy then
|
if declSpecifiers.storageClass <> staticsy then
|
||||||
Error(120);
|
Error(120);
|
||||||
if alignmentSpecified then
|
if alignmentSpecified then
|
||||||
Error(142);
|
Error(142);
|
||||||
@ -3777,7 +3769,7 @@ if isFunction then begin
|
|||||||
NextToken; {allow further declarations}
|
NextToken; {allow further declarations}
|
||||||
variable := nil;
|
variable := nil;
|
||||||
isFunction := false;
|
isFunction := false;
|
||||||
Declarator (typeSpec, variable, variableSpace, doingPrototypes);
|
Declarator(declSpecifiers, variable, variableSpace, doingPrototypes);
|
||||||
if variable = nil then begin
|
if variable = nil then begin
|
||||||
inhibitHeader := false;
|
inhibitHeader := false;
|
||||||
if token.kind = semicolonch then
|
if token.kind = semicolonch then
|
||||||
@ -3980,7 +3972,7 @@ if isFunction then begin
|
|||||||
else {if not isFunction then} begin
|
else {if not isFunction then} begin
|
||||||
noFDefinitions := true;
|
noFDefinitions := true;
|
||||||
if alignmentSpecified then
|
if alignmentSpecified then
|
||||||
if storageClass in [typedefsy,registersy] then
|
if declSpecifiers.storageClass in [typedefsy,registersy] then
|
||||||
Error(142);
|
Error(142);
|
||||||
if not SkipDeclarator then
|
if not SkipDeclarator then
|
||||||
repeat
|
repeat
|
||||||
@ -3991,7 +3983,7 @@ else {if not isFunction then} begin
|
|||||||
if isNoreturn then
|
if isNoreturn then
|
||||||
Error(141);
|
Error(141);
|
||||||
if token.kind = eqch then begin
|
if token.kind = eqch then begin
|
||||||
if storageClass = typedefsy then
|
if declSpecifiers.storageClass = typedefsy then
|
||||||
Error(52);
|
Error(52);
|
||||||
if doingPrototypes then
|
if doingPrototypes then
|
||||||
Error(88);
|
Error(88);
|
||||||
@ -4004,14 +3996,13 @@ else {if not isFunction then} begin
|
|||||||
end;
|
end;
|
||||||
TermHeader; {make sure the header file is closed}
|
TermHeader; {make sure the header file is closed}
|
||||||
NextToken; {handle an initializer}
|
NextToken; {handle an initializer}
|
||||||
ltypeSpec := typeSpec;
|
|
||||||
Initializer(variable);
|
Initializer(variable);
|
||||||
typeSpec := ltypeSpec;
|
|
||||||
end; {if}
|
end; {if}
|
||||||
{check to insure array sizes are specified}
|
{check to insure array sizes are specified}
|
||||||
if storageClass <> typedefsy then
|
if declSpecifiers.storageClass <> typedefsy then
|
||||||
CheckArray(variable,
|
CheckArray(variable,
|
||||||
(storageClass = externsy) or doingParameters or not doingFunction);
|
(declSpecifiers.storageClass = externsy)
|
||||||
|
or doingParameters or not doingFunction);
|
||||||
{allocate space}
|
{allocate space}
|
||||||
if variable^.storage = stackFrame then begin
|
if variable^.storage = stackFrame then begin
|
||||||
variable^.lln := GetLocalLabel;
|
variable^.lln := GetLocalLabel;
|
||||||
@ -4023,7 +4014,7 @@ else {if not isFunction then} begin
|
|||||||
done := false; {allow multiple variables on one line}
|
done := false; {allow multiple variables on one line}
|
||||||
NextToken;
|
NextToken;
|
||||||
variable := nil;
|
variable := nil;
|
||||||
Declarator(typeSpec, variable, variableSpace, doingPrototypes);
|
Declarator(declSpecifiers, variable, variableSpace, doingPrototypes);
|
||||||
if variable = nil then begin
|
if variable = nil then begin
|
||||||
if token.kind = semicolonch then
|
if token.kind = semicolonch then
|
||||||
NextToken
|
NextToken
|
||||||
@ -4041,7 +4032,7 @@ else {if not isFunction then} begin
|
|||||||
if doingPrototypes then begin
|
if doingPrototypes then begin
|
||||||
protoVariable := variable; {make the var available to Declarator}
|
protoVariable := variable; {make the var available to Declarator}
|
||||||
if protoVariable = nil then
|
if protoVariable = nil then
|
||||||
protoType := typeSpec
|
protoType := declSpecifiers.typeSpec
|
||||||
else
|
else
|
||||||
protoType := protoVariable^.iType;
|
protoType := protoVariable^.iType;
|
||||||
end {if}
|
end {if}
|
||||||
@ -4063,16 +4054,15 @@ inhibitHeader := false;
|
|||||||
end; {DoDeclaration}
|
end; {DoDeclaration}
|
||||||
|
|
||||||
|
|
||||||
procedure TypeName;
|
function TypeName{: typePtr};
|
||||||
|
|
||||||
{ process a type name (used for casts and sizeof/_Alignof) }
|
{ process a type name (used for casts and sizeof/_Alignof) }
|
||||||
{ }
|
{ }
|
||||||
{ outputs: }
|
{ returns: a pointer to the type }
|
||||||
{ typeSpec - pointer to the type }
|
|
||||||
|
|
||||||
var
|
var
|
||||||
tl,tp: typePtr; {for creating/reversing the type list}
|
tl,tp: typePtr; {for creating/reversing the type list}
|
||||||
isPascal: boolean; {is the type "pascal" qualified?}
|
declSpecifiers: declSpecifiersRecord; {type & specifiers for the type name}
|
||||||
|
|
||||||
|
|
||||||
procedure AbstractDeclarator;
|
procedure AbstractDeclarator;
|
||||||
@ -4097,7 +4087,6 @@ var
|
|||||||
var
|
var
|
||||||
pcount: integer; {paren counter}
|
pcount: integer; {paren counter}
|
||||||
tp: typePtr; {work pointer}
|
tp: typePtr; {work pointer}
|
||||||
ltypeSpec: typePtr; {copy of type specifier}
|
|
||||||
|
|
||||||
begin {NonEmptyAbstractDeclarator}
|
begin {NonEmptyAbstractDeclarator}
|
||||||
if token.kind = lparench then begin
|
if token.kind = lparench then begin
|
||||||
@ -4167,9 +4156,7 @@ var
|
|||||||
if token.kind = rbrackch then
|
if token.kind = rbrackch then
|
||||||
expressionValue := 0
|
expressionValue := 0
|
||||||
else begin
|
else begin
|
||||||
ltypeSpec := typeSpec;
|
|
||||||
Expression(arrayExpression, [rbrackch]);
|
Expression(arrayExpression, [rbrackch]);
|
||||||
typeSpec := ltypeSpec;
|
|
||||||
if expressionValue <= 0 then begin
|
if expressionValue <= 0 then begin
|
||||||
Error(45);
|
Error(45);
|
||||||
expressionValue := 1;
|
expressionValue := 1;
|
||||||
@ -4221,12 +4208,11 @@ var
|
|||||||
|
|
||||||
begin {TypeName}
|
begin {TypeName}
|
||||||
{read and process the type specifier}
|
{read and process the type specifier}
|
||||||
DeclarationSpecifiers(specifierQualifierListElement, rparench);
|
DeclarationSpecifiers(declSpecifiers, specifierQualifierListElement, rparench);
|
||||||
isPascal := pascalsy in declarationModifiers;
|
|
||||||
|
|
||||||
{_Alignas is not allowed in most uses of type names. }
|
{_Alignas is not allowed in most uses of type names. }
|
||||||
{TODO: _Alignas should be allowed in compound literals. }
|
{TODO: _Alignas should be allowed in compound literals. }
|
||||||
if _Alignassy in declarationModifiers then
|
if _Alignassy in declSpecifiers.declarationModifiers then
|
||||||
Error(142);
|
Error(142);
|
||||||
|
|
||||||
{handle the abstract-declarator part}
|
{handle the abstract-declarator part}
|
||||||
@ -4234,14 +4220,15 @@ tl := nil; {no types so far}
|
|||||||
AbstractDeclarator; {create the type list}
|
AbstractDeclarator; {create the type list}
|
||||||
while tl <> nil do begin {reverse the list & compute array sizes}
|
while tl <> nil do begin {reverse the list & compute array sizes}
|
||||||
tp := tl^.aType; {NOTE: assumes aType, pType and fType overlap in typeRecord}
|
tp := tl^.aType; {NOTE: assumes aType, pType and fType overlap in typeRecord}
|
||||||
tl^.aType := typeSpec;
|
tl^.aType := declSpecifiers.typeSpec;
|
||||||
if tl^.kind = arrayType then
|
if tl^.kind = arrayType then
|
||||||
tl^.size := tl^.elements * typeSpec^.size;
|
tl^.size := tl^.elements * declSpecifiers.typeSpec^.size;
|
||||||
typeSpec := tl;
|
declSpecifiers.typeSpec := tl;
|
||||||
tl := tp;
|
tl := tp;
|
||||||
end; {while}
|
end; {while}
|
||||||
if isPascal then
|
if pascalsy in declSpecifiers.declarationModifiers then
|
||||||
typeSpec := MakePascalType(typeSpec);
|
declSpecifiers.typeSpec := MakePascalType(declSpecifiers.typeSpec);
|
||||||
|
TypeName := declSpecifiers.typeSpec;
|
||||||
end; {TypeName}
|
end; {TypeName}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user