/* $RCSfile: toke.c,v $$Revision: 4.1 $$Date: 92/08/07 18:28:39 $ * * Copyright (c) 1991, Larry Wall * * You may distribute under the terms of either the GNU General Public * License or the Artistic License, as specified in the README file. * * $Log: toke.c,v $ * Revision 4.1 92/08/07 18:28:39 lwall * * Revision 4.0.1.7 92/06/11 21:16:30 lwall * patch34: expect incorrectly set to indicate start of program or block * * Revision 4.0.1.6 92/06/08 16:03:49 lwall * patch20: an EXPR may now start with a bareword * patch20: print $fh EXPR can now expect term rather than operator in EXPR * patch20: added ... as variant on .. * patch20: new warning on spurious backslash * patch20: new warning on missing $ for foreach variable * patch20: "foo"x1024 now legal without space after x * patch20: new warning on print accidentally used as function * patch20: tr/stuff// wasn't working right * patch20: 2. now eats the dot * patch20: <@ARGV> now notices @ARGV * patch20: tr/// now lets you say \- * * Revision 4.0.1.5 91/11/11 16:45:51 lwall * patch19: default arg for shift was wrong after first subroutine definition * * Revision 4.0.1.4 91/11/05 19:02:48 lwall * patch11: \x and \c were subject to double interpretation in regexps * patch11: prepared for ctype implementations that don't define isascii() * patch11: nested list operators could miscount parens * patch11: once-thru blocks didn't display right in the debugger * patch11: sort eval "whatever" didn't work * patch11: underscore is now allowed within literal octal and hex numbers * * Revision 4.0.1.3 91/06/10 01:32:26 lwall * patch10: m'$foo' now treats string as single quoted * patch10: certain pattern optimizations were botched * * Revision 4.0.1.2 91/06/07 12:05:56 lwall * patch4: new copyright notice * patch4: debugger lost track of lines in eval * patch4: //o and s///o now optimize themselves fully at runtime * patch4: added global modifier for pattern matches * * Revision 4.0.1.1 91/04/12 09:18:18 lwall * patch1: perl -de "print" wouldn't stop at the first statement * * Revision 4.0 91/03/20 01:42:14 lwall * 4.0 baseline. * */ #include "EXTERN.h" #include "perl.h" #include "perly.h" static void set_csh(); /* The following are arranged oddly so that the guard on the switch statement * can get by with a single comparison (if the compiler is smart enough). */ #define LEX_NORMAL 8 #define LEX_INTERPNORMAL 7 #define LEX_INTERPCASEMOD 6 #define LEX_INTERPSTART 5 #define LEX_INTERPEND 4 #define LEX_INTERPENDMAYBE 3 #define LEX_INTERPCONCAT 2 #define LEX_INTERPCONST 1 #define LEX_KNOWNEXT 0 static U32 lex_state = LEX_NORMAL; /* next token is determined */ static U32 lex_defer; /* state after determined token */ static expectation lex_expect; /* expect after determined token */ static I32 lex_brackets; /* bracket count */ static I32 lex_fakebrack; /* outer bracket is mere delimiter */ static I32 lex_casemods; /* casemod count */ static I32 lex_dojoin; /* doing an array interpolation */ static I32 lex_starts; /* how many interps done on level */ static SV * lex_stuff; /* runtime pattern from m// or s/// */ static SV * lex_repl; /* runtime replacement from s/// */ static OP * lex_op; /* extra info to pass back on op */ static I32 lex_inpat; /* in pattern $) and $| are special */ static I32 lex_inwhat; /* what kind of quoting are we in */ static char * lex_brackstack; /* what kind of brackets to pop */ /* What we know when we're in LEX_KNOWNEXT state. */ static YYSTYPE nextval[5]; /* value of next token, if any */ static I32 nexttype[5]; /* type of next token */ static I32 nexttoke = 0; #ifdef I_FCNTL #include #endif #ifdef I_SYS_FILE #include #endif #ifdef ff_next #undef ff_next #endif #include "keywords.h" void checkcomma(); #ifdef CLINE #undef CLINE #endif #define CLINE (copline = (curcop->cop_line < copline ? curcop->cop_line : copline)) #ifdef atarist #define PERL_META(c) ((c) | 128) #else #define META(c) ((c) | 128) #endif #define TOKEN(retval) return (bufptr = s,(int)retval) #define OPERATOR(retval) return (expect = XTERM,bufptr = s,(int)retval) #define PREBLOCK(retval) return (expect = XBLOCK,bufptr = s,(int)retval) #define PREREF(retval) return (expect = XREF,bufptr = s,(int)retval) #define TERM(retval) return (CLINE, expect = XOPERATOR,bufptr = s,(int)retval) #define LOOPX(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)LOOPEX) #define FTST(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)UNIOP) #define FUN0(f) return(yylval.ival = f,expect = XOPERATOR,bufptr = s,(int)FUNC0) #define FUN1(f) return(yylval.ival = f,expect = XOPERATOR,bufptr = s,(int)FUNC1) #define BOop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)BITOROP) #define BAop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)BITANDOP) #define SHop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)SHIFTOP) #define PWop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)POWOP) #define PMop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)MATCHOP) #define Aop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)ADDOP) #define Mop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)MULOP) #define Eop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)EQOP) #define Rop(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)RELOP) /* This bit of chicanery makes a unary function followed by * a parenthesis into a function with one argument, highest precedence. */ #define UNI(f) return(yylval.ival = f, \ expect = XTERM, \ bufptr = s, \ last_uni = oldbufptr, \ (*s == '(' || (s = skipspace(s), *s == '(') ? (int)FUNC1 : (int)UNIOP) ) #define UNIBRACK(f) return(yylval.ival = f, \ bufptr = s, \ last_uni = oldbufptr, \ (*s == '(' || (s = skipspace(s), *s == '(') ? (int)FUNC1 : (int)UNIOP) ) /* This does similarly for list operators */ #define LOP(f) return(yylval.ival = f, \ CLINE, \ expect = XREF, \ bufptr = s, \ last_lop = oldbufptr, \ (*s == '(' || (s = skipspace(s), *s == '(') ? (int)FUNC : (int)LSTOP) ) /* grandfather return to old style */ #define OLDLOP(f) return(yylval.ival=f,expect = XTERM,bufptr = s,(int)LSTOP) void no_op(what) char *what; { warn("%s found where operator expected", what); if (bufptr == SvPVX(linestr)) warn("\t(Missing semicolon on previous line?)\n", what); } void lex_start() { ENTER; SAVEINT(lex_dojoin); SAVEINT(lex_brackets); SAVEINT(lex_fakebrack); SAVEINT(lex_casemods); SAVEINT(lex_starts); SAVEINT(lex_state); SAVEINT(lex_inpat); SAVEINT(lex_inwhat); SAVEINT(curcop->cop_line); SAVESPTR(bufptr); SAVESPTR(oldbufptr); SAVESPTR(oldoldbufptr); SAVESPTR(linestr); SAVESPTR(lex_brackstack); lex_state = LEX_NORMAL; lex_defer = 0; lex_expect = XBLOCK; lex_brackets = 0; lex_fakebrack = 0; if (lex_brackstack) SAVESPTR(lex_brackstack); lex_brackstack = malloc(120); lex_casemods = 0; lex_dojoin = 0; lex_starts = 0; if (lex_stuff) sv_free(lex_stuff); lex_stuff = Nullsv; if (lex_repl) sv_free(lex_repl); lex_repl = Nullsv; lex_inpat = 0; lex_inwhat = 0; oldoldbufptr = oldbufptr = bufptr = SvPVX(linestr); bufend = bufptr + SvCUR(linestr); rs = "\n"; rslen = 1; rschar = '\n'; rspara = 0; } void lex_end() { free(lex_brackstack); lex_brackstack = 0; LEAVE; } static void incline(s) char *s; { char *t; char *n; char ch; int sawline = 0; curcop->cop_line++; if (*s++ != '#') return; while (*s == ' ' || *s == '\t') s++; if (strnEQ(s, "line ", 5)) { s += 5; sawline = 1; } if (!isDIGIT(*s)) return; n = s; while (isDIGIT(*s)) s++; while (*s == ' ' || *s == '\t') s++; if (*s == '"' && (t = strchr(s+1, '"'))) s++; else { if (!sawline) return; /* false alarm */ for (t = s; !isSPACE(*t); t++) ; } ch = *t; *t = '\0'; if (t - s > 0) curcop->cop_filegv = gv_fetchfile(s); else curcop->cop_filegv = gv_fetchfile(origfilename); *t = ch; curcop->cop_line = atoi(n)-1; } char * skipspace(s) register char *s; { if (in_format && lex_brackets <= 1) { while (s < bufend && (*s == ' ' || *s == '\t')) s++; return s; } for (;;) { while (s < bufend && isSPACE(*s)) s++; if (s < bufend && *s == '#') { while (s < bufend && *s != '\n') s++; if (s < bufend) s++; } if (s < bufend || !rsfp) return s; if ((s = sv_gets(linestr, rsfp, 0)) == Nullch) { sv_setpv(linestr,""); bufend = oldoldbufptr = oldbufptr = bufptr = s = SvPVX(linestr); return s; } oldoldbufptr = oldbufptr = bufptr = s; bufend = bufptr + SvCUR(linestr); incline(s); } } void check_uni() { char *s; char ch; if (oldoldbufptr != last_uni) return; while (isSPACE(*last_uni)) last_uni++; for (s = last_uni; isALNUM(*s) || *s == '-'; s++) ; ch = *s; *s = '\0'; warn("Warning: Use of \"%s\" without parens is ambiguous", last_uni); *s = ch; } #ifdef CRIPPLED_CC #undef UNI #undef LOP #define UNI(f) return uni(f,s) #define LOP(f) return lop(f,s) int uni(f,s) I32 f; char *s; { yylval.ival = f; expect = XTERM; bufptr = s; last_uni = oldbufptr; if (*s == '(') return FUNC1; s = skipspace(s); if (*s == '(') return FUNC1; else return UNIOP; } I32 lop(f,s) I32 f; char *s; { yylval.ival = f; CLINE; expect = XREF; bufptr = s; last_uni = oldbufptr; if (*s == '(') return FUNC; s = skipspace(s); if (*s == '(') return FUNC; else return LSTOP; } #endif /* CRIPPLED_CC */ void force_next(type) I32 type; { nexttype[nexttoke] = type; nexttoke++; if (lex_state != LEX_KNOWNEXT) { lex_defer = lex_state; lex_expect = expect; lex_state = LEX_KNOWNEXT; } } char * force_word(start,token,check_keyword,allow_tick) register char *start; int token; int check_keyword; int allow_tick; { register char *s; STRLEN len; start = skipspace(start); s = start; if (isIDFIRST(*s) || (allow_tick && (*s == '\'' || *s == ':'))) { s = scan_word(s, tokenbuf, allow_tick, &len); if (check_keyword && keyword(tokenbuf, len)) return start; if (token == METHOD) { s = skipspace(s); if (*s == '(') expect = XTERM; else { expect = XOPERATOR; force_next(')'); force_next('('); } } nextval[nexttoke].opval = (OP*)newSVOP(OP_CONST,0, newSVpv(tokenbuf,0)); force_next(token); } return s; } void force_ident(s) register char *s; { if (s && *s) { nextval[nexttoke].opval = (OP*)newSVOP(OP_CONST, 0, newSVpv(s,0)); force_next(WORD); } } SV * q(sv) SV *sv; { register char *s; register char *send; register char *d; register char delim; STRLEN len; if (!SvLEN(sv)) return sv; s = SvPV(sv, len); send = s + len; while (s < send && *s != '\\') s++; if (s == send) return sv; d = s; delim = SvIVX(sv); while (s < send) { if (*s == '\\') { if (s + 1 < send && (s[1] == '\\' || s[1] == delim)) s++; /* all that, just for this */ } *d++ = *s++; } *d = '\0'; SvCUR_set(sv, d - SvPVX(sv)); return sv; } I32 sublex_start() { register I32 op_type = yylval.ival; SV *sv; STRLEN len; if (op_type == OP_NULL) { yylval.opval = lex_op; lex_op = Nullop; return THING; } if (op_type == OP_CONST || op_type == OP_READLINE) { yylval.opval = (OP*)newSVOP(op_type, 0, q(lex_stuff)); lex_stuff = Nullsv; return THING; } push_scope(); SAVEINT(lex_dojoin); SAVEINT(lex_brackets); SAVEINT(lex_fakebrack); SAVEINT(lex_casemods); SAVEINT(lex_starts); SAVEINT(lex_state); SAVEINT(lex_inpat); SAVEINT(lex_inwhat); SAVEINT(curcop->cop_line); SAVESPTR(bufptr); SAVESPTR(oldbufptr); SAVESPTR(oldoldbufptr); SAVESPTR(linestr); SAVESPTR(lex_brackstack); linestr = lex_stuff; lex_stuff = Nullsv; bufend = bufptr = oldbufptr = oldoldbufptr = SvPVX(linestr); bufend += SvCUR(linestr); lex_dojoin = FALSE; lex_brackets = 0; lex_fakebrack = 0; lex_brackstack = malloc(120); lex_casemods = 0; lex_starts = 0; lex_state = LEX_INTERPCONCAT; curcop->cop_line = multi_start; lex_inwhat = op_type; if (op_type == OP_MATCH || op_type == OP_SUBST) lex_inpat = op_type; else lex_inpat = 0; expect = XTERM; force_next('('); if (lex_op) { yylval.opval = lex_op; lex_op = Nullop; return PMFUNC; } else return FUNC; } I32 sublex_done() { if (!lex_starts++) { expect = XOPERATOR; yylval.opval = (OP*)newSVOP(OP_CONST, 0, newSVpv("",0)); return THING; } if (lex_casemods) { /* oops, we've got some unbalanced parens */ lex_state = LEX_INTERPCASEMOD; return yylex(); } sv_free(linestr); /* Is there a right-hand side to take care of? */ if (lex_repl && (lex_inwhat == OP_SUBST || lex_inwhat == OP_TRANS)) { linestr = lex_repl; lex_inpat = 0; bufend = bufptr = oldbufptr = oldoldbufptr = SvPVX(linestr); bufend += SvCUR(linestr); lex_dojoin = FALSE; lex_brackets = 0; lex_fakebrack = 0; lex_casemods = 0; lex_starts = 0; if (SvCOMPILED(lex_repl)) { lex_state = LEX_INTERPNORMAL; lex_starts++; } else lex_state = LEX_INTERPCONCAT; lex_repl = Nullsv; return ','; } else { if (lex_brackstack) free(lex_brackstack); lex_brackstack = 0; pop_scope(); bufend = SvPVX(linestr); bufend += SvCUR(linestr); expect = XOPERATOR; return ')'; } } char * scan_const(start) char *start; { register char *send = bufend; SV *sv = NEWSV(93, send - start); register char *s = start; register char *d = SvPVX(sv); char delim = SvIVX(linestr); bool dorange = FALSE; I32 len; char *leave = lex_inpat ? "\\.^$@dDwWsSbB+*?|()-nrtfeaxc0123456789[{]}" : (lex_inwhat & OP_TRANS) ? "" : ""; while (s < send || dorange) { if (lex_inwhat == OP_TRANS) { if (dorange) { I32 i; I32 max; i = d - SvPVX(sv); SvGROW(sv, SvLEN(sv) + 256); d = SvPVX(sv) + i; d -= 2; max = d[1] & 0377; for (i = (*d & 0377); i <= max; i++) *d++ = i; dorange = FALSE; continue; } else if (*s == '-' && s+1 < send && s != start) { dorange = TRUE; s++; } } else if (*s == '@') break; else if (*s == '$') { if (!lex_inpat) /* not a regexp, so $ must be var */ break; if (s + 1 < send && s[1] != ')' && s[1] != '|') break; /* in regexp, $ might be tail anchor */ } if (*s == '\\' && s+1 < send) { s++; if (*s == delim) { *d++ = *s++; continue; } if (*s && strchr(leave, *s)) { *d++ = '\\'; *d++ = *s++; continue; } if (lex_inwhat == OP_SUBST && !lex_inpat && isDIGIT(*s) && !isDIGIT(s[1])) { *--s = '$'; break; } if (lex_inwhat != OP_TRANS && *s && strchr("lLuUE", *s)) { --s; break; } switch (*s) { case '-': if (lex_inwhat == OP_TRANS) { *d++ = *s++; continue; } /* FALL THROUGH */ default: *d++ = *s++; continue; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': *d++ = scan_oct(s, 3, &len); s += len; continue; case 'x': *d++ = scan_hex(++s, 2, &len); s += len; continue; case 'c': s++; *d = *s++; if (isLOWER(*d)) *d = toupper(*d); *d++ ^= 64; continue; case 'b': *d++ = '\b'; break; case 'n': *d++ = '\n'; break; case 'r': *d++ = '\r'; break; case 'f': *d++ = '\f'; break; case 't': *d++ = '\t'; break; case 'e': *d++ = '\033'; break; case 'a': *d++ = '\007'; break; } s++; continue; } *d++ = *s++; } *d = '\0'; SvCUR_set(sv, d - SvPVX(sv)); SvPOK_on(sv); if (SvCUR(sv) + 5 < SvLEN(sv)) { SvLEN_set(sv, SvCUR(sv) + 1); Renew(SvPVX(sv), SvLEN(sv), char); } if (s > bufptr) yylval.opval = (OP*)newSVOP(OP_CONST, 0, sv); else sv_free(sv); return s; } /* This is the one truly awful dwimmer necessary to conflate C and sed. */ int intuit_more(s) register char *s; { if (lex_brackets) return TRUE; if (*s == '-' && s[1] == '>' && (s[2] == '[' || s[2] == '{')) return TRUE; if (*s != '{' && *s != '[') return FALSE; if (!lex_inpat) return TRUE; /* In a pattern, so maybe we have {n,m}. */ if (*s == '{') { s++; if (!isDIGIT(*s)) return TRUE; while (isDIGIT(*s)) s++; if (*s == ',') s++; while (isDIGIT(*s)) s++; if (*s == '}') return FALSE; return TRUE; } /* On the other hand, maybe we have a character class */ s++; if (*s == ']' || *s == '^') return FALSE; else { int weight = 2; /* let's weigh the evidence */ char seen[256]; unsigned char un_char = 0, last_un_char; char *send = strchr(s,']'); char tmpbuf[512]; if (!send) /* has to be an expression */ return TRUE; Zero(seen,256,char); if (*s == '$') weight -= 3; else if (isDIGIT(*s)) { if (s[1] != ']') { if (isDIGIT(s[1]) && s[2] == ']') weight -= 10; } else weight -= 100; } for (; s < send; s++) { last_un_char = un_char; un_char = (unsigned char)*s; switch (*s) { case '@': case '&': case '$': weight -= seen[un_char] * 10; if (isALNUM(s[1])) { scan_ident(s,send,tmpbuf,FALSE); if (strlen(tmpbuf) > 1 && gv_fetchpv(tmpbuf,FALSE)) weight -= 100; else weight -= 10; } else if (*s == '$' && s[1] && strchr("[#!%*<>()-=",s[1])) { if (/*{*/ strchr("])} =",s[2])) weight -= 10; else weight -= 1; } break; case '\\': un_char = 254; if (s[1]) { if (strchr("wds]",s[1])) weight += 100; else if (seen['\''] || seen['"']) weight += 1; else if (strchr("rnftbxcav",s[1])) weight += 40; else if (isDIGIT(s[1])) { weight += 40; while (s[1] && isDIGIT(s[1])) s++; } } else weight += 100; break; case '-': if (s[1] == '\\') weight += 50; if (strchr("aA01! ",last_un_char)) weight += 30; if (strchr("zZ79~",s[1])) weight += 30; break; default: if (!isALNUM(last_un_char) && !strchr("$@&",last_un_char) && isALPHA(*s) && s[1] && isALPHA(s[1])) { char *d = tmpbuf; while (isALPHA(*s)) *d++ = *s++; *d = '\0'; if (keyword(tmpbuf, d - tmpbuf)) weight -= 150; } if (un_char == last_un_char + 1) weight += 5; weight -= seen[un_char]; break; } seen[un_char]++; } if (weight >= 0) /* probably a character class */ return FALSE; } return TRUE; } static char* exp_name[] = { "OPERATOR", "TERM", "BLOCK", "REF" }; extern int yychar; /* last token */ int yylex() { register char *s; register char *d; register I32 tmp; STRLEN len; switch (lex_state) { #ifdef COMMENTARY case LEX_NORMAL: /* Some compilers will produce faster */ case LEX_INTERPNORMAL: /* code if we comment these out. */ break; #endif case LEX_KNOWNEXT: nexttoke--; yylval = nextval[nexttoke]; if (!nexttoke) { lex_state = lex_defer; expect = lex_expect; } return(nexttype[nexttoke]); case LEX_INTERPCASEMOD: #ifdef DEBUGGING if (bufptr != bufend && *bufptr != '\\') croak("panic: INTERPCASEMOD"); #endif if (bufptr == bufend || bufptr[1] == 'E') { if (lex_casemods <= 1) { if (bufptr != bufend) bufptr += 2; lex_state = LEX_INTERPSTART; } if (lex_casemods) { --lex_casemods; return ')'; } return yylex(); } else if (lex_casemods) { --lex_casemods; return ')'; } else { s = bufptr + 1; if (strnEQ(s, "L\\u", 3) || strnEQ(s, "U\\l", 3)) tmp = *s, *s = s[2], s[2] = tmp; /* misordered... */ ++lex_casemods; lex_state = LEX_INTERPCONCAT; nextval[nexttoke].ival = 0; force_next('('); if (*s == 'l') nextval[nexttoke].ival = OP_LCFIRST; else if (*s == 'u') nextval[nexttoke].ival = OP_UCFIRST; else if (*s == 'L') nextval[nexttoke].ival = OP_LC; else if (*s == 'U') nextval[nexttoke].ival = OP_UC; else croak("panic: yylex"); bufptr = s + 1; force_next(FUNC); if (lex_starts) { s = bufptr; lex_starts = 0; Aop(OP_CONCAT); } else return yylex(); } case LEX_INTERPSTART: if (bufptr == bufend) return sublex_done(); expect = XTERM; lex_dojoin = (*bufptr == '@'); lex_state = LEX_INTERPNORMAL; if (lex_dojoin) { nextval[nexttoke].ival = 0; force_next(','); force_ident("\""); nextval[nexttoke].ival = 0; force_next('$'); nextval[nexttoke].ival = 0; force_next('('); nextval[nexttoke].ival = OP_JOIN; /* emulate join($", ...) */ force_next(FUNC); } if (lex_starts++) { s = bufptr; Aop(OP_CONCAT); } else return yylex(); break; case LEX_INTERPENDMAYBE: if (intuit_more(bufptr)) { lex_state = LEX_INTERPNORMAL; /* false alarm, more expr */ break; } /* FALL THROUGH */ case LEX_INTERPEND: if (lex_dojoin) { lex_dojoin = FALSE; lex_state = LEX_INTERPCONCAT; return ')'; } /* FALLTHROUGH */ case LEX_INTERPCONCAT: #ifdef DEBUGGING if (lex_brackets) croak("panic: INTERPCONCAT"); #endif if (bufptr == bufend) return sublex_done(); if (SvIVX(linestr) == '\'') { SV *sv = newSVsv(linestr); if (!lex_inpat) sv = q(sv); yylval.opval = (OP*)newSVOP(OP_CONST, 0, sv); s = bufend; } else { s = scan_const(bufptr); if (*s == '\\') lex_state = LEX_INTERPCASEMOD; else lex_state = LEX_INTERPSTART; } if (s != bufptr) { nextval[nexttoke] = yylval; expect = XTERM; force_next(THING); if (lex_starts++) Aop(OP_CONCAT); else { bufptr = s; return yylex(); } } return yylex(); } s = bufptr; oldoldbufptr = oldbufptr; oldbufptr = s; DEBUG_p( { fprintf(stderr,"### Tokener expecting %s at %s\n", exp_name[expect], s); } ) retry: #ifdef BADSWITCH if (*s & 128) { if ((*s & 127) == '}') { *s++ = '}'; TOKEN('}'); } else warn("Unrecognized character \\%03o ignored", *s++ & 255); goto retry; } #endif switch (*s) { default: if ((*s & 127) == '}') { *s++ = '}'; TOKEN('}'); } else warn("Unrecognized character \\%03o ignored", *s++ & 255); goto retry; case 4: case 26: goto fake_eof; /* emulate EOF on ^D or ^Z */ case 0: if (!rsfp) { if (lex_brackets) yyerror("Missing right bracket"); TOKEN(0); } if (s++ < bufend) goto retry; /* ignore stray nulls */ last_uni = 0; last_lop = 0; if (!preambled) { preambled = TRUE; sv_setpv(linestr,""); if (perldb) { char *pdb = getenv("PERLDB"); sv_catpv(linestr,"{"); sv_catpv(linestr, pdb ? pdb : "require 'perldb.pl'"); sv_catpv(linestr, "}"); } if (minus_n || minus_p) { sv_catpv(linestr, "LINE: while (<>) {"); if (minus_l) sv_catpv(linestr,"chop;"); if (minus_a) sv_catpv(linestr,"@F=split(' ');"); } oldoldbufptr = oldbufptr = s = SvPVX(linestr); bufend = SvPVX(linestr) + SvCUR(linestr); goto retry; } #ifdef CRYPTSCRIPT cryptswitch(); #endif /* CRYPTSCRIPT */ do { if ((s = sv_gets(linestr, rsfp, 0)) == Nullch) { fake_eof: if (rsfp) { if (preprocess) (void)my_pclose(rsfp); else if ((FILE*)rsfp == stdin) clearerr(stdin); else (void)fclose(rsfp); rsfp = Nullfp; } if (minus_n || minus_p) { sv_setpv(linestr,minus_p ? ";}continue{print" : ""); sv_catpv(linestr,";}"); oldoldbufptr = oldbufptr = s = SvPVX(linestr); bufend = SvPVX(linestr) + SvCUR(linestr); minus_n = minus_p = 0; goto retry; } oldoldbufptr = oldbufptr = s = SvPVX(linestr); sv_setpv(linestr,""); TOKEN(';'); /* not infinite loop because rsfp is NULL now */ } if (doextract && *s == '#') doextract = FALSE; incline(s); } while (doextract); oldoldbufptr = oldbufptr = bufptr = s; if (perldb) { SV *sv = NEWSV(85,0); sv_upgrade(sv, SVt_PVMG); sv_setsv(sv,linestr); av_store(GvAV(curcop->cop_filegv),(I32)curcop->cop_line,sv); } bufend = SvPVX(linestr) + SvCUR(linestr); if (curcop->cop_line == 1) { while (s < bufend && isSPACE(*s)) s++; if (*s == ':') /* for csh's that have to exec sh scripts */ s++; if (*s == '#' && s[1] == '!') { if (!in_eval && !instr(s,"perl") && instr(origargv[0],"perl")) { char **newargv; char *cmd; s += 2; if (*s == ' ') s++; cmd = s; while (s < bufend && !isSPACE(*s)) s++; *s++ = '\0'; while (s < bufend && isSPACE(*s)) s++; if (s < bufend) { Newz(899,newargv,origargc+3,char*); newargv[1] = s; while (s < bufend && !isSPACE(*s)) s++; *s = '\0'; Copy(origargv+1, newargv+2, origargc+1, char*); } else newargv = origargv; newargv[0] = cmd; execv(cmd,newargv); croak("Can't exec %s", cmd); } if (d = instr(s, "perl -")) { d += 6; /*SUPPRESS 530*/ while (d = moreswitches(d)) ; } } } if (in_format && lex_brackets <= 1) { s = scan_formline(s); if (!in_format) goto rightbracket; OPERATOR(';'); } goto retry; case ' ': case '\t': case '\f': case '\r': case 013: s++; goto retry; case '#': case '\n': if (lex_state != LEX_NORMAL || (in_eval && !rsfp)) { d = bufend; while (s < d && *s != '\n') s++; if (s < d) s++; incline(s); if (in_format && lex_brackets <= 1) { s = scan_formline(s); if (!in_format) goto rightbracket; OPERATOR(';'); } } else { *s = '\0'; bufend = s; } goto retry; case '-': if (s[1] && isALPHA(s[1]) && !isALNUM(s[2])) { s++; last_uni = oldbufptr; switch (*s++) { case 'r': FTST(OP_FTEREAD); case 'w': FTST(OP_FTEWRITE); case 'x': FTST(OP_FTEEXEC); case 'o': FTST(OP_FTEOWNED); case 'R': FTST(OP_FTRREAD); case 'W': FTST(OP_FTRWRITE); case 'X': FTST(OP_FTREXEC); case 'O': FTST(OP_FTROWNED); case 'e': FTST(OP_FTIS); case 'z': FTST(OP_FTZERO); case 's': FTST(OP_FTSIZE); case 'f': FTST(OP_FTFILE); case 'd': FTST(OP_FTDIR); case 'l': FTST(OP_FTLINK); case 'p': FTST(OP_FTPIPE); case 'S': FTST(OP_FTSOCK); case 'u': FTST(OP_FTSUID); case 'g': FTST(OP_FTSGID); case 'k': FTST(OP_FTSVTX); case 'b': FTST(OP_FTBLK); case 'c': FTST(OP_FTCHR); case 't': FTST(OP_FTTTY); case 'T': FTST(OP_FTTEXT); case 'B': FTST(OP_FTBINARY); case 'M': gv_fetchpv("\024",TRUE); FTST(OP_FTMTIME); case 'A': gv_fetchpv("\024",TRUE); FTST(OP_FTATIME); case 'C': gv_fetchpv("\024",TRUE); FTST(OP_FTCTIME); default: s -= 2; break; } } tmp = *s++; if (*s == tmp) { s++; if (expect == XOPERATOR) TERM(POSTDEC); else OPERATOR(PREDEC); } else if (*s == '>') { s++; s = skipspace(s); if (isIDFIRST(*s)) { s = force_word(s,METHOD,TRUE,FALSE); TOKEN(ARROW); } else PREBLOCK(ARROW); } if (expect == XOPERATOR) Aop(OP_SUBTRACT); else { if (isSPACE(*s) || !isSPACE(*bufptr)) check_uni(); OPERATOR('-'); /* unary minus */ } case '+': tmp = *s++; if (*s == tmp) { s++; if (expect == XOPERATOR) TERM(POSTINC); else OPERATOR(PREINC); } if (expect == XOPERATOR) Aop(OP_ADD); else { if (isSPACE(*s) || !isSPACE(*bufptr)) check_uni(); OPERATOR('+'); } case '*': if (expect != XOPERATOR) { s = scan_ident(s, bufend, tokenbuf, TRUE); expect = XOPERATOR; force_ident(tokenbuf); TERM('*'); } s++; if (*s == '*') { s++; PWop(OP_POW); } Mop(OP_MULTIPLY); case '%': if (expect != XOPERATOR) { s = scan_ident(s, bufend, tokenbuf + 1, TRUE); if (tokenbuf[1]) { expect = XOPERATOR; tokenbuf[0] = '%'; if (in_my) { if (strchr(tokenbuf,':')) croak("\"my\" variable %s can't be in a package",tokenbuf); nextval[nexttoke].opval = newOP(OP_PADANY, 0); nextval[nexttoke].opval->op_targ = pad_allocmy(tokenbuf); force_next(PRIVATEREF); TERM('%'); } if (!strchr(tokenbuf,':')) { if (tmp = pad_findmy(tokenbuf)) { nextval[nexttoke].opval = newOP(OP_PADANY, 0); nextval[nexttoke].opval->op_targ = tmp; force_next(PRIVATEREF); TERM('%'); } } force_ident(tokenbuf + 1); } else PREREF('%'); TERM('%'); } ++s; Mop(OP_MODULO); case '^': s++; BOop(OP_XOR); case '[': lex_brackets++; /* FALL THROUGH */ case '~': case ',': case '(': case ':': tmp = *s++; OPERATOR(tmp); case ';': if (curcop->cop_line < copline) copline = curcop->cop_line; tmp = *s++; OPERATOR(tmp); case ')': tmp = *s++; TERM(tmp); case ']': s++; if (lex_brackets <= 0) yyerror("Unmatched right bracket"); else --lex_brackets; if (lex_state == LEX_INTERPNORMAL) { if (lex_brackets == 0) { if (*s != '-' || s[1] != '>') lex_state = LEX_INTERPEND; } } TOKEN(']'); case '{': leftbracket: if (in_format == 2) in_format = 0; s++; if (lex_brackets > 100) realloc(lex_brackstack, lex_brackets + 1); if (oldoldbufptr == last_lop) lex_brackstack[lex_brackets++] = XTERM; else lex_brackstack[lex_brackets++] = XOPERATOR; if (expect == XTERM) OPERATOR(HASHBRACK); else if (expect == XREF) { char *t; s = skipspace(s); if (*s == '}') OPERATOR(HASHBRACK); for (t = s; t < bufend && (isSPACE(*t) || isALPHA(*t) || *t == '"' || *t == '\''); t++) ; if (*t == ',' || (*t == '=' && t[1] == '>')) OPERATOR(HASHBRACK); expect = XTERM; } else { lex_brackstack[lex_brackets-1] = XBLOCK; expect = XBLOCK; } yylval.ival = curcop->cop_line; if (isSPACE(*s) || *s == '#') copline = NOLINE; /* invalidate current command line number */ TOKEN('{'); case '}': rightbracket: s++; if (lex_brackets <= 0) yyerror("Unmatched right bracket"); else expect = (expectation)lex_brackstack[--lex_brackets]; if (lex_state == LEX_INTERPNORMAL) { if (lex_brackets == 0) { if (lex_fakebrack) { lex_state = LEX_INTERPEND; bufptr = s; return yylex(); /* ignore fake brackets */ } if (*s != '-' || s[1] != '>') lex_state = LEX_INTERPEND; } } force_next('}'); TOKEN(';'); case '&': s++; tmp = *s++; if (tmp == '&') OPERATOR(ANDAND); s--; if (expect == XOPERATOR) { if (dowarn && isALPHA(*s) && bufptr == SvPVX(linestr)) { curcop->cop_line--; warn(warn_nosemi); curcop->cop_line++; } BAop(OP_BIT_AND); } s = scan_ident(s-1, bufend, tokenbuf, TRUE); if (*tokenbuf) { expect = XOPERATOR; force_ident(tokenbuf); } else PREREF('&'); TERM('&'); case '|': s++; tmp = *s++; if (tmp == '|') OPERATOR(OROR); s--; BOop(OP_BIT_OR); case '=': s++; tmp = *s++; if (tmp == '=') Eop(OP_EQ); if (tmp == '>') OPERATOR(','); if (tmp == '~') PMop(OP_MATCH); if (dowarn && tmp && isSPACE(*s) && strchr("+-*/%.^&|<",tmp)) warn("Reversed %c= operator",tmp); s--; if (in_format == 2 && (tmp == '\n' || s[1] == '\n')) { in_format = 1; s--; expect = XBLOCK; goto leftbracket; } OPERATOR('='); case '!': s++; tmp = *s++; if (tmp == '=') Eop(OP_NE); if (tmp == '~') PMop(OP_NOT); s--; OPERATOR('!'); case '<': if (expect != XOPERATOR) { if (s[1] != '<' && !strchr(s,'>')) check_uni(); if (s[1] == '<') s = scan_heredoc(s); else s = scan_inputsymbol(s); TERM(sublex_start()); } s++; tmp = *s++; if (tmp == '<') SHop(OP_LEFT_SHIFT); if (tmp == '=') { tmp = *s++; if (tmp == '>') Eop(OP_NCMP); s--; Rop(OP_LE); } s--; Rop(OP_LT); case '>': s++; tmp = *s++; if (tmp == '>') SHop(OP_RIGHT_SHIFT); if (tmp == '=') Rop(OP_GE); s--; Rop(OP_GT); case '$': if (expect == XOPERATOR) { if (in_format) OPERATOR(','); /* grandfather non-comma-format format */ else no_op("Scalar"); } if (s[1] == '#' && (isALPHA(s[2]) || s[2] == '_' || s[2] == '{')) { s = scan_ident(s+1, bufend, tokenbuf, FALSE); expect = XOPERATOR; force_ident(tokenbuf); TOKEN(DOLSHARP); } s = scan_ident(s, bufend, tokenbuf+1, FALSE); if (tokenbuf[1]) { tokenbuf[0] = '$'; if (dowarn && *s == '[') { char *t; for (t = s+1; isSPACE(*t) || isALNUM(*t) || *t == '$'; t++) ; if (*t++ == ',') { bufptr = skipspace(bufptr); while (t < bufend && *t != ']') t++; warn("Multidimensional syntax %.*s not supported", t-bufptr+1, bufptr); } } expect = XOPERATOR; if (lex_state == LEX_NORMAL && isSPACE(*s)) { bool islop = (last_lop == oldoldbufptr); s = skipspace(s); if (strchr("$@\"'`q", *s)) expect = XTERM; /* e.g. print $fh "foo" */ else if (!islop) expect = XOPERATOR; else if (strchr("&*<%", *s) && isIDFIRST(s[1])) expect = XTERM; /* e.g. print $fh &sub */ else if (isDIGIT(*s)) expect = XTERM; /* e.g. print $fh 3 */ else if (*s == '.' && isDIGIT(s[1])) expect = XTERM; /* e.g. print $fh .3 */ else if (strchr("/?-+", *s) && !isSPACE(s[1])) expect = XTERM; /* e.g. print $fh -1 */ else if (*s == '<' && s[1] == '<' && !isSPACE(s[2])) expect = XTERM; /* print $fh <<"EOF" */ } if (in_my) { if (strchr(tokenbuf,':')) croak("\"my\" variable %s can't be in a package",tokenbuf); nextval[nexttoke].opval = newOP(OP_PADANY, 0); nextval[nexttoke].opval->op_targ = pad_allocmy(tokenbuf); force_next(PRIVATEREF); } else if (!strchr(tokenbuf,':')) { if (*s == '[') tokenbuf[0] = '@'; else if (*s == '{') tokenbuf[0] = '%'; if (tmp = pad_findmy(tokenbuf)) { nextval[nexttoke].opval = newOP(OP_PADANY, 0); nextval[nexttoke].opval->op_targ = tmp; force_next(PRIVATEREF); } else force_ident(tokenbuf+1); } else force_ident(tokenbuf+1); } else { if (s == bufend) yyerror("Final $ should be \\$ or $name"); PREREF('$'); } TOKEN('$'); case '@': if (expect == XOPERATOR) no_op("Array"); s = scan_ident(s, bufend, tokenbuf+1, FALSE); if (tokenbuf[1]) { tokenbuf[0] = '@'; expect = XOPERATOR; if (in_my) { if (strchr(tokenbuf,':')) croak("\"my\" variable %s can't be in a package",tokenbuf); nextval[nexttoke].opval = newOP(OP_PADANY, 0); nextval[nexttoke].opval->op_targ = pad_allocmy(tokenbuf); force_next(PRIVATEREF); TERM('@'); } else if (!strchr(tokenbuf,':')) { if (*s == '{') tokenbuf[0] = '%'; if (tmp = pad_findmy(tokenbuf)) { nextval[nexttoke].opval = newOP(OP_PADANY, 0); nextval[nexttoke].opval->op_targ = tmp; force_next(PRIVATEREF); TERM('@'); } } if (dowarn && *s == '[') { char *t; for (t = s+1; isSPACE(*t) || isALNUM(*t) || *t == '$'; t++) ; if (*t++ == ']') { bufptr = skipspace(bufptr); warn("Scalar value %.*s better written as $%.*s", t-bufptr, bufptr, t-bufptr-1, bufptr+1); } } force_ident(tokenbuf+1); } else { if (s == bufend) yyerror("Final @ should be \\@ or @name"); PREREF('@'); } TERM('@'); case '/': /* may either be division or pattern */ case '?': /* may either be conditional or pattern */ if (expect != XOPERATOR) { check_uni(); s = scan_pat(s); TERM(sublex_start()); } tmp = *s++; if (tmp == '/') Mop(OP_DIVIDE); OPERATOR(tmp); case '.': if (in_format == 2) { in_format = 0; expect = XBLOCK; goto rightbracket; } if (expect == XOPERATOR || !isDIGIT(s[1])) { tmp = *s++; if (*s == tmp) { s++; if (*s == tmp) { s++; yylval.ival = OPf_SPECIAL; } else yylval.ival = 0; OPERATOR(DOTDOT); } if (expect != XOPERATOR) check_uni(); Aop(OP_CONCAT); } /* FALL THROUGH */ case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': if (expect == XOPERATOR) no_op("Number"); s = scan_num(s); TERM(THING); case '\'': if (expect == XOPERATOR) { if (in_format) OPERATOR(','); /* grandfather non-comma-format format */ else no_op("String"); } s = scan_str(s); if (!s) croak("EOF in string"); yylval.ival = OP_CONST; TERM(sublex_start()); case '"': if (expect == XOPERATOR) { if (in_format) OPERATOR(','); /* grandfather non-comma-format format */ else no_op("String"); } s = scan_str(s); if (!s) croak("EOF in string"); yylval.ival = OP_SCALAR; TERM(sublex_start()); case '`': if (expect == XOPERATOR) no_op("Backticks"); s = scan_str(s); if (!s) croak("EOF in backticks"); yylval.ival = OP_BACKTICK; set_csh(); TERM(sublex_start()); case '\\': if (expect == XOPERATOR) no_op("Backslash"); s++; OPERATOR(REFGEN); case 'x': if (isDIGIT(s[1]) && expect == XOPERATOR) { s++; Mop(OP_REPEAT); } goto keylookup; case '_': case 'a': case 'A': case 'b': case 'B': case 'c': case 'C': case 'd': case 'D': case 'e': case 'E': case 'f': case 'F': case 'g': case 'G': case 'h': case 'H': case 'i': case 'I': case 'j': case 'J': case 'k': case 'K': case 'l': case 'L': case 'm': case 'M': case 'n': case 'N': case 'o': case 'O': case 'p': case 'P': case 'q': case 'Q': case 'r': case 'R': case 's': case 'S': case 't': case 'T': case 'u': case 'U': case 'v': case 'V': case 'w': case 'W': case 'X': case 'y': case 'Y': case 'z': case 'Z': keylookup: d = s; s = scan_word(s, tokenbuf, FALSE, &len); switch (tmp = keyword(tokenbuf, len)) { default: /* not a keyword */ just_a_word: { GV *gv; if (*s == '\'' || *s == ':') s = scan_word(s, tokenbuf + len, TRUE, &len); if (expect == XBLOCK) { /* special case: start of statement */ while (isSPACE(*s)) s++; if (*s == ':') { yylval.pval = savestr(tokenbuf); s++; CLINE; TOKEN(LABEL); } } else if (dowarn && expect == XOPERATOR) { if (bufptr == SvPVX(linestr)) { curcop->cop_line--; warn(warn_nosemi); curcop->cop_line++; } else no_op("Bare word"); } gv = gv_fetchpv(tokenbuf,FALSE); if (gv && GvCV(gv)) { nextval[nexttoke].opval = (OP*)newSVOP(OP_CONST, 0, newSVpv(tokenbuf,0)); nextval[nexttoke].opval->op_private = OPpCONST_BARE; s = skipspace(s); if (*s == '(') { expect = XTERM; force_next(WORD); TOKEN('&'); } else { last_lop = oldbufptr; expect = XBLOCK; force_next(WORD); TOKEN(NOAMP); } } expect = XOPERATOR; if (oldoldbufptr && oldoldbufptr < bufptr) { if (oldoldbufptr == last_lop) { expect = XTERM; CLINE; yylval.opval = (OP*)newSVOP(OP_CONST, 0, newSVpv(tokenbuf,0)); yylval.opval->op_private = OPpCONST_BARE; for (d = tokenbuf; *d && isLOWER(*d); d++) ; if (dowarn && !*d) warn(warn_reserved, tokenbuf); TOKEN(WORD); } } while (s < bufend && isSPACE(*s)) s++; if (*s == '(') { CLINE; nextval[nexttoke].opval = (OP*)newSVOP(OP_CONST, 0, newSVpv(tokenbuf,0)); nextval[nexttoke].opval->op_private = OPpCONST_BARE; expect = XOPERATOR; force_next(WORD); TOKEN('&'); } CLINE; yylval.opval = (OP*)newSVOP(OP_CONST, 0, newSVpv(tokenbuf,0)); yylval.opval->op_private = OPpCONST_BARE; if (*s == '$' || *s == '{') { last_lop = oldbufptr; PREBLOCK(METHOD); } if (isALPHA(*s)) { char *olds = s; char tmpbuf[1024]; s = scan_word(s, tmpbuf, TRUE, &len); if (!keyword(tmpbuf, len)) { gv = gv_fetchpv(tmpbuf,FALSE); if (!gv || !GvCV(gv)) { nextval[nexttoke].opval = (OP*)newSVOP(OP_CONST, 0, newSVpv(tmpbuf,0)); nextval[nexttoke].opval->op_private = OPpCONST_BARE; expect = XBLOCK; force_next(WORD); TOKEN(METHOD); } } s = olds; } for (d = tokenbuf; *d && isLOWER(*d); d++) ; if (dowarn && !*d) warn(warn_reserved, tokenbuf); TOKEN(WORD); } case KEY___LINE__: case KEY___FILE__: { if (tokenbuf[2] == 'L') (void)sprintf(tokenbuf,"%ld",(long)curcop->cop_line); else strcpy(tokenbuf, SvPVX(GvSV(curcop->cop_filegv))); yylval.opval = (OP*)newSVOP(OP_CONST, 0, newSVpv(tokenbuf,0)); TERM(THING); } case KEY___END__: { GV *gv; int fd; /*SUPPRESS 560*/ if (!in_eval && (gv = gv_fetchpv("DATA",FALSE))) { SvMULTI_on(gv); if (!GvIO(gv)) GvIO(gv) = newIO(); GvIO(gv)->ifp = rsfp; #if defined(HAS_FCNTL) && defined(FFt_SETFD) fd = fileno(rsfp); fcntl(fd,FFt_SETFD,fd >= 3); #endif if (preprocess) GvIO(gv)->type = '|'; else if ((FILE*)rsfp == stdin) GvIO(gv)->type = '-'; else GvIO(gv)->type = '<'; rsfp = Nullfp; } goto fake_eof; } case KEY_DESTROY: case KEY_BEGIN: case KEY_END: s = skipspace(s); if (expect == XBLOCK && (minus_p || minus_n || *s == '{' )) { s = bufptr; goto really_sub; } goto just_a_word; case KEY_abs: UNI(OP_ABS); case KEY_alarm: UNI(OP_ALARM); case KEY_accept: LOP(OP_ACCEPT); case KEY_and: OPERATOR(ANDOP); case KEY_atan2: LOP(OP_ATAN2); case KEY_bind: LOP(OP_BIND); case KEY_binmode: UNI(OP_BINMODE); case KEY_bless: LOP(OP_BLESS); case KEY_chop: UNI(OP_CHOP); case KEY_continue: PREBLOCK(CONTINUE); case KEY_chdir: (void)gv_fetchpv("ENV",TRUE); /* may use HOME */ UNI(OP_CHDIR); case KEY_close: UNI(OP_CLOSE); case KEY_closedir: UNI(OP_CLOSEDIR); case KEY_cmp: Eop(OP_SCMP); case KEY_caller: UNI(OP_CALLER); case KEY_crypt: #ifdef FCRYPT if (!cryptseen++) init_des(); #endif LOP(OP_CRYPT); case KEY_chmod: s = skipspace(s); if (dowarn && *s != '0' && isDIGIT(*s)) warn("chmod: mode argument is missing initial 0"); LOP(OP_CHMOD); case KEY_chown: LOP(OP_CHOWN); case KEY_connect: LOP(OP_CONNECT); case KEY_chr: UNI(OP_CHR); case KEY_cos: UNI(OP_COS); case KEY_chroot: UNI(OP_CHROOT); case KEY_do: s = skipspace(s); if (*s == '{') PREBLOCK(DO); if (*s != '\'') s = force_word(s,WORD,FALSE,TRUE); OPERATOR(DO); case KEY_die: LOP(OP_DIE); case KEY_defined: UNI(OP_DEFINED); case KEY_delete: OPERATOR(DELETE); case KEY_dbmopen: LOP(OP_DBMOPEN); case KEY_dbmclose: UNI(OP_DBMCLOSE); case KEY_dump: LOOPX(OP_DUMP); case KEY_else: PREBLOCK(ELSE); case KEY_elsif: yylval.ival = curcop->cop_line; OPERATOR(ELSIF); case KEY_eq: Eop(OP_SEQ); case KEY_exit: UNI(OP_EXIT); case KEY_eval: s = skipspace(s); expect = (*s == '{') ? XBLOCK : XTERM; UNIBRACK(OP_ENTEREVAL); case KEY_eof: UNI(OP_EOF); case KEY_exp: UNI(OP_EXP); case KEY_each: UNI(OP_EACH); case KEY_exec: set_csh(); LOP(OP_EXEC); case KEY_endhostent: FUN0(OP_EHOSTENT); case KEY_endnetent: FUN0(OP_ENETENT); case KEY_endservent: FUN0(OP_ESERVENT); case KEY_endprotoent: FUN0(OP_EPROTOENT); case KEY_endpwent: FUN0(OP_EPWENT); case KEY_endgrent: FUN0(OP_EGRENT); case KEY_for: case KEY_foreach: yylval.ival = curcop->cop_line; while (s < bufend && isSPACE(*s)) s++; if (isIDFIRST(*s)) croak("Missing $ on loop variable"); OPERATOR(FOR); case KEY_formline: LOP(OP_FORMLINE); case KEY_fork: FUN0(OP_FORK); case KEY_fcntl: LOP(OP_FCNTL); case KEY_fileno: UNI(OP_FILENO); case KEY_flock: LOP(OP_FLOCK); case KEY_gt: Rop(OP_SGT); case KEY_ge: Rop(OP_SGE); case KEY_grep: LOP(OP_GREPSTART); case KEY_goto: LOOPX(OP_GOTO); case KEY_gmtime: UNI(OP_GMTIME); case KEY_getc: UNI(OP_GETC); case KEY_getppid: FUN0(OP_GETPPID); case KEY_getpgrp: UNI(OP_GETPGRP); case KEY_getpriority: LOP(OP_GETPRIORITY); case KEY_getprotobyname: UNI(OP_GPBYNAME); case KEY_getprotobynumber: LOP(OP_GPBYNUMBER); case KEY_getprotoent: FUN0(OP_GPROTOENT); case KEY_getpwent: FUN0(OP_GPWENT); case KEY_getpwnam: FUN1(OP_GPWNAM); case KEY_getpwuid: FUN1(OP_GPWUID); case KEY_getpeername: UNI(OP_GETPEERNAME); case KEY_gethostbyname: UNI(OP_GHBYNAME); case KEY_gethostbyaddr: LOP(OP_GHBYADDR); case KEY_gethostent: FUN0(OP_GHOSTENT); case KEY_getnetbyname: UNI(OP_GNBYNAME); case KEY_getnetbyaddr: LOP(OP_GNBYADDR); case KEY_getnetent: FUN0(OP_GNETENT); case KEY_getservbyname: LOP(OP_GSBYNAME); case KEY_getservbyport: LOP(OP_GSBYPORT); case KEY_getservent: FUN0(OP_GSERVENT); case KEY_getsockname: UNI(OP_GETSOCKNAME); case KEY_getsockopt: LOP(OP_GSOCKOPT); case KEY_getgrent: FUN0(OP_GGRENT); case KEY_getgrnam: FUN1(OP_GGRNAM); case KEY_getgrgid: FUN1(OP_GGRGID); case KEY_getlogin: FUN0(OP_GETLOGIN); case KEY_glob: UNI(OP_GLOB); case KEY_hex: UNI(OP_HEX); case KEY_if: yylval.ival = curcop->cop_line; OPERATOR(IF); case KEY_index: LOP(OP_INDEX); case KEY_int: UNI(OP_INT); case KEY_ioctl: LOP(OP_IOCTL); case KEY_join: LOP(OP_JOIN); case KEY_keys: UNI(OP_KEYS); case KEY_kill: LOP(OP_KILL); case KEY_last: s = force_word(s,WORD,TRUE,FALSE); LOOPX(OP_LAST); case KEY_lc: UNI(OP_LC); case KEY_lcfirst: UNI(OP_LCFIRST); case KEY_local: yylval.ival = 0; OPERATOR(LOCAL); case KEY_length: UNI(OP_LENGTH); case KEY_lt: Rop(OP_SLT); case KEY_le: Rop(OP_SLE); case KEY_localtime: UNI(OP_LOCALTIME); case KEY_log: UNI(OP_LOG); case KEY_link: LOP(OP_LINK); case KEY_listen: LOP(OP_LISTEN); case KEY_lstat: UNI(OP_LSTAT); case KEY_m: s = scan_pat(s); TERM(sublex_start()); case KEY_mkdir: LOP(OP_MKDIR); case KEY_msgctl: LOP(OP_MSGCTL); case KEY_msgget: LOP(OP_MSGGET); case KEY_msgrcv: LOP(OP_MSGRCV); case KEY_msgsnd: LOP(OP_MSGSND); case KEY_my: in_my = TRUE; yylval.ival = 1; OPERATOR(LOCAL); case KEY_next: s = force_word(s,WORD,TRUE,FALSE); LOOPX(OP_NEXT); case KEY_ne: Eop(OP_SNE); case KEY_open: s = skipspace(s); if (isIDFIRST(*s)) { char *t; for (d = s; isALNUM(*d); d++) ; t = skipspace(d); if (strchr("|&*+-=!?:.", *t)) warn("Precedence problem: open %.*s should be open(%.*s)", d-s,s, d-s,s); } LOP(OP_OPEN); case KEY_or: OPERATOR(OROP); case KEY_ord: UNI(OP_ORD); case KEY_oct: UNI(OP_OCT); case KEY_opendir: LOP(OP_OPEN_DIR); case KEY_print: checkcomma(s,tokenbuf,"filehandle"); LOP(OP_PRINT); case KEY_printf: checkcomma(s,tokenbuf,"filehandle"); LOP(OP_PRTF); case KEY_push: LOP(OP_PUSH); case KEY_pop: UNI(OP_POP); case KEY_pack: LOP(OP_PACK); case KEY_package: s = force_word(s,WORD,FALSE,TRUE); OPERATOR(PACKAGE); case KEY_pipe: LOP(OP_PIPE_OP); case KEY_q: s = scan_str(s); if (!s) croak("EOF in string"); yylval.ival = OP_CONST; TERM(sublex_start()); case KEY_qq: s = scan_str(s); if (!s) croak("EOF in string"); yylval.ival = OP_SCALAR; if (SvIVX(lex_stuff) == '\'') SvIVX(lex_stuff) = 0; /* qq'$foo' should intepolate */ TERM(sublex_start()); case KEY_qx: s = scan_str(s); if (!s) croak("EOF in string"); yylval.ival = OP_BACKTICK; set_csh(); TERM(sublex_start()); case KEY_return: OLDLOP(OP_RETURN); case KEY_require: UNI(OP_REQUIRE); case KEY_reset: UNI(OP_RESET); case KEY_redo: s = force_word(s,WORD,TRUE,FALSE); LOOPX(OP_REDO); case KEY_rename: LOP(OP_RENAME); case KEY_rand: UNI(OP_RAND); case KEY_rmdir: UNI(OP_RMDIR); case KEY_rindex: LOP(OP_RINDEX); case KEY_read: LOP(OP_READ); case KEY_readdir: UNI(OP_READDIR); case KEY_readline: set_csh(); UNI(OP_READLINE); case KEY_readpipe: set_csh(); UNI(OP_BACKTICK); case KEY_rewinddir: UNI(OP_REWINDDIR); case KEY_recv: LOP(OP_RECV); case KEY_reverse: LOP(OP_REVERSE); case KEY_readlink: UNI(OP_READLINK); case KEY_ref: UNI(OP_REF); case KEY_s: s = scan_subst(s); if (yylval.opval) TERM(sublex_start()); else TOKEN(1); /* force error */ case KEY_scalar: UNI(OP_SCALAR); case KEY_select: LOP(OP_SELECT); case KEY_seek: LOP(OP_SEEK); case KEY_semctl: LOP(OP_SEMCTL); case KEY_semget: LOP(OP_SEMGET); case KEY_semop: LOP(OP_SEMOP); case KEY_send: LOP(OP_SEND); case KEY_setpgrp: LOP(OP_SETPGRP); case KEY_setpriority: LOP(OP_SETPRIORITY); case KEY_sethostent: FUN1(OP_SHOSTENT); case KEY_setnetent: FUN1(OP_SNETENT); case KEY_setservent: FUN1(OP_SSERVENT); case KEY_setprotoent: FUN1(OP_SPROTOENT); case KEY_setpwent: FUN0(OP_SPWENT); case KEY_setgrent: FUN0(OP_SGRENT); case KEY_seekdir: LOP(OP_SEEKDIR); case KEY_setsockopt: LOP(OP_SSOCKOPT); case KEY_shift: UNI(OP_SHIFT); case KEY_shmctl: LOP(OP_SHMCTL); case KEY_shmget: LOP(OP_SHMGET); case KEY_shmread: LOP(OP_SHMREAD); case KEY_shmwrite: LOP(OP_SHMWRITE); case KEY_shutdown: LOP(OP_SHUTDOWN); case KEY_sin: UNI(OP_SIN); case KEY_sleep: UNI(OP_SLEEP); case KEY_socket: LOP(OP_SOCKET); case KEY_socketpair: LOP(OP_SOCKPAIR); case KEY_sort: checkcomma(s,tokenbuf,"subroutine name"); s = skipspace(s); if (*s == ';' || *s == ')') /* probably a close */ croak("sort is now a reserved word"); expect = XTERM; s = force_word(s,WORD,TRUE,TRUE); LOP(OP_SORT); case KEY_split: LOP(OP_SPLIT); case KEY_sprintf: LOP(OP_SPRINTF); case KEY_splice: LOP(OP_SPLICE); case KEY_sqrt: UNI(OP_SQRT); case KEY_srand: UNI(OP_SRAND); case KEY_stat: UNI(OP_STAT); case KEY_study: sawstudy++; UNI(OP_STUDY); case KEY_substr: LOP(OP_SUBSTR); case KEY_format: case KEY_sub: really_sub: yylval.ival = savestack_ix; /* restore stuff on reduce */ save_I32(&subline); save_item(subname); SAVEINT(padix); SAVESPTR(curpad); SAVESPTR(comppad); SAVESPTR(comppadname); SAVEINT(comppadnamefill); comppad = newAV(); comppadname = newAV(); comppadnamefill = -1; av_push(comppad, Nullsv); curpad = AvARRAY(comppad); padix = 0; subline = curcop->cop_line; s = skipspace(s); if (tmp == KEY_format) expect = XTERM; else expect = XBLOCK; if (isIDFIRST(*s) || *s == '\'' || *s == ':') { char tmpbuf[128]; d = scan_word(s, tmpbuf, TRUE, &len); if (strchr(tmpbuf, ':')) sv_setpv(subname, tmpbuf); else { sv_setsv(subname,curstname); sv_catpvn(subname,"'",1); sv_catpvn(subname,tmpbuf,len); } s = force_word(s,WORD,FALSE,TRUE); } else sv_setpv(subname,"?"); if (tmp != KEY_format) PREBLOCK(SUB); in_format = 2; lex_brackets = 0; OPERATOR(FORMAT); case KEY_system: set_csh(); LOP(OP_SYSTEM); case KEY_symlink: LOP(OP_SYMLINK); case KEY_syscall: LOP(OP_SYSCALL); case KEY_sysread: LOP(OP_SYSREAD); case KEY_syswrite: LOP(OP_SYSWRITE); case KEY_tr: s = scan_trans(s); TERM(sublex_start()); case KEY_tell: UNI(OP_TELL); case KEY_telldir: UNI(OP_TELLDIR); case KEY_tie: LOP(OP_TIE); case KEY_time: FUN0(OP_TIME); case KEY_times: FUN0(OP_TMS); case KEY_truncate: LOP(OP_TRUNCATE); case KEY_uc: UNI(OP_UC); case KEY_ucfirst: UNI(OP_UCFIRST); case KEY_untie: UNI(OP_UNTIE); case KEY_until: yylval.ival = curcop->cop_line; OPERATOR(UNTIL); case KEY_unless: yylval.ival = curcop->cop_line; OPERATOR(UNLESS); case KEY_unlink: LOP(OP_UNLINK); case KEY_undef: UNI(OP_UNDEF); case KEY_unpack: LOP(OP_UNPACK); case KEY_utime: LOP(OP_UTIME); case KEY_umask: s = skipspace(s); if (dowarn && *s != '0' && isDIGIT(*s)) warn("umask: argument is missing initial 0"); UNI(OP_UMASK); case KEY_unshift: LOP(OP_UNSHIFT); case KEY_values: UNI(OP_VALUES); case KEY_vec: sawvec = TRUE; LOP(OP_VEC); case KEY_while: yylval.ival = curcop->cop_line; OPERATOR(WHILE); case KEY_warn: LOP(OP_WARN); case KEY_wait: FUN0(OP_WAIT); case KEY_waitpid: LOP(OP_WAITPID); case KEY_wantarray: FUN0(OP_WANTARRAY); case KEY_write: UNI(OP_ENTERWRITE); case KEY_x: if (expect == XOPERATOR) Mop(OP_REPEAT); check_uni(); goto just_a_word; case KEY_y: s = scan_trans(s); TERM(sublex_start()); } } } I32 keyword(d, len) register char *d; I32 len; { switch (*d) { case '_': if (d[1] == '_') { if (strEQ(d,"__LINE__")) return KEY___LINE__; if (strEQ(d,"__FILE__")) return KEY___FILE__; if (strEQ(d,"__END__")) return KEY___END__; } break; case 'a': switch (len) { case 3: if (strEQ(d,"and")) return KEY_and; if (strEQ(d,"abs")) return KEY_abs; break; case 5: if (strEQ(d,"alarm")) return KEY_alarm; if (strEQ(d,"atan2")) return KEY_atan2; break; case 6: if (strEQ(d,"accept")) return KEY_accept; break; } break; case 'B': if (strEQ(d,"BEGIN")) return KEY_BEGIN; break; case 'b': if (strEQ(d,"bless")) return KEY_bless; if (strEQ(d,"bind")) return KEY_bind; if (strEQ(d,"binmode")) return KEY_binmode; break; case 'c': switch (len) { case 3: if (strEQ(d,"cmp")) return KEY_cmp; if (strEQ(d,"chr")) return KEY_chr; if (strEQ(d,"cos")) return KEY_cos; break; case 4: if (strEQ(d,"chop")) return KEY_chop; break; case 5: if (strEQ(d,"close")) return KEY_close; if (strEQ(d,"chdir")) return KEY_chdir; if (strEQ(d,"chmod")) return KEY_chmod; if (strEQ(d,"chown")) return KEY_chown; if (strEQ(d,"crypt")) return KEY_crypt; break; case 6: if (strEQ(d,"chroot")) return KEY_chroot; if (strEQ(d,"caller")) return KEY_caller; break; case 7: if (strEQ(d,"connect")) return KEY_connect; break; case 8: if (strEQ(d,"closedir")) return KEY_closedir; if (strEQ(d,"continue")) return KEY_continue; break; } break; case 'D': if (strEQ(d,"DESTROY")) return KEY_DESTROY; break; case 'd': switch (len) { case 2: if (strEQ(d,"do")) return KEY_do; break; case 3: if (strEQ(d,"die")) return KEY_die; break; case 4: if (strEQ(d,"dump")) return KEY_dump; break; case 6: if (strEQ(d,"delete")) return KEY_delete; break; case 7: if (strEQ(d,"defined")) return KEY_defined; if (strEQ(d,"dbmopen")) return KEY_dbmopen; break; case 8: if (strEQ(d,"dbmclose")) return KEY_dbmclose; break; } break; case 'E': if (strEQ(d,"EQ")) return KEY_eq; if (strEQ(d,"END")) return KEY_END; break; case 'e': switch (len) { case 2: if (strEQ(d,"eq")) return KEY_eq; break; case 3: if (strEQ(d,"eof")) return KEY_eof; if (strEQ(d,"exp")) return KEY_exp; break; case 4: if (strEQ(d,"else")) return KEY_else; if (strEQ(d,"exit")) return KEY_exit; if (strEQ(d,"eval")) return KEY_eval; if (strEQ(d,"exec")) return KEY_exec; if (strEQ(d,"each")) return KEY_each; break; case 5: if (strEQ(d,"elsif")) return KEY_elsif; break; case 8: if (strEQ(d,"endgrent")) return KEY_endgrent; if (strEQ(d,"endpwent")) return KEY_endpwent; break; case 9: if (strEQ(d,"endnetent")) return KEY_endnetent; break; case 10: if (strEQ(d,"endhostent")) return KEY_endhostent; if (strEQ(d,"endservent")) return KEY_endservent; break; case 11: if (strEQ(d,"endprotoent")) return KEY_endprotoent; break; } break; case 'f': switch (len) { case 3: if (strEQ(d,"for")) return KEY_for; break; case 4: if (strEQ(d,"fork")) return KEY_fork; break; case 5: if (strEQ(d,"fcntl")) return KEY_fcntl; if (strEQ(d,"flock")) return KEY_flock; break; case 6: if (strEQ(d,"format")) return KEY_format; if (strEQ(d,"fileno")) return KEY_fileno; break; case 7: if (strEQ(d,"foreach")) return KEY_foreach; break; case 8: if (strEQ(d,"formline")) return KEY_formline; break; } break; case 'G': if (len == 2) { if (strEQ(d,"GT")) return KEY_gt; if (strEQ(d,"GE")) return KEY_ge; } break; case 'g': if (strnEQ(d,"get",3)) { d += 3; if (*d == 'p') { switch (len) { case 7: if (strEQ(d,"ppid")) return KEY_getppid; if (strEQ(d,"pgrp")) return KEY_getpgrp; break; case 8: if (strEQ(d,"pwent")) return KEY_getpwent; if (strEQ(d,"pwnam")) return KEY_getpwnam; if (strEQ(d,"pwuid")) return KEY_getpwuid; break; case 11: if (strEQ(d,"peername")) return KEY_getpeername; if (strEQ(d,"protoent")) return KEY_getprotoent; if (strEQ(d,"priority")) return KEY_getpriority; break; case 14: if (strEQ(d,"protobyname")) return KEY_getprotobyname; break; case 16: if (strEQ(d,"protobynumber"))return KEY_getprotobynumber; break; } } else if (*d == 'h') { if (strEQ(d,"hostbyname")) return KEY_gethostbyname; if (strEQ(d,"hostbyaddr")) return KEY_gethostbyaddr; if (strEQ(d,"hostent")) return KEY_gethostent; } else if (*d == 'n') { if (strEQ(d,"netbyname")) return KEY_getnetbyname; if (strEQ(d,"netbyaddr")) return KEY_getnetbyaddr; if (strEQ(d,"netent")) return KEY_getnetent; } else if (*d == 's') { if (strEQ(d,"servbyname")) return KEY_getservbyname; if (strEQ(d,"servbyport")) return KEY_getservbyport; if (strEQ(d,"servent")) return KEY_getservent; if (strEQ(d,"sockname")) return KEY_getsockname; if (strEQ(d,"sockopt")) return KEY_getsockopt; } else if (*d == 'g') { if (strEQ(d,"grent")) return KEY_getgrent; if (strEQ(d,"grnam")) return KEY_getgrnam; if (strEQ(d,"grgid")) return KEY_getgrgid; } else if (*d == 'l') { if (strEQ(d,"login")) return KEY_getlogin; } else if (strEQ(d,"c")) return KEY_getc; break; } switch (len) { case 2: if (strEQ(d,"gt")) return KEY_gt; if (strEQ(d,"ge")) return KEY_ge; break; case 4: if (strEQ(d,"grep")) return KEY_grep; if (strEQ(d,"goto")) return KEY_goto; if (strEQ(d,"glob")) return KEY_glob; break; case 6: if (strEQ(d,"gmtime")) return KEY_gmtime; break; } break; case 'h': if (strEQ(d,"hex")) return KEY_hex; break; case 'i': switch (len) { case 2: if (strEQ(d,"if")) return KEY_if; break; case 3: if (strEQ(d,"int")) return KEY_int; break; case 5: if (strEQ(d,"index")) return KEY_index; if (strEQ(d,"ioctl")) return KEY_ioctl; break; } break; case 'j': if (strEQ(d,"join")) return KEY_join; break; case 'k': if (len == 4) { if (strEQ(d,"keys")) return KEY_keys; if (strEQ(d,"kill")) return KEY_kill; } break; case 'L': if (len == 2) { if (strEQ(d,"LT")) return KEY_lt; if (strEQ(d,"LE")) return KEY_le; } break; case 'l': switch (len) { case 2: if (strEQ(d,"lt")) return KEY_lt; if (strEQ(d,"le")) return KEY_le; if (strEQ(d,"lc")) return KEY_lc; break; case 3: if (strEQ(d,"log")) return KEY_log; break; case 4: if (strEQ(d,"last")) return KEY_last; if (strEQ(d,"link")) return KEY_link; break; case 5: if (strEQ(d,"local")) return KEY_local; if (strEQ(d,"lstat")) return KEY_lstat; break; case 6: if (strEQ(d,"length")) return KEY_length; if (strEQ(d,"listen")) return KEY_listen; break; case 7: if (strEQ(d,"lcfirst")) return KEY_lcfirst; break; case 9: if (strEQ(d,"localtime")) return KEY_localtime; break; } break; case 'm': switch (len) { case 1: return KEY_m; case 2: if (strEQ(d,"my")) return KEY_my; break; case 5: if (strEQ(d,"mkdir")) return KEY_mkdir; break; case 6: if (strEQ(d,"msgctl")) return KEY_msgctl; if (strEQ(d,"msgget")) return KEY_msgget; if (strEQ(d,"msgrcv")) return KEY_msgrcv; if (strEQ(d,"msgsnd")) return KEY_msgsnd; break; } break; case 'N': if (strEQ(d,"NE")) return KEY_ne; break; case 'n': if (strEQ(d,"next")) return KEY_next; if (strEQ(d,"ne")) return KEY_ne; break; case 'o': switch (len) { case 2: if (strEQ(d,"or")) return KEY_or; break; case 3: if (strEQ(d,"ord")) return KEY_ord; if (strEQ(d,"oct")) return KEY_oct; break; case 4: if (strEQ(d,"open")) return KEY_open; break; case 7: if (strEQ(d,"opendir")) return KEY_opendir; break; } break; case 'p': switch (len) { case 3: if (strEQ(d,"pop")) return KEY_pop; break; case 4: if (strEQ(d,"push")) return KEY_push; if (strEQ(d,"pack")) return KEY_pack; if (strEQ(d,"pipe")) return KEY_pipe; break; case 5: if (strEQ(d,"print")) return KEY_print; break; case 6: if (strEQ(d,"printf")) return KEY_printf; break; case 7: if (strEQ(d,"package")) return KEY_package; break; } break; case 'q': if (len <= 2) { if (strEQ(d,"q")) return KEY_q; if (strEQ(d,"qq")) return KEY_qq; if (strEQ(d,"qx")) return KEY_qx; } break; case 'r': switch (len) { case 3: if (strEQ(d,"ref")) return KEY_ref; break; case 4: if (strEQ(d,"read")) return KEY_read; if (strEQ(d,"rand")) return KEY_rand; if (strEQ(d,"recv")) return KEY_recv; if (strEQ(d,"redo")) return KEY_redo; break; case 5: if (strEQ(d,"rmdir")) return KEY_rmdir; if (strEQ(d,"reset")) return KEY_reset; break; case 6: if (strEQ(d,"return")) return KEY_return; if (strEQ(d,"rename")) return KEY_rename; if (strEQ(d,"rindex")) return KEY_rindex; break; case 7: if (strEQ(d,"require")) return KEY_require; if (strEQ(d,"reverse")) return KEY_reverse; if (strEQ(d,"readdir")) return KEY_readdir; break; case 8: if (strEQ(d,"readlink")) return KEY_readlink; if (strEQ(d,"readline")) return KEY_readline; if (strEQ(d,"readpipe")) return KEY_readpipe; break; case 9: if (strEQ(d,"rewinddir")) return KEY_rewinddir; break; } break; case 's': switch (d[1]) { case 0: return KEY_s; case 'c': if (strEQ(d,"scalar")) return KEY_scalar; break; case 'e': switch (len) { case 4: if (strEQ(d,"seek")) return KEY_seek; if (strEQ(d,"send")) return KEY_send; break; case 5: if (strEQ(d,"semop")) return KEY_semop; break; case 6: if (strEQ(d,"select")) return KEY_select; if (strEQ(d,"semctl")) return KEY_semctl; if (strEQ(d,"semget")) return KEY_semget; break; case 7: if (strEQ(d,"setpgrp")) return KEY_setpgrp; if (strEQ(d,"seekdir")) return KEY_seekdir; break; case 8: if (strEQ(d,"setpwent")) return KEY_setpwent; if (strEQ(d,"setgrent")) return KEY_setgrent; break; case 9: if (strEQ(d,"setnetent")) return KEY_setnetent; break; case 10: if (strEQ(d,"setsockopt")) return KEY_setsockopt; if (strEQ(d,"sethostent")) return KEY_sethostent; if (strEQ(d,"setservent")) return KEY_setservent; break; case 11: if (strEQ(d,"setpriority")) return KEY_setpriority; if (strEQ(d,"setprotoent")) return KEY_setprotoent; break; } break; case 'h': switch (len) { case 5: if (strEQ(d,"shift")) return KEY_shift; break; case 6: if (strEQ(d,"shmctl")) return KEY_shmctl; if (strEQ(d,"shmget")) return KEY_shmget; break; case 7: if (strEQ(d,"shmread")) return KEY_shmread; break; case 8: if (strEQ(d,"shmwrite")) return KEY_shmwrite; if (strEQ(d,"shutdown")) return KEY_shutdown; break; } break; case 'i': if (strEQ(d,"sin")) return KEY_sin; break; case 'l': if (strEQ(d,"sleep")) return KEY_sleep; break; case 'o': if (strEQ(d,"sort")) return KEY_sort; if (strEQ(d,"socket")) return KEY_socket; if (strEQ(d,"socketpair")) return KEY_socketpair; break; case 'p': if (strEQ(d,"split")) return KEY_split; if (strEQ(d,"sprintf")) return KEY_sprintf; if (strEQ(d,"splice")) return KEY_splice; break; case 'q': if (strEQ(d,"sqrt")) return KEY_sqrt; break; case 'r': if (strEQ(d,"srand")) return KEY_srand; break; case 't': if (strEQ(d,"stat")) return KEY_stat; if (strEQ(d,"study")) return KEY_study; break; case 'u': if (strEQ(d,"substr")) return KEY_substr; if (strEQ(d,"sub")) return KEY_sub; break; case 'y': switch (len) { case 6: if (strEQ(d,"system")) return KEY_system; break; case 7: if (strEQ(d,"sysread")) return KEY_sysread; if (strEQ(d,"symlink")) return KEY_symlink; if (strEQ(d,"syscall")) return KEY_syscall; break; case 8: if (strEQ(d,"syswrite")) return KEY_syswrite; break; } break; } break; case 't': switch (len) { case 2: if (strEQ(d,"tr")) return KEY_tr; break; case 3: if (strEQ(d,"tie")) return KEY_tie; break; case 4: if (strEQ(d,"tell")) return KEY_tell; if (strEQ(d,"time")) return KEY_time; break; case 5: if (strEQ(d,"times")) return KEY_times; break; case 7: if (strEQ(d,"telldir")) return KEY_telldir; break; case 8: if (strEQ(d,"truncate")) return KEY_truncate; break; } break; case 'u': switch (len) { case 2: if (strEQ(d,"uc")) return KEY_uc; break; case 5: if (strEQ(d,"undef")) return KEY_undef; if (strEQ(d,"until")) return KEY_until; if (strEQ(d,"untie")) return KEY_untie; if (strEQ(d,"utime")) return KEY_utime; if (strEQ(d,"umask")) return KEY_umask; break; case 6: if (strEQ(d,"unless")) return KEY_unless; if (strEQ(d,"unpack")) return KEY_unpack; if (strEQ(d,"unlink")) return KEY_unlink; break; case 7: if (strEQ(d,"unshift")) return KEY_unshift; if (strEQ(d,"ucfirst")) return KEY_ucfirst; break; } break; case 'v': if (strEQ(d,"values")) return KEY_values; if (strEQ(d,"vec")) return KEY_vec; break; case 'w': switch (len) { case 4: if (strEQ(d,"warn")) return KEY_warn; if (strEQ(d,"wait")) return KEY_wait; break; case 5: if (strEQ(d,"while")) return KEY_while; if (strEQ(d,"write")) return KEY_write; break; case 7: if (strEQ(d,"waitpid")) return KEY_waitpid; break; case 9: if (strEQ(d,"wantarray")) return KEY_wantarray; break; } break; case 'x': if (len == 1) return KEY_x; break; case 'y': if (len == 1) return KEY_y; break; case 'z': break; } return 0; } void checkcomma(s,name,what) register char *s; char *name; char *what; { char *w; if (dowarn && *s == ' ' && s[1] == '(') { /* XXX gotta be a better way */ w = strchr(s,')'); if (w) for (w++; *w && isSPACE(*w); w++) ; if (!w || !*w || !strchr(";|}", *w)) /* an advisory hack only... */ warn("%s (...) interpreted as function",name); } while (s < bufend && isSPACE(*s)) s++; if (*s == '(') s++; while (s < bufend && isSPACE(*s)) s++; if (isIDFIRST(*s)) { w = s++; while (isALNUM(*s)) s++; while (s < bufend && isSPACE(*s)) s++; if (*s == ',') { int kw; *s = '\0'; kw = keyword(w, s - w); *s = ','; if (kw) return; croak("No comma allowed after %s", what); } } } char * scan_word(s, dest, allow_package, slp) register char *s; char *dest; int allow_package; STRLEN *slp; { register char *d = dest; for (;;) { if (isALNUM(*s)) *d++ = *s++; else if (*s == '\'' && allow_package && isIDFIRST(s[1])) { *d++ = ':'; *d++ = ':'; s++; } else if (*s == ':' && s[1] == ':' && allow_package && isIDFIRST(s[2])) { *d++ = *s++; *d++ = *s++; } else { *d = '\0'; *slp = d - dest; return s; } } } char * scan_ident(s,send,dest,ck_uni) register char *s; register char *send; char *dest; I32 ck_uni; { register char *d; char *bracket = 0; if (lex_brackets == 0) lex_fakebrack = 0; s++; d = dest; if (isDIGIT(*s)) { while (isDIGIT(*s)) *d++ = *s++; } else { for (;;) { if (isALNUM(*s)) *d++ = *s++; else if (*s == '\'' && isIDFIRST(s[1])) { *d++ = ':'; *d++ = ':'; s++; } else if (*s == ':' && s[1] == ':' && isIDFIRST(s[2])) { *d++ = *s++; *d++ = *s++; } else break; } } *d = '\0'; d = dest; if (*d) { if (lex_state != LEX_NORMAL) lex_state = LEX_INTERPENDMAYBE; return s; } if (isSPACE(*s) || (*s == '$' && (isALPHA(s[1]) || s[1] == '$' || s[1] == '_'))) return s; if (*s == '{') { bracket = s; s++; } else if (ck_uni) check_uni(); if (s < send) *d = *s++; d[1] = '\0'; if (*d == '^' && (isUPPER(*s) || strchr("[\\]^_?", *s))) { if (*s == 'D') debug |= 32768; *d = *s++ ^ 64; } if (bracket) { if (isALPHA(*d) || *d == '_') { d++; while (isALNUM(*s)) *d++ = *s++; *d = '\0'; if (*s == '[' || *s == '{') { if (lex_brackets) croak("Can't use delimiter brackets within expression"); lex_fakebrack = TRUE; bracket++; lex_brackets++; return s; } } if (*s == '}') { s++; if (lex_state == LEX_INTERPNORMAL && !lex_brackets) lex_state = LEX_INTERPEND; } else { s = bracket; /* let the parser handle it */ *dest = '\0'; } } else if (lex_state == LEX_INTERPNORMAL && !lex_brackets && !intuit_more(s)) lex_state = LEX_INTERPEND; return s; } void scan_prefix(pm,string,len) PMOP *pm; char *string; I32 len; { register SV *tmpstr; register char *t; register char *d; register char *e; char *origstring = string; if (ninstr(string, string+len, vert, vert+1)) return; if (*string == '^') string++, len--; tmpstr = NEWSV(86,len); sv_upgrade(tmpstr, SVt_PVBM); sv_setpvn(tmpstr,string,len); t = SvPVX(tmpstr); e = t + len; BmUSEFUL(tmpstr) = 100; for (d=t; d < e; ) { switch (*d) { case '{': if (isDIGIT(d[1])) e = d; else goto defchar; break; case '.': case '[': case '$': case '(': case ')': case '|': case '+': case '^': e = d; break; case '\\': if (d[1] && strchr("wWbB0123456789sSdDlLuUExc",d[1])) { e = d; break; } Move(d+1,d,e-d,char); e--; switch(*d) { case 'n': *d = '\n'; break; case 't': *d = '\t'; break; case 'f': *d = '\f'; break; case 'r': *d = '\r'; break; case 'e': *d = '\033'; break; case 'a': *d = '\007'; break; } /* FALL THROUGH */ default: defchar: if (d[1] == '*' || (d[1] == '{' && d[2] == '0') || d[1] == '?') { e = d; break; } d++; } } if (d == t) { sv_free(tmpstr); return; } *d = '\0'; SvCUR_set(tmpstr, d - t); if (d == t+len) pm->op_pmflags |= PMf_ALL; if (*origstring != '^') pm->op_pmflags |= PMf_SCANFIRST; pm->op_pmshort = tmpstr; pm->op_pmslen = d - t; } char * scan_pat(start) char *start; { PMOP *pm; char *s; multi_start = curcop->cop_line; s = scan_str(start); if (!s) { if (lex_stuff) sv_free(lex_stuff); lex_stuff = Nullsv; croak("Search pattern not terminated"); } pm = (PMOP*)newPMOP(OP_MATCH, 0); if (*start == '?') pm->op_pmflags |= PMf_ONCE; while (*s == 'i' || *s == 'o' || *s == 'g') { if (*s == 'i') { s++; sawi = TRUE; pm->op_pmflags |= PMf_FOLD; } if (*s == 'o') { s++; pm->op_pmflags |= PMf_KEEP; } if (*s == 'g') { s++; pm->op_pmflags |= PMf_GLOBAL; } } lex_op = (OP*)pm; yylval.ival = OP_MATCH; return s; } char * scan_subst(start) char *start; { register char *s = start; register PMOP *pm; I32 es = 0; multi_start = curcop->cop_line; yylval.ival = OP_NULL; s = scan_str(s); if (!s) { if (lex_stuff) sv_free(lex_stuff); lex_stuff = Nullsv; croak("Substitution pattern not terminated"); } if (s[-1] == *start) s--; s = scan_str(s); if (!s) { if (lex_stuff) sv_free(lex_stuff); lex_stuff = Nullsv; if (lex_repl) sv_free(lex_repl); lex_repl = Nullsv; croak("Substitution replacement not terminated"); } pm = (PMOP*)newPMOP(OP_SUBST, 0); while (*s == 'g' || *s == 'i' || *s == 'e' || *s == 'o') { if (*s == 'e') { s++; es++; } if (*s == 'g') { s++; pm->op_pmflags |= PMf_GLOBAL; } if (*s == 'i') { s++; sawi = TRUE; pm->op_pmflags |= PMf_FOLD; } if (*s == 'o') { s++; pm->op_pmflags |= PMf_KEEP; } } if (es) { SV *repl; pm->op_pmflags |= PMf_EVAL; repl = newSVpv("",0); while (es-- > 0) sv_catpvn(repl, "eval ", 5); sv_catpvn(repl, "{ ", 2); sv_catsv(repl, lex_repl); sv_catpvn(repl, " };", 2); SvCOMPILED_on(repl); sv_free(lex_repl); lex_repl = repl; } lex_op = (OP*)pm; yylval.ival = OP_SUBST; return s; } void hoistmust(pm) register PMOP *pm; { if (!pm->op_pmshort && pm->op_pmregexp->regstart && (!pm->op_pmregexp->regmust || pm->op_pmregexp->reganch & ROPT_ANCH) ) { if (!(pm->op_pmregexp->reganch & ROPT_ANCH)) pm->op_pmflags |= PMf_SCANFIRST; else if (pm->op_pmflags & PMf_FOLD) return; pm->op_pmshort = sv_ref(pm->op_pmregexp->regstart); } else if (pm->op_pmregexp->regmust) {/* is there a better short-circuit? */ if (pm->op_pmshort && sv_eq(pm->op_pmshort,pm->op_pmregexp->regmust)) { if (pm->op_pmflags & PMf_SCANFIRST) { sv_free(pm->op_pmshort); pm->op_pmshort = Nullsv; } else { sv_free(pm->op_pmregexp->regmust); pm->op_pmregexp->regmust = Nullsv; return; } } if (!pm->op_pmshort || /* promote the better string */ ((pm->op_pmflags & PMf_SCANFIRST) && (SvCUR(pm->op_pmshort) < SvCUR(pm->op_pmregexp->regmust)) )){ sv_free(pm->op_pmshort); /* ok if null */ pm->op_pmshort = pm->op_pmregexp->regmust; pm->op_pmregexp->regmust = Nullsv; pm->op_pmflags |= PMf_SCANFIRST; } } } char * scan_trans(start) char *start; { register char *s = start; OP *op; short *tbl; I32 squash; I32 delete; I32 complement; yylval.ival = OP_NULL; s = scan_str(s); if (!s) { if (lex_stuff) sv_free(lex_stuff); lex_stuff = Nullsv; croak("Translation pattern not terminated"); } if (s[-1] == *start) s--; s = scan_str(s); if (!s) { if (lex_stuff) sv_free(lex_stuff); lex_stuff = Nullsv; if (lex_repl) sv_free(lex_repl); lex_repl = Nullsv; croak("Translation replacement not terminated"); } New(803,tbl,256,short); op = newPVOP(OP_TRANS, 0, (char*)tbl); complement = delete = squash = 0; while (*s == 'c' || *s == 'd' || *s == 's') { if (*s == 'c') complement = OPpTRANS_COMPLEMENT; else if (*s == 'd') delete = OPpTRANS_DELETE; else squash = OPpTRANS_SQUASH; s++; } op->op_private = delete|squash|complement; lex_op = op; yylval.ival = OP_TRANS; return s; } char * scan_heredoc(s) register char *s; { SV *herewas; I32 op_type = OP_SCALAR; I32 len; SV *tmpstr; char term; register char *d; s += 2; d = tokenbuf; if (!rsfp) *d++ = '\n'; if (*s && strchr("`'\"",*s)) { term = *s++; s = cpytill(d,s,bufend,term,&len); if (s < bufend) s++; d += len; } else { if (*s == '\\') s++, term = '\''; else term = '"'; while (isALNUM(*s)) *d++ = *s++; } /* assuming tokenbuf won't clobber */ *d++ = '\n'; *d = '\0'; len = d - tokenbuf; d = "\n"; if (rsfp || !(d=ninstr(s,bufend,d,d+1))) herewas = newSVpv(s,bufend-s); else s--, herewas = newSVpv(s,d-s); s += SvCUR(herewas); if (term == '\'') op_type = OP_CONST; if (term == '`') op_type = OP_BACKTICK; CLINE; multi_start = curcop->cop_line; multi_open = multi_close = '<'; tmpstr = NEWSV(87,80); term = *tokenbuf; if (!rsfp) { d = s; while (s < bufend && (*s != term || bcmp(s,tokenbuf,len) != 0) ) { if (*s++ == '\n') curcop->cop_line++; } if (s >= bufend) { curcop->cop_line = multi_start; croak("EOF in string"); } sv_setpvn(tmpstr,d+1,s-d); s += len - 1; sv_catpvn(herewas,s,bufend-s); sv_setsv(linestr,herewas); oldoldbufptr = oldbufptr = bufptr = s = SvPVX(linestr); bufend = SvPVX(linestr) + SvCUR(linestr); } else sv_setpvn(tmpstr,"",0); /* avoid "uninitialized" warning */ while (s >= bufend) { /* multiple line string? */ if (!rsfp || !(oldoldbufptr = oldbufptr = s = sv_gets(linestr, rsfp, 0))) { curcop->cop_line = multi_start; croak("EOF in string"); } curcop->cop_line++; if (perldb) { SV *sv = NEWSV(88,0); sv_upgrade(sv, SVt_PVMG); sv_setsv(sv,linestr); av_store(GvAV(curcop->cop_filegv), (I32)curcop->cop_line,sv); } bufend = SvPVX(linestr) + SvCUR(linestr); if (*s == term && bcmp(s,tokenbuf,len) == 0) { s = bufend - 1; *s = ' '; sv_catsv(linestr,herewas); bufend = SvPVX(linestr) + SvCUR(linestr); } else { s = bufend; sv_catsv(tmpstr,linestr); } } multi_end = curcop->cop_line; s++; if (SvCUR(tmpstr) + 5 < SvLEN(tmpstr)) { SvLEN_set(tmpstr, SvCUR(tmpstr) + 1); Renew(SvPVX(tmpstr), SvLEN(tmpstr), char); } sv_free(herewas); lex_stuff = tmpstr; yylval.ival = op_type; return s; } char * scan_inputsymbol(start) char *start; { register char *s = start; register char *d; I32 len; d = tokenbuf; s = cpytill(d, s+1, bufend, '>', &len); if (s < bufend) s++; else croak("Unterminated <> operator"); if (*d == '$') d++; while (*d && (isALNUM(*d) || *d == '\'')) d++; if (d - tokenbuf != len) { yylval.ival = OP_GLOB; set_csh(); s = scan_str(start); if (!s) croak("Glob not terminated"); return s; } else { d = tokenbuf; if (!len) (void)strcpy(d,"ARGV"); if (*d == '$') { GV *gv = gv_fetchpv(d+1,TRUE); lex_op = (OP*)newUNOP(OP_READLINE, 0, newUNOP(OP_RV2GV, 0, newUNOP(OP_RV2SV, 0, newGVOP(OP_GV, 0, gv)))); yylval.ival = OP_NULL; } else { IO *io; GV *gv = gv_fetchpv(d,TRUE); io = GvIOn(gv); if (strEQ(d,"ARGV")) { GvAVn(gv); io->flags |= IOf_ARGV|IOf_START; } lex_op = (OP*)newUNOP(OP_READLINE, 0, newGVOP(OP_GV, 0, gv)); yylval.ival = OP_NULL; } } return s; } char * scan_str(start) char *start; { SV *sv; char *tmps; register char *s = start; register char term = *s; register char *to; I32 brackets = 1; CLINE; multi_start = curcop->cop_line; multi_open = term; if (term && (tmps = strchr("([{< )]}> )]}>",term))) term = tmps[5]; multi_close = term; sv = NEWSV(87,80); sv_upgrade(sv, SVt_PVIV); SvIVX(sv) = term; SvPOK_only(sv); /* validate pointer */ s++; for (;;) { SvGROW(sv, SvCUR(sv) + (bufend - s) + 1); to = SvPVX(sv)+SvCUR(sv); if (multi_open == multi_close) { for (; s < bufend; s++,to++) { if (*s == '\n' && !rsfp) curcop->cop_line++; if (*s == '\\' && s+1 < bufend && term != '\\') *to++ = *s++; else if (*s == term) break; *to = *s; } } else { for (; s < bufend; s++,to++) { if (*s == '\n' && !rsfp) curcop->cop_line++; if (*s == '\\' && s+1 < bufend && term != '\\') *to++ = *s++; else if (*s == term && --brackets <= 0) break; else if (*s == multi_open) brackets++; *to = *s; } } *to = '\0'; SvCUR_set(sv, to - SvPVX(sv)); if (s < bufend) break; /* string ends on this line? */ if (!rsfp || !(oldoldbufptr = oldbufptr = s = sv_gets(linestr, rsfp, 0))) { curcop->cop_line = multi_start; return Nullch; } curcop->cop_line++; if (perldb) { SV *sv = NEWSV(88,0); sv_upgrade(sv, SVt_PVMG); sv_setsv(sv,linestr); av_store(GvAV(curcop->cop_filegv), (I32)curcop->cop_line, sv); } bufend = SvPVX(linestr) + SvCUR(linestr); } multi_end = curcop->cop_line; s++; if (SvCUR(sv) + 5 < SvLEN(sv)) { SvLEN_set(sv, SvCUR(sv) + 1); Renew(SvPVX(sv), SvLEN(sv), char); } if (lex_stuff) lex_repl = sv; else lex_stuff = sv; return s; } char * scan_num(start) char *start; { register char *s = start; register char *d; I32 tryi32; double value; SV *sv; I32 floatit; char *lastub = 0; switch (*s) { default: croak("panic: scan_num"); case '0': { U32 i; I32 shift; if (s[1] == 'x') { shift = 4; s += 2; } else if (s[1] == '.') goto decimal; else shift = 3; i = 0; for (;;) { switch (*s) { default: goto out; case '_': s++; break; case '8': case '9': if (shift != 4) yyerror("Illegal octal digit"); /* FALL THROUGH */ case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': i <<= shift; i += *s++ & 15; break; case 'a': case 'b': case 'c': case 'd': case 'e': case 'f': case 'A': case 'B': case 'C': case 'D': case 'E': case 'F': if (shift != 4) goto out; i <<= 4; i += (*s++ & 7) + 9; break; } } out: sv = NEWSV(92,0); tryi32 = i; if (tryi32 == i && tryi32 >= 0) sv_setiv(sv,tryi32); else sv_setnv(sv,(double)i); } break; case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': case '.': decimal: d = tokenbuf; floatit = FALSE; while (isDIGIT(*s) || *s == '_') { if (*s == '_') { if (dowarn && lastub && s - lastub != 3) warn("Misplaced _"); lastub = ++s; } else *d++ = *s++; } if (dowarn && lastub && s - lastub != 3) warn("Misplaced _"); if (*s == '.' && s[1] != '.') { floatit = TRUE; *d++ = *s++; while (isDIGIT(*s) || *s == '_') { if (*s == '_') s++; else *d++ = *s++; } } if (*s && strchr("eE",*s) && strchr("+-0123456789",s[1])) { floatit = TRUE; s++; *d++ = 'e'; /* At least some Mach atof()s don't grok 'E' */ if (*s == '+' || *s == '-') *d++ = *s++; while (isDIGIT(*s)) *d++ = *s++; } *d = '\0'; sv = NEWSV(92,0); value = atof(tokenbuf); tryi32 = I_32(value); if (!floatit && (double)tryi32 == value) sv_setiv(sv,tryi32); else sv_setnv(sv,value); break; } yylval.opval = newSVOP(OP_CONST, 0, sv); return s; } char * scan_formline(s) register char *s; { register char *eol; register char *t; SV *stuff = newSV(0); bool needargs = FALSE; while (!needargs) { if (*s == '.') { /*SUPPRESS 530*/ for (t = s+1; *t == ' ' || *t == '\t'; t++) ; if (*t == '\n') break; } if (in_eval && !rsfp) { eol = strchr(s,'\n'); if (!eol++) eol = bufend; } else eol = bufend = SvPVX(linestr) + SvCUR(linestr); if (*s != '#') { sv_catpvn(stuff, s, eol-s); while (s < eol) { if (*s == '@' || *s == '^') { needargs = TRUE; break; } s++; } } s = eol; if (rsfp) { s = sv_gets(linestr, rsfp, 0); oldoldbufptr = oldbufptr = bufptr = SvPVX(linestr); if (!s) { s = bufptr; yyerror("Format not terminated"); break; } } incline(s); } if (SvPOK(stuff)) { expect = XTERM; if (needargs) { nextval[nexttoke].ival = 0; force_next(','); } else in_format = 2; nextval[nexttoke].opval = (OP*)newSVOP(OP_CONST, 0, stuff); force_next(THING); nextval[nexttoke].ival = OP_FORMLINE; force_next(LSTOP); } else { sv_free(stuff); in_format = 0; bufptr = s; } return s; } static void set_csh() { #ifdef CSH if (!cshlen) cshlen = strlen(cshname); #endif } int yyerror(s) char *s; { char tmpbuf[258]; char tmp2buf[258]; char *tname = tmpbuf; if (bufptr > oldoldbufptr && bufptr - oldoldbufptr < 200 && oldoldbufptr != oldbufptr && oldbufptr != bufptr) { while (isSPACE(*oldoldbufptr)) oldoldbufptr++; cpy7bit(tmp2buf, oldoldbufptr, bufptr - oldoldbufptr); sprintf(tname,"near \"%s\"",tmp2buf); } else if (bufptr > oldbufptr && bufptr - oldbufptr < 200 && oldbufptr != bufptr) { while (isSPACE(*oldbufptr)) oldbufptr++; cpy7bit(tmp2buf, oldbufptr, bufptr - oldbufptr); sprintf(tname,"near \"%s\"",tmp2buf); } else if (yychar > 255) tname = "next token ???"; else if (!yychar || (yychar == ';' && !rsfp)) (void)strcpy(tname,"at EOF"); else if ((yychar & 127) == 127) { if (lex_state == LEX_NORMAL || (lex_state == LEX_KNOWNEXT && lex_defer == LEX_NORMAL)) (void)strcpy(tname,"at end of line"); else (void)strcpy(tname,"at end of string"); } else if (yychar < 32) (void)sprintf(tname,"next char ^%c",yychar+64); else (void)sprintf(tname,"next char %c",yychar); (void)sprintf(buf, "%s at %s line %d, %s\n", s,SvPVX(GvSV(curcop->cop_filegv)),curcop->cop_line,tname); if (curcop->cop_line == multi_end && multi_start < multi_end) sprintf(buf+strlen(buf), " (Might be a runaway multi-line %c%c string starting on line %d)\n", multi_open,multi_close,multi_start); if (in_eval) sv_catpv(GvSV(gv_fetchpv("@",TRUE)),buf); else fputs(buf,stderr); if (++error_count >= 10) croak("%s has too many errors.\n", SvPVX(GvSV(curcop->cop_filegv))); return 0; }