1
0
mirror of https://github.com/cc65/cc65.git synced 2025-01-10 19:29:45 +00:00

Merge pull request #2307 from acqn/FuncDeclFix

[cc65] Fixed function declarator parser when a parameter has a function type
This commit is contained in:
Bob Andrews 2023-12-31 19:11:25 +01:00 committed by GitHub
commit 0ece9449d7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 100 additions and 33 deletions

View File

@ -206,6 +206,14 @@ static void Parse (void)
} else { } else {
/* Just a declaration */ /* Just a declaration */
Decl.StorageClass |= SC_DECL; Decl.StorageClass |= SC_DECL;
FuncDef = GetFuncDesc (Decl.Type);
if ((FuncDef->Flags & (FD_EMPTY | FD_OLDSTYLE)) == FD_OLDSTYLE) {
/* A parameter list without types is only allowed in a
** function definition.
*/
Error ("Parameter names without types in function declaration");
}
} }
} }

View File

@ -1989,7 +1989,7 @@ static void ParseAnsiParamList (FuncDesc* F)
static FuncDesc* ParseFuncDecl (void) static void ParseFuncDecl (Declarator* D, declmode_t Mode, TypeCode Qualifiers)
/* Parse the argument list of a function with the enclosing parentheses */ /* Parse the argument list of a function with the enclosing parentheses */
{ {
/* Create a new function descriptor */ /* Create a new function descriptor */
@ -2009,14 +2009,17 @@ static FuncDesc* ParseFuncDecl (void)
/* Parameter list declared as void */ /* Parameter list declared as void */
NextToken (); NextToken ();
F->Flags |= FD_VOID_PARAM; F->Flags |= FD_VOID_PARAM;
} else if (CurTok.Tok == TOK_IDENT && } else if (Mode != DM_NO_IDENT &&
CurTok.Tok == TOK_IDENT &&
(NextTok.Tok == TOK_COMMA || NextTok.Tok == TOK_RPAREN)) { (NextTok.Tok == TOK_COMMA || NextTok.Tok == TOK_RPAREN)) {
/* If the identifier is a typedef, we have a new-style parameter list; /* If the identifier is a typedef, we have a new-style parameter list;
** if it's some other identifier, it's an old-style parameter list. ** if it's some other identifier, it's an old-style parameter list.
** Note: Non-empty Old-style (K&R) parameter list is not allowed in
** type names.
*/ */
SymEntry* Sym = FindSym (CurTok.Ident); SymEntry* Sym = FindSym (CurTok.Ident);
if (Sym == 0 || !SymIsTypeDef (Sym)) { if (Sym == 0 || !SymIsTypeDef (Sym)) {
/* Old-style (K&R) function. */ /* Old-style (K&R) function */
F->Flags |= FD_OLDSTYLE; F->Flags |= FD_OLDSTYLE;
} }
} }
@ -2026,13 +2029,20 @@ static FuncDesc* ParseFuncDecl (void)
if ((F->Flags & FD_OLDSTYLE) == 0) { if ((F->Flags & FD_OLDSTYLE) == 0) {
/* New-style function */ /* New-style function */
ParseAnsiParamList (F); ParseAnsiParamList (F);
ConsumeRParen ();
} else { } else {
/* Old-style function */ /* Old-style function */
ParseOldStyleParamList (F); ParseOldStyleParamList (F);
ConsumeRParen (); }
if (!ConsumeRParen ()) {
/* Try some smart error recovery */
SimpleErrorSkip ();
NextToken ();
} else if (Mode == DM_IDENT_OR_EMPTY && (F->Flags & FD_OLDSTYLE) != 0) {
/* Parameter declaration list is only allowed in function definitions */
ParseOldStyleParamDeclList (F); ParseOldStyleParamDeclList (F);
} }
PopLexicalLevel (); PopLexicalLevel ();
/* Remember the last function parameter. We need it later for several /* Remember the last function parameter. We need it later for several
@ -2040,18 +2050,27 @@ static FuncDesc* ParseFuncDecl (void)
** more symbols are added to the table, it is easier if we remember it ** more symbols are added to the table, it is easier if we remember it
** now, since it is currently the last entry in the symbol table. ** now, since it is currently the last entry in the symbol table.
*/ */
F->LastParam = GetSymTab()->SymTail; F->LastParam = GetSymTab ()->SymTail;
/* Leave the lexical level remembering the symbol tables */
RememberFunctionLevel (F);
/* It is allowed to use incomplete types in function prototypes, so we /* It is allowed to use incomplete types in function prototypes, so we
** won't always get to know the parameter sizes here and may do that later. ** won't always get to know the parameter sizes here and may do that later.
*/ */
F->Flags |= FD_INCOMPLETE_PARAM; F->Flags |= FD_INCOMPLETE_PARAM;
/* Leave the lexical level remembering the symbol tables */ /* We cannot specify fastcall for variadic functions */
RememberFunctionLevel (F); if ((F->Flags & FD_VARIADIC) && (Qualifiers & T_QUAL_FASTCALL)) {
Error ("Variadic functions cannot be __fastcall__");
Qualifiers &= ~T_QUAL_FASTCALL;
}
/* Return the function descriptor */ /* Add the function type. Be sure to bounds check the type buffer */
return F; NeedTypeSpace (D, 1);
D->Type[D->Index].C = T_FUNC | Qualifiers;
D->Type[D->Index].A.F = F;
++D->Index;
} }
@ -2090,15 +2109,37 @@ static void DirectDecl (DeclSpec* Spec, Declarator* D, declmode_t Mode)
} }
if (CurTok.Tok == TOK_LPAREN) { if (CurTok.Tok == TOK_LPAREN) {
NextToken (); SymEntry* Entry;
/* An empty declaration cannot contain parentheses where an identifier /* An empty declaration cannot contain parentheses where an identifier
** would show up if it were a non-empty declaration. ** would show up if it were a non-empty declaration.
*/ */
if (Mode == DM_IDENT_OR_EMPTY) { if (Mode == DM_IDENT_OR_EMPTY) {
Spec->Flags |= DS_NO_EMPTY_DECL; Spec->Flags |= DS_NO_EMPTY_DECL;
} }
DirectDecl (Spec, D, Mode);
ConsumeRParen (); /* We have to disambiguate the meanings of 'type (identifier' when
** the identifier can be a typedef'ed parameter type specifier or
** a declarator enclosed in parentheses in some cases.
*/
if (Mode == DM_IDENT_OR_EMPTY || /* If we are in a declaration... */
NextTok.Tok == TOK_LPAREN || /* or the next token is one more paren... */
NextTok.Tok == TOK_STAR || /* or a '*' ... */
(NextTok.Tok == TOK_IDENT && /* or an identifier that... */
((Entry = FindSym (NextTok.Ident)) == 0 || /* is not a typedef. */
!SymIsTypeDef (Entry)))) {
/* Parse the direct declarator in parentheses */
NextToken ();
DirectDecl (Spec, D, Mode);
ConsumeRParen ();
} else {
/* This is a parameter type list in parentheses */
ParseFuncDecl (D, Mode, Qualifiers);
/* Qualifiers now used */
Qualifiers = T_QUAL_NONE;
}
} else if (CurTok.Tok == TOK_IDENT) { } else if (CurTok.Tok == TOK_IDENT) {
if (Mode == DM_NO_IDENT) { if (Mode == DM_NO_IDENT) {
Error ("Unexpected identifier in type name"); Error ("Unexpected identifier in type name");
@ -2119,28 +2160,11 @@ static void DirectDecl (DeclSpec* Spec, Declarator* D, declmode_t Mode)
while (CurTok.Tok == TOK_LBRACK || CurTok.Tok == TOK_LPAREN) { while (CurTok.Tok == TOK_LBRACK || CurTok.Tok == TOK_LPAREN) {
if (CurTok.Tok == TOK_LPAREN) { if (CurTok.Tok == TOK_LPAREN) {
/* Function declarator */ /* Function declarator */
FuncDesc* F; ParseFuncDecl (D, Mode, Qualifiers);
/* Parse the function declarator */
F = ParseFuncDecl ();
/* We cannot specify fastcall for variadic functions */
if ((F->Flags & FD_VARIADIC) && (Qualifiers & T_QUAL_FASTCALL)) {
Error ("Variadic functions cannot be __fastcall__");
Qualifiers &= ~T_QUAL_FASTCALL;
}
/* Add the function type. Be sure to bounds check the type buffer */
NeedTypeSpace (D, 1);
D->Type[D->Index].C = T_FUNC | Qualifiers;
D->Type[D->Index].A.F = F;
++D->Index;
/* Qualifiers now used */ /* Qualifiers now used */
Qualifiers = T_QUAL_NONE; Qualifiers = T_QUAL_NONE;
} else { } else {
/* Array declarator */ /* Array declarator */
long Size = UNSPECIFIED; long Size = UNSPECIFIED;
@ -2385,9 +2409,11 @@ int ParseDecl (DeclSpec* Spec, Declarator* D, declmode_t Mode)
} }
/* Try some smart error recovery */ /* Try some smart error recovery */
if (CurTok.Tok != TOK_LCURLY || !IsTypeFunc (D->Type)) { if (Mode == DM_NO_IDENT) {
return SimpleErrorSkip ();
} else if (CurTok.Tok != TOK_LCURLY || !IsTypeFunc (D->Type)) {
/* Skip to the end of the whole declaration if it is not part of a /* Skip to the end of the whole declaration if it is not part of a
** parameter list or a type cast. ** parameter list.
*/ */
return SmartErrorSkip (Mode == DM_IDENT_OR_EMPTY); return SmartErrorSkip (Mode == DM_IDENT_OR_EMPTY);
} }

1
test/err/bug2303.c Normal file
View File

@ -0,0 +1 @@
int f(a); /* Should be an error */

32
test/val/bug2302.c Normal file
View File

@ -0,0 +1,32 @@
/* Bug #2302 - Parameters of function types not parsed correctly */
#include <stdio.h>
typedef int A;
int zoo(A ()); /* OK: int zoo(int (*)()) */
int zoo(A (())); /* OK: int zoo(int ((*)())) aka. int zoo(int (*)()) */
int zoo(A (A)); /* OK: int zoo(int (*)(int)) */
int zoo(A ((A))); /* OK: int zoo(int ((*)(int))) aka. int zoo(int (*)(int)) */
int zoo(A A(A)); /* OK: int zoo(int (*A)(int)) */
int zoo(A (*)(A)); /* OK: int zoo(int (*)(int)) */
int zoo(A (*A)(A)); /* OK: int zoo(int (*A)(int)) */
int zoo(A ((*A))(A)); /* OK: int zoo(int (*A)(int)) */
int zoo(A ((((*((fp))))(A A)))) /* OK: int zoo(int (*fp)(int A)) */
{
return fp(42);
}
int bar(int a)
{
return a ^ 42;
}
int main(void)
{
int a = zoo((int (*)())bar);
if (a != 0)
{
printf("failed: a = %d\n", a);
}
return a;
}