mirror of
https://github.com/cc65/cc65.git
synced 2024-12-25 02:29:52 +00:00
Added C99 variadic macros
git-svn-id: svn://svn.cc65.org/cc65/trunk@3547 b7a2c559-68d2-44c3-8de9-860c34a00d81
This commit is contained in:
parent
a45649752f
commit
1aeeffdc52
@ -6,7 +6,7 @@
|
||||
/* */
|
||||
/* */
|
||||
/* */
|
||||
/* (C) 2000-2004 Ullrich von Bassewitz */
|
||||
/* (C) 2000-2005, Ullrich von Bassewitz */
|
||||
/* Römerstraße 52 */
|
||||
/* D-70794 Filderstadt */
|
||||
/* EMail: uz@cc65.org */
|
||||
@ -82,6 +82,7 @@ Macro* NewMacro (const char* Name)
|
||||
M->MaxArgs = 0;
|
||||
InitCollection (&M->FormalArgs);
|
||||
InitStrBuf (&M->Replacement);
|
||||
M->Variadic = 0;
|
||||
memcpy (M->Name, Name, Len+1);
|
||||
|
||||
/* Return the new macro */
|
||||
|
@ -6,7 +6,7 @@
|
||||
/* */
|
||||
/* */
|
||||
/* */
|
||||
/* (C) 2000-2004 Ullrich von Bassewitz */
|
||||
/* (C) 2000-2005, Ullrich von Bassewitz */
|
||||
/* Römerstraße 52 */
|
||||
/* D-70794 Filderstadt */
|
||||
/* EMail: uz@cc65.org */
|
||||
@ -60,6 +60,7 @@ struct Macro {
|
||||
unsigned MaxArgs; /* Size of formal argument list */
|
||||
Collection FormalArgs; /* Formal argument list (char*) */
|
||||
StrBuf Replacement; /* Replacement text */
|
||||
unsigned char Variadic; /* C99 variadic macro */
|
||||
char Name[1]; /* Name, dynamically allocated */
|
||||
};
|
||||
|
||||
|
@ -205,6 +205,15 @@ static StrBuf* ME_GetActual (MacroExp* E, unsigned Index)
|
||||
|
||||
|
||||
|
||||
static int ME_ArgIsVariadic (const MacroExp* E)
|
||||
/* Return true if the next actual argument we will add is a variadic one */
|
||||
{
|
||||
return (E->M->Variadic &&
|
||||
E->M->ArgCount == (int) CollCount (&E->ActualArgs) + 1);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/*****************************************************************************/
|
||||
/* Code */
|
||||
/*****************************************************************************/
|
||||
@ -362,21 +371,37 @@ static void ReadMacroArgs (MacroExp* E)
|
||||
unsigned Parens; /* Number of open parenthesis */
|
||||
StrBuf Arg = STATIC_STRBUF_INITIALIZER;
|
||||
|
||||
/* Read the actual macro arguments and store pointers to these arguments
|
||||
* into the array of actual arguments in the macro definition.
|
||||
*/
|
||||
/* Read the actual macro arguments */
|
||||
Parens = 0;
|
||||
while (1) {
|
||||
if (CurC == '(') {
|
||||
|
||||
/* Nested parenthesis */
|
||||
SB_AppendChar (&Arg, CurC);
|
||||
NextChar ();
|
||||
++Parens;
|
||||
} else if (IsQuote (CurC)) {
|
||||
CopyQuotedString (&Arg);
|
||||
} else if (CurC == ',' || CurC == ')') {
|
||||
if (Parens == 0) {
|
||||
|
||||
} else if (IsQuote (CurC)) {
|
||||
|
||||
/* Quoted string - just copy */
|
||||
CopyQuotedString (&Arg);
|
||||
|
||||
} else if (CurC == ',' || CurC == ')') {
|
||||
|
||||
if (Parens) {
|
||||
/* Comma or right paren inside nested parenthesis */
|
||||
if (CurC == ')') {
|
||||
--Parens;
|
||||
}
|
||||
SB_AppendChar (&Arg, CurC);
|
||||
NextChar ();
|
||||
} else if (CurC == ',' && ME_ArgIsVariadic (E)) {
|
||||
/* It's a comma, but we're inside a variadic macro argument, so
|
||||
* just copy it and proceed.
|
||||
*/
|
||||
SB_AppendChar (&Arg, CurC);
|
||||
NextChar ();
|
||||
} else {
|
||||
/* End of actual argument. Remove whitespace from the end. */
|
||||
while (IsSpace (SB_LookAtLast (&Arg))) {
|
||||
SB_Drop (&Arg, 1);
|
||||
@ -398,13 +423,6 @@ static void ReadMacroArgs (MacroExp* E)
|
||||
/* Start the next param */
|
||||
NextChar ();
|
||||
SB_Clear (&Arg);
|
||||
} else {
|
||||
/* Comma or right paren inside nested parenthesis */
|
||||
if (CurC == ')') {
|
||||
--Parens;
|
||||
}
|
||||
SB_AppendChar (&Arg, CurC);
|
||||
NextChar ();
|
||||
}
|
||||
} else if (IsSpace (CurC)) {
|
||||
/* Squeeze runs of blanks within an arg */
|
||||
@ -549,7 +567,6 @@ static void MacroArgSubst (MacroExp* E)
|
||||
NextChar ();
|
||||
SkipWhitespace ();
|
||||
if (!IsSym (Ident) || (ArgIdx = FindMacroArg (E->M, Ident)) < 0) {
|
||||
printf ("<%.*s>\n", SB_GetLen (Line), SB_GetConstBuf (Line));
|
||||
PPError ("`#' is not followed by a macro parameter");
|
||||
} else {
|
||||
/* Make a valid string from Replacement */
|
||||
@ -675,6 +692,7 @@ static void DefineMacro (void)
|
||||
ident Ident;
|
||||
Macro* M;
|
||||
Macro* Existing;
|
||||
int C89;
|
||||
|
||||
/* Read the macro name */
|
||||
SkipWhitespace ();
|
||||
@ -682,6 +700,9 @@ static void DefineMacro (void)
|
||||
return;
|
||||
}
|
||||
|
||||
/* Remember if we're in C89 mode */
|
||||
C89 = (IS_Get (&Standard) == STD_C89);
|
||||
|
||||
/* Get an existing macro definition with this name */
|
||||
Existing = FindMacro (Ident);
|
||||
|
||||
@ -703,12 +724,39 @@ static void DefineMacro (void)
|
||||
if (CurC == ')') {
|
||||
break;
|
||||
}
|
||||
|
||||
/* The next token must be either an identifier, or - if not in
|
||||
* C89 mode - the ellipsis.
|
||||
*/
|
||||
if (!C89 && CurC == '.') {
|
||||
/* Ellipsis */
|
||||
NextChar ();
|
||||
if (CurC != '.' || NextC != '.') {
|
||||
PPError ("`...' expected");
|
||||
ClearLine ();
|
||||
return;
|
||||
}
|
||||
NextChar ();
|
||||
NextChar ();
|
||||
strcpy (Ident, "__VA_ARGS__");
|
||||
M->Variadic = 1;
|
||||
} else {
|
||||
/* Must be macro argument name */
|
||||
if (MacName (Ident) == 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
/* __VA_ARGS__ is only allowed in C89 mode */
|
||||
if (!C89 && strcmp (Ident, "__VA_ARGS__") == 0) {
|
||||
PPWarning ("`__VA_ARGS__' can only appear in the expansion "
|
||||
"of a C99 variadic macro");
|
||||
}
|
||||
}
|
||||
|
||||
/* Add the macro argument */
|
||||
AddMacroArg (M, Ident);
|
||||
SkipWhitespace ();
|
||||
if (CurC != ',') {
|
||||
if (M->Variadic || CurC != ',') {
|
||||
break;
|
||||
}
|
||||
NextChar ();
|
||||
|
Loading…
Reference in New Issue
Block a user