/* * (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands. * See the copyright notice in the ACK home directory, in the file "Copyright". */ /* $Header$ */ /* PREPROCESSOR: SCANNER FOR THE ACTUAL PARAMETERS OF MACROS */ #include "nopp.h" #ifndef NOPP /* This file contains the function getactuals() which scans an actual parameter list and splits it up into a list of strings, each one representing an actual parameter. */ #include "lapbuf.h" /* UF */ #include "nparams.h" /* UF */ #include "input.h" #include "class.h" #include "idf.h" #include "macro.h" #include "interface.h" #include "file_info.h" #define EOS '\0' #define overflow() (fatal("actual parameter buffer overflow")) PRIVATE char apbuf[LAPBUF]; /* temporary storage for actual parameters */ PRIVATE char *actparams[NPARAMS]; /* pointers to the text of the actuals */ PRIVATE char *aptr; /* pointer to last inserted character in apbuf */ #define copy(ch) ((aptr < &apbuf[LAPBUF]) ? (*aptr++ = ch) : overflow()) PRIVATE int nr_of_params; /* number of actuals read until now */ PRIVATE char ** getactuals(idef) register struct idf *idef; { /* getactuals() collects the actual parameters and turns them into a list of strings, a pointer to which is returned. */ register acnt = idef->id_macro->mc_nps; nr_of_params = 0; actparams[0] = aptr = &apbuf[0]; copyact('(', ')', 0); /* read the actual parameters */ copy(EOS); /* mark the end of it all */ if (!nr_of_params++) { /* 0 or 1 parameter */ /* there could be a ( ) */ register char *p = actparams[0]; while ((class(*p) == STSKIP) || (*p == '\n')) { ++p; } if (!*p) { /* the case () : 0 parameters */ nr_of_params--; } } if (nr_of_params != acnt) { /* argument mismatch: too many or too few actual parameters. */ lexwarning("argument mismatch, %s", idef->id_text); while (nr_of_params < acnt) { /* too few paraeters: remaining actuals are "" */ actparams[nr_of_params] = ""; nr_of_params++; } } return actparams; } PRIVATE copyact(ch1, ch2, lvl) char ch1, ch2; int lvl; { /* copyact() is taken from Ceriel Jacobs' LLgen, with permission. Its task is to build a list of actuals parameters, which list is surrounded by '(' and ')' and in which the parameters are separated by ',' if there are more than 1. The balancing of '(',')' and '[',']' and '{','}' is taken care of by calling this function recursively. At each level lvl, copyact() reads the input, upto the corresponding closing bracket. Opening bracket is ch1, closing bracket is ch2. If lvl != 0, copy opening and closing parameters too. */ register int ch; /* Current char */ register int match; /* used to read strings */ if (lvl) { copy(ch1); } for (;;) { LoadChar(ch); if (ch == ch2) { if (lvl) { copy(ch); } return; } switch(ch) { #ifdef __MATCHING_PAR__ case ')': case '}': case ']': lexerror("unbalanced parenthesis"); break; #endif __MATCHING_PAR__ case '(': copyact('(', ')', lvl+1); break; #ifdef __MATCHING_PAR__ case '{': /* example: #define declare(v, t) t v declare(v, union{int i, j; float r;}); */ copyact('{', '}', lvl+1); break; case '[': copyact('[', ']', lvl+1); break; #endif __MATCHING_PAR__ case '\n': LineNumber++; while (LoadChar(ch), ch == '#') { /* This piece of code needs some explanation: consider the call of the macro defined as: #define sum(b,c) (b + c) in the following form: sum( #include my_phone_number ,2) in which case the include must be interpreted as such. */ domacro(); /* has read nl, vt or ff */ /* Loop, for another control line */ } PushBack(); copy('\n'); break; case '/': LoadChar(ch); if (ch == '*') { /* skip comment */ skipcomment(); continue; } PushBack(); copy('/'); break; case ',': if (!lvl) { /* next parameter encountered */ copy(EOS); if (++nr_of_params >= NPARAMS) { fatal("too many actual parameters"); } actparams[nr_of_params] = aptr; } else { copy(ch); } break; case '\'': case '"' : /* watch out for brackets in strings, they do not count ! */ match = ch; copy(ch); while (LoadChar(ch), ch != EOI) { if (ch == match) { break; } if (ch == '\\') { copy(ch); LoadChar(ch); } else if (ch == '\n') { lexerror("newline in string"); LineNumber++; copy(match); break; } copy(ch); } if (ch == match) { copy(ch); break; } /* Fall through */ case EOI : lexerror("unterminated macro call"); return; default: copy(ch); break; } } } #endif NOPP