From f263066f616b096fe209abefe59fe335df72b655 Mon Sep 17 00:00:00 2001 From: Stephen Heumann Date: Thu, 6 Oct 2022 20:42:11 -0500 Subject: [PATCH] Give an error for declarations that do not declare anything. This enforces the constraint from C17 section 6.7 p2 that declarations "shall declare at least a declarator (other than the parameters of a function or the members of a structure or union), a tag, or the members of an enumeration." Somewhat relaxed rules are used for enums in the default loose type checking mode, similar to what GCC and Clang do. --- Parser.pas | 32 ++++++++++++++++++++++++++++---- Scanner.pas | 1 + 2 files changed, 29 insertions(+), 4 deletions(-) diff --git a/Parser.pas b/Parser.pas index 444557a..e6b3e3f 100644 --- a/Parser.pas +++ b/Parser.pas @@ -196,6 +196,7 @@ var compoundLiteralNumber: integer; {number of compound literal} compoundLiteralToAllocate: identPtr; {compound literal that needs space allocated} vaInfoLLN: integer; {label number of internal va info (0 for none)} + declaredTagOrEnumConst: boolean; {was a tag or enum const declared?} {parameter processing variables} {------------------------------} @@ -2755,6 +2756,8 @@ procedure DeclarationSpecifiers (var declSpecifiers: declSpecifiersRecord; { referencing a forward struct/union? } { skipDeclarator - for enum,struct,union with no } { declarator } +{ declaredTagOrEnumConst - set if a tag or an enum const } +{ is declared (otherwise unchanged) } label 1,2,3; @@ -3178,9 +3181,16 @@ while token.kind in allowedTokens do begin end {if} else if (variable <> nil) and (variable^.itype^.kind = enumType) then - goto 1 - else if not looseTypeChecks then - Error(171); + begin + if looseTypeChecks then + declaredTagOrEnumConst := true; + goto 1; + end {if} + else begin + declaredTagOrEnumConst := true; + if not looseTypeChecks then + Error(171); + end; {else} tPtr := pointer(Malloc(sizeof(typeRecord))); tPtr^.size := cgWordSize; tPtr^.saveDisp := 0; @@ -3193,6 +3203,7 @@ while token.kind in allowedTokens do begin Error(9); enumVal := 0; {set the default value} if token.kind = lbracech then begin + declaredTagOrEnumConst := true; NextToken; {skip the '{'} repeat {declare the enum constants} tPtr := pointer(Malloc(sizeof(typeRecord))); @@ -3263,6 +3274,10 @@ while token.kind in allowedTokens do begin structPtr := FindSymbol(token, tagSpace, true, true); ttoken := token; {record the structure name} NextToken; {skip the structure name} + if (token.kind = lbracech) or + ((token.kind = semicolonch) and (myDeclarationModifiers = [])) + then + declaredTagOrEnumConst := true; if structPtr = nil then begin {if the name hasn't been defined then...} if token.kind <> lbracech then if (token.kind <> semicolonch) or @@ -3289,19 +3304,23 @@ while token.kind in allowedTokens do begin structPtr := NewSymbol(ttoken.name, structTypePtr, ident, tagSpace, defined); structTypePtr^.sName := structPtr^.name; + declaredTagOrEnumConst := true; end; end {if} {the name has been defined, so...} else if structPtr^.itype^.kind <> tKind then begin Error(42); {it's an error if it's not a struct} + declaredTagOrEnumConst := true; {avoid extra errors} structPtr := nil; end {else} else begin {record the existing structure type} structTypePtr := structPtr^.itype; end; {else} end {if} - else if token.kind <> lbracech then + else if token.kind <> lbracech then begin Error(9); {its an error if there's no name or struct} + declaredTagOrEnumConst := true; {avoid extra errors} + end; {else if} 2: if token.kind = lbracech then {handle a structure definition...} begin {error if we already have one!} if (structTypePtr <> defaultStruct) @@ -3651,12 +3670,17 @@ inhibitHeader := true; {block imbedded includes in headers} lUseGlobalPool := useGlobalPool; {handle a TypeSpecifier/declarator} typeFound := token.kind in declarationSpecifiersElement; +declaredTagOrEnumConst := false; DeclarationSpecifiers(declSpecifiers, declarationSpecifiersElement, ident); isPascal := pascalsy in declSpecifiers.declarationModifiers; isAsm := asmsy in declSpecifiers.declarationModifiers; isInline := inlinesy in declSpecifiers.declarationModifiers; isNoreturn := _Noreturnsy in declSpecifiers.declarationModifiers; alignmentSpecified := _Alignassy in declSpecifiers.declarationModifiers; +if token.kind = semicolonch then + if not doingPrototypes then + if not declaredTagOrEnumConst then + Error(176); if not skipDeclarator then begin variable := nil; Declarator(declSpecifiers, variable, variableSpace, doingPrototypes); diff --git a/Scanner.pas b/Scanner.pas index fdc3b76..9362be7 100644 --- a/Scanner.pas +++ b/Scanner.pas @@ -759,6 +759,7 @@ if list or (numErr <> 0) then begin 173: msg := @'''#'' must be followed by a macro parameter'; 174: msg := @'''__VA_ARGS__'' may only be used in a variadic macro'; 175: msg := @'duplicate macro parameter name'; + 176: msg := @'declarator expected'; otherwise: Error(57); end; {case} writeln(msg^);