summaryrefslogtreecommitdiff
path: root/op.c
diff options
context:
space:
mode:
Diffstat (limited to 'op.c')
-rw-r--r--op.c565
1 files changed, 361 insertions, 204 deletions
diff --git a/op.c b/op.c
index 9c522b1502..743c7cad87 100644
--- a/op.c
+++ b/op.c
@@ -11,8 +11,6 @@
#include "EXTERN.h"
#include "perl.h"
-extern int yychar;
-
/* Lowest byte of opargs */
#define OA_MARK 1
#define OA_FOLDCONST 2
@@ -43,54 +41,6 @@ register I32 l;
*d = '\0';
}
-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,"next 2 tokens \"%s\"",tmp2buf);
- }
- else if (bufptr > oldbufptr && bufptr - oldbufptr < 200 &&
- oldbufptr != bufptr) {
- while (isSPACE(*oldbufptr))
- oldbufptr++;
- cpy7bit(tmp2buf, oldbufptr, bufptr - oldbufptr);
- sprintf(tname,"next token \"%s\"",tmp2buf);
- }
- else if (yychar > 255)
- tname = "next token ???";
- else if (!yychar || (yychar == ';' && !rsfp))
- (void)strcpy(tname,"at EOF");
- else if ((yychar & 127) == 127)
- (void)strcpy(tname,"at end of line");
- 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,SvPV(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)
- fatal("%s has too many errors.\n",
- SvPV(GvSV(curcop->cop_filegv)));
- return 0;
-}
-
OP *
no_fh_allowed(op)
OP *op;
@@ -130,12 +80,12 @@ char *name;
sv_upgrade(sv, SVt_PVNV);
sv_setpv(sv, name);
av_store(comppadname, off, sv);
- SvNV(sv) = (double)cop_seq;
- SvIV(sv) = 99999999;
+ SvNVX(sv) = (double)cop_seqmax;
+ SvIVX(sv) = 99999999;
if (*name == '@')
- av_store(comppad, off, newAV());
+ av_store(comppad, off, (SV*)newAV());
else if (*name == '%')
- av_store(comppad, off, newHV(COEFFSIZE));
+ av_store(comppad, off, (SV*)newHV());
return off;
}
@@ -152,13 +102,13 @@ char *name;
AV *curlist;
AV *curname;
CV *cv;
- I32 seq = cop_seq;
+ I32 seq = cop_seqmax;
for (off = comppadnamefill; off > 0; off--) {
if ((sv = svp[off]) &&
- seq <= SvIV(sv) &&
- seq > (I32)SvNV(sv) &&
- strEQ(SvPV(sv), name))
+ seq <= SvIVX(sv) &&
+ seq > (I32)SvNVX(sv) &&
+ strEQ(SvPVX(sv), name))
{
return (PADOFFSET)off;
}
@@ -190,9 +140,9 @@ char *name;
svp = AvARRAY(curname);
for (off = AvFILL(curname); off > 0; off--) {
if ((sv = svp[off]) &&
- seq <= SvIV(sv) &&
- seq > (I32)SvNV(sv) &&
- strEQ(SvPV(sv), name))
+ seq <= SvIVX(sv) &&
+ seq > (I32)SvNVX(sv) &&
+ strEQ(SvPVX(sv), name))
{
PADOFFSET newoff = pad_alloc(OP_PADSV, 'M');
AV *oldpad = (AV*)*av_fetch(curlist, CvDEPTH(cv), FALSE);
@@ -201,8 +151,8 @@ char *name;
sv_upgrade(sv, SVt_PVNV);
sv_setpv(sv, name);
av_store(comppadname, newoff, sv);
- SvNV(sv) = (double)curcop->cop_seq;
- SvIV(sv) = 99999999;
+ SvNVX(sv) = (double)curcop->cop_seq;
+ SvIVX(sv) = 99999999;
av_store(comppad, newoff, sv_ref(oldsv));
return newoff;
}
@@ -223,7 +173,7 @@ I32 fill;
SV *sv;
for (off = AvFILL(comppadname); off > fill; off--) {
if (sv = svp[off])
- SvIV(sv) = cop_seq;
+ SvIVX(sv) = cop_seqmax;
}
}
@@ -236,7 +186,7 @@ char tmptype;
I32 retval;
if (AvARRAY(comppad) != curpad)
- fatal("panic: pad_alloc");
+ croak("panic: pad_alloc");
if (tmptype == 'M') {
do {
sv = *av_fetch(comppad, AvFILL(comppad) + 1, TRUE);
@@ -260,7 +210,7 @@ pad_sv(po)
PADOFFSET po;
{
if (!po)
- fatal("panic: pad_sv po");
+ croak("panic: pad_sv po");
DEBUG_X(fprintf(stderr, "Pad sv %d\n", po));
return curpad[po]; /* eventually we'll turn this into a macro */
}
@@ -270,9 +220,9 @@ pad_free(po)
PADOFFSET po;
{
if (AvARRAY(comppad) != curpad)
- fatal("panic: pad_free curpad");
+ croak("panic: pad_free curpad");
if (!po)
- fatal("panic: pad_free po");
+ croak("panic: pad_free po");
DEBUG_X(fprintf(stderr, "Pad free %d\n", po));
if (curpad[po])
SvSTORAGE(curpad[po]) = 'F';
@@ -285,9 +235,9 @@ pad_swipe(po)
PADOFFSET po;
{
if (AvARRAY(comppad) != curpad)
- fatal("panic: pad_swipe curpad");
+ croak("panic: pad_swipe curpad");
if (!po)
- fatal("panic: pad_swipe po");
+ croak("panic: pad_swipe po");
DEBUG_X(fprintf(stderr, "Pad swipe %d\n", po));
curpad[po] = NEWSV(0,0);
SvSTORAGE(curpad[po]) = 'F';
@@ -301,7 +251,7 @@ pad_reset()
register I32 po;
if (AvARRAY(comppad) != curpad)
- fatal("panic: pad_reset curpad");
+ croak("panic: pad_reset curpad");
DEBUG_X(fprintf(stderr, "Pad reset\n"));
for (po = AvMAX(comppad); po > 0; po--) {
if (curpad[po] && SvSTORAGE(curpad[po]) == 'T')
@@ -330,8 +280,9 @@ OP *op;
pad_free(op->op_targ);
switch (op->op_type) {
+ case OP_GVSV:
case OP_GV:
-/*XXX sv_free(cGVOP->op_gv); */
+ sv_free((SV*)cGVOP->op_gv);
break;
case OP_CONST:
sv_free(cSVOP->op_sv);
@@ -343,7 +294,7 @@ OP *op;
/* Contextualizers */
-#define LINKLIST(o) ((o)->op_next ? (o)->op_next : linklist(o))
+#define LINKLIST(o) ((o)->op_next ? (o)->op_next : linklist((OP*)o))
OP *
linklist(op)
@@ -409,6 +360,7 @@ OP *op;
if (!(op->op_flags & OPf_KIDS))
return op;
break;
+ case OP_SCOPE:
case OP_LEAVE:
case OP_LEAVETRY:
case OP_LINESEQ:
@@ -444,7 +396,8 @@ OP *op;
switch (op->op_type) {
default:
- if (dowarn && (opargs[op->op_type] & OA_FOLDCONST))
+ if (dowarn && (opargs[op->op_type] & OA_FOLDCONST) &&
+ !(op->op_flags & OPf_STACKED))
warn("Useless use of %s", op_name[op->op_type]);
return op;
@@ -482,6 +435,7 @@ OP *op;
case OP_NULL:
if (!(op->op_flags & OPf_KIDS))
break;
+ case OP_SCOPE:
case OP_LEAVE:
case OP_LEAVETRY:
case OP_LINESEQ:
@@ -544,6 +498,7 @@ OP *op;
case OP_LIST:
listkids(op);
break;
+ case OP_SCOPE:
case OP_LEAVE:
case OP_LEAVETRY:
case OP_LINESEQ:
@@ -565,37 +520,42 @@ OP *op;
{
OP *kid;
- if (op &&
- (op->op_type == OP_LINESEQ ||
+ if (op) {
+ if (op->op_type == OP_LINESEQ ||
+ op->op_type == OP_SCOPE ||
op->op_type == OP_LEAVE ||
- op->op_type == OP_LEAVETRY) )
- {
- for (kid = cLISTOP->op_first; kid; kid = kid->op_sibling) {
- if (kid->op_sibling)
- scalarvoid(kid);
+ op->op_type == OP_LEAVETRY)
+ {
+ for (kid = cLISTOP->op_first; kid; kid = kid->op_sibling) {
+ if (kid->op_sibling)
+ scalarvoid(kid);
+ }
+ curcop = &compiling;
}
- curcop = &compiling;
+ op->op_flags &= ~OPf_PARENS;
+ if (needblockscope)
+ op->op_flags |= OPf_PARENS;
}
return op;
}
OP *
-refkids(op, type)
+modkids(op, type)
OP *op;
I32 type;
{
OP *kid;
if (op && op->op_flags & OPf_KIDS) {
for (kid = cLISTOP->op_first; kid; kid = kid->op_sibling)
- ref(kid, type);
+ mod(kid, type);
}
return op;
}
-static I32 refcount;
+static I32 modcount;
OP *
-ref(op, type)
+mod(op, type)
OP *op;
I32 type;
{
@@ -628,7 +588,7 @@ I32 type;
case OP_COND_EXPR:
for (kid = cUNOP->op_first->op_sibling; kid; kid = kid->op_sibling)
- ref(kid, type);
+ mod(kid, type);
break;
case OP_RV2AV:
@@ -641,20 +601,134 @@ I32 type;
case OP_HSLICE:
case OP_NEXTSTATE:
case OP_DBSTATE:
- refcount = 10000;
+ modcount = 10000;
break;
+ case OP_RV2SV:
+ if (type == OP_RV2AV || type == OP_RV2HV)
+ op->op_private = type;
+ /* FALL THROUGH */
case OP_PADSV:
case OP_PADAV:
case OP_PADHV:
case OP_UNDEF:
case OP_GV:
+ case OP_AV2ARYLEN:
+ case OP_SASSIGN:
+ case OP_REFGEN:
+ case OP_ANONLIST:
+ case OP_ANONHASH:
+ modcount++;
+ break;
+
+ case OP_PUSHMARK:
+ break;
+
+ case OP_SUBSTR:
+ case OP_VEC:
+ op->op_targ = pad_alloc(op->op_type,'M');
+ sv = PAD_SV(op->op_targ);
+ sv_upgrade(sv, SVt_PVLV);
+ sv_magic(sv, 0, op->op_type == OP_VEC ? 'v' : 'x', 0, 0);
+ curpad[op->op_targ] = sv;
+ /* FALL THROUGH */
+ case OP_NULL:
+ if (!(op->op_flags & OPf_KIDS))
+ croak("panic: mod");
+ mod(cBINOP->op_first, type ? type : op->op_type);
+ break;
+ case OP_AELEM:
+ case OP_HELEM:
+ mod(cBINOP->op_first, type ? type : op->op_type);
+ if (type == OP_RV2AV || type == OP_RV2HV)
+ op->op_private = type;
+ break;
+
+ case OP_SCOPE:
+ case OP_LEAVE:
+ case OP_ENTER:
+ if (type != OP_RV2HV && type != OP_RV2AV)
+ break;
+ if (!(op->op_flags & OPf_KIDS))
+ break;
+ /* FALL THROUGH */
+ case OP_LIST:
+ for (kid = cLISTOP->op_first; kid; kid = kid->op_sibling)
+ mod(kid, type);
+ break;
+ }
+ op->op_flags |= OPf_LVAL;
+ if (!type) {
+ op->op_flags &= ~OPf_SPECIAL;
+ op->op_flags |= OPf_INTRO;
+ }
+ else if (type == OP_AASSIGN || type == OP_SASSIGN)
+ op->op_flags |= OPf_SPECIAL;
+ return op;
+}
+
+OP *
+refkids(op, type)
+OP *op;
+I32 type;
+{
+ OP *kid;
+ if (op && op->op_flags & OPf_KIDS) {
+ for (kid = cLISTOP->op_first; kid; kid = kid->op_sibling)
+ ref(kid, type);
+ }
+ return op;
+}
+
+OP *
+ref(op, type)
+OP *op;
+I32 type;
+{
+ OP *kid;
+ SV *sv;
+
+ if (!op)
+ return op;
+
+ switch (op->op_type) {
+ default:
+ sprintf(tokenbuf, "Can't use %s as reference in %s",
+ op_name[op->op_type],
+ type ? op_name[type] : "local");
+ yyerror(tokenbuf);
+ return op;
+
+ case OP_COND_EXPR:
+ for (kid = cUNOP->op_first->op_sibling; kid; kid = kid->op_sibling)
+ ref(kid, type);
+ break;
+
+ case OP_RV2AV:
+ case OP_RV2HV:
+ case OP_RV2GV:
+ ref(cUNOP->op_first, op->op_type);
+ /* FALL THROUGH */
+ case OP_AASSIGN:
+ case OP_ASLICE:
+ case OP_HSLICE:
+ case OP_NEXTSTATE:
+ case OP_DBSTATE:
+ case OP_ENTERSUBR:
+ break;
case OP_RV2SV:
+ if (type == OP_RV2AV || type == OP_RV2HV)
+ op->op_private = type;
+ /* FALL THROUGH */
+ case OP_PADSV:
+ case OP_PADAV:
+ case OP_PADHV:
+ case OP_UNDEF:
+ case OP_GV:
case OP_AV2ARYLEN:
case OP_SASSIGN:
case OP_REFGEN:
case OP_ANONLIST:
case OP_ANONHASH:
- refcount++;
break;
case OP_PUSHMARK:
@@ -670,7 +744,7 @@ I32 type;
/* FALL THROUGH */
case OP_NULL:
if (!(op->op_flags & OPf_KIDS))
- fatal("panic: ref");
+ break;
ref(cBINOP->op_first, type ? type : op->op_type);
break;
case OP_AELEM:
@@ -680,6 +754,7 @@ I32 type;
op->op_private = type;
break;
+ case OP_SCOPE:
case OP_LEAVE:
case OP_ENTER:
if (type != OP_RV2HV && type != OP_RV2AV)
@@ -753,7 +828,7 @@ OP *right;
right->op_type == OP_TRANS) {
right->op_flags |= OPf_STACKED;
if (right->op_type != OP_MATCH)
- left = ref(left, right->op_type);
+ left = mod(left, right->op_type);
if (right->op_type == OP_TRANS)
op = newBINOP(OP_NULL, 0, scalar(left), right);
else
@@ -782,9 +857,25 @@ scope(o)
OP *o;
{
if (o) {
- o = prepend_elem(OP_LINESEQ, newOP(OP_ENTER, 0), o);
- o->op_type = OP_LEAVE;
- o->op_ppaddr = ppaddr[OP_LEAVE];
+ if (o->op_flags & OPf_PARENS) {
+ o = prepend_elem(OP_LINESEQ, newOP(OP_ENTER, 0), o);
+ o->op_type = OP_LEAVE;
+ o->op_ppaddr = ppaddr[OP_LEAVE];
+ }
+ else {
+ if (o->op_type == OP_LINESEQ) {
+ OP *kid;
+ o->op_type = OP_SCOPE;
+ o->op_ppaddr = ppaddr[OP_SCOPE];
+ kid = ((LISTOP*)o)->op_first;
+ if (kid->op_type == OP_NEXTSTATE) {
+ kid->op_type = OP_NULL;
+ kid->op_ppaddr = ppaddr[OP_NULL];
+ }
+ }
+ else
+ o = newUNOP(OP_SCOPE, 0, o);
+ }
}
return o;
}
@@ -798,7 +889,7 @@ OP **startp;
*startp = 0;
return o;
}
- o = scalarseq(scope(o));
+ o = scope(scalarseq(o));
*startp = LINKLIST(o);
o->op_next = 0;
peep(*startp);
@@ -818,7 +909,7 @@ I32 lex;
if (lex)
return my(o);
else
- return ref(o, OP_NULL); /* a bit kludgey */
+ return mod(o, OP_NULL); /* a bit kludgey */
}
OP *
@@ -1172,10 +1263,10 @@ OP *repl;
PMOP *pm = (PMOP*)op;
SV *tstr = ((SVOP*)expr)->op_sv;
SV *rstr = ((SVOP*)repl)->op_sv;
- register char *t = SvPVn(tstr);
- register char *r = SvPVn(rstr);
- I32 tlen = SvCUR(tstr);
- I32 rlen = SvCUR(rstr);
+ STRLEN tlen;
+ STRLEN rlen;
+ register char *t = SvPV(tstr, tlen);
+ register char *r = SvPV(rstr, rlen);
register I32 i;
register I32 j;
I32 squash;
@@ -1269,21 +1360,25 @@ OP *repl;
pm = (PMOP*)op;
if (expr->op_type == OP_CONST) {
+ STRLEN plen;
SV *pat = ((SVOP*)expr)->op_sv;
- char *p = SvPVn(pat);
+ char *p = SvPV(pat, plen);
if ((op->op_flags & OPf_SPECIAL) && strEQ(p, " ")) {
sv_setpvn(pat, "\\s+", 3);
- p = SvPVn(pat);
+ p = SvPV(pat, plen);
pm->op_pmflags |= PMf_SKIPWHITE;
}
- scan_prefix(pm, p, SvCUR(pat));
+ scan_prefix(pm, p, plen);
if (pm->op_pmshort && (pm->op_pmflags & PMf_SCANFIRST))
fbm_compile(pm->op_pmshort, pm->op_pmflags & PMf_FOLD);
- pm->op_pmregexp = regcomp(p, p + SvCUR(pat), pm->op_pmflags & PMf_FOLD);
+ pm->op_pmregexp = regcomp(p, p + plen, pm->op_pmflags & PMf_FOLD);
hoistmust(pm);
op_free(expr);
}
else {
+ if (pm->op_pmflags & PMf_KEEP)
+ expr = newUNOP(OP_REGCMAYBE,0,expr);
+
Newz(1101, rcop, 1, LOGOP);
rcop->op_type = OP_REGCOMP;
rcop->op_ppaddr = ppaddr[OP_REGCOMP];
@@ -1293,10 +1388,17 @@ OP *repl;
rcop->op_other = op;
/* establish postfix order */
- rcop->op_next = LINKLIST(expr);
- expr->op_next = (OP*)rcop;
+ if (pm->op_pmflags & PMf_KEEP) {
+ LINKLIST(expr);
+ rcop->op_next = expr;
+ ((UNOP*)expr)->op_first->op_next = (OP*)rcop;
+ }
+ else {
+ rcop->op_next = LINKLIST(expr);
+ expr->op_next = (OP*)rcop;
+ }
- prepend_elem(op->op_type, scalar(rcop), op);
+ prepend_elem(op->op_type, scalar((OP*)rcop), op);
}
if (repl) {
@@ -1345,7 +1447,7 @@ OP *repl;
rcop->op_next = LINKLIST(repl);
repl->op_next = (OP*)rcop;
- pm->op_pmreplroot = scalar(rcop);
+ pm->op_pmreplroot = scalar((OP*)rcop);
pm->op_pmreplstart = LINKLIST(rcop);
rcop->op_next = 0;
}
@@ -1369,7 +1471,7 @@ SV *sv;
svop->op_next = (OP*)svop;
svop->op_flags = flags;
if (opargs[type] & OA_RETSCALAR)
- scalar(svop);
+ scalar((OP*)svop);
if (opargs[type] & OA_TARGET)
svop->op_targ = pad_alloc(type,'T');
return (*check[type])((OP*)svop);
@@ -1389,7 +1491,7 @@ GV *gv;
gvop->op_next = (OP*)gvop;
gvop->op_flags = flags;
if (opargs[type] & OA_RETSCALAR)
- scalar(gvop);
+ scalar((OP*)gvop);
if (opargs[type] & OA_TARGET)
gvop->op_targ = pad_alloc(type,'T');
return (*check[type])((OP*)gvop);
@@ -1409,7 +1511,7 @@ char *pv;
pvop->op_next = (OP*)pvop;
pvop->op_flags = flags;
if (opargs[type] & OA_RETSCALAR)
- scalar(pvop);
+ scalar((OP*)pvop);
if (opargs[type] & OA_TARGET)
pvop->op_targ = pad_alloc(type,'T');
return (*check[type])((OP*)pvop);
@@ -1431,7 +1533,7 @@ OP *cont;
cvop->op_next = (OP*)cvop;
cvop->op_flags = flags;
if (opargs[type] & OA_RETSCALAR)
- scalar(cvop);
+ scalar((OP*)cvop);
if (opargs[type] & OA_TARGET)
cvop->op_targ = pad_alloc(type,'T');
return (*check[type])((OP*)cvop);
@@ -1441,25 +1543,17 @@ void
package(op)
OP *op;
{
- char tmpbuf[256];
- GV *tmpgv;
SV *sv;
- char *name;
save_hptr(&curstash);
save_item(curstname);
if (op) {
+ STRLEN len;
+ char *name;
sv = cSVOP->op_sv;
- name = SvPVn(sv);
- sv_setpv(curstname,name);
- sprintf(tmpbuf,"'_%s",name);
- tmpgv = gv_fetchpv(tmpbuf,TRUE);
- if (!GvHV(tmpgv))
- GvHV(tmpgv) = newHV(0);
- curstash = GvHV(tmpgv);
- if (!HvNAME(curstash))
- HvNAME(curstash) = savestr(name);
- HvCOEFFSIZE(curstash) = 0;
+ curstash = fetch_stash(sv,TRUE);
+ name = SvPV(sv, len);
+ sv_setpvn(curstname, name, len);
op_free(op);
}
else {
@@ -1470,6 +1564,27 @@ OP *op;
expect = XBLOCK;
}
+HV*
+fetch_stash(sv,create)
+SV *sv;
+I32 create;
+{
+ char tmpbuf[256];
+ HV *stash;
+ GV *tmpgv;
+ char *name = SvPV(sv, na);
+ sprintf(tmpbuf,"%s::",name);
+ tmpgv = gv_fetchpv(tmpbuf,create);
+ if (!tmpgv)
+ return 0;
+ if (!GvHV(tmpgv))
+ GvHV(tmpgv) = newHV();
+ stash = GvHV(tmpgv);
+ if (!HvNAME(stash))
+ HvNAME(stash) = savestr(name);
+ return stash;
+}
+
OP *
newSLICEOP(flags, subscript, listval)
I32 flags;
@@ -1525,8 +1640,8 @@ OP *right;
OP *op;
if (list_assignment(left)) {
- refcount = 0;
- left = ref(left, OP_AASSIGN);
+ modcount = 0;
+ left = mod(left, OP_AASSIGN);
if (right && right->op_type == OP_SPLIT) {
if ((op = ((LISTOP*)right)->op_first) && op->op_type == OP_PUSHRE) {
PMOP *pm = (PMOP*)op;
@@ -1540,10 +1655,10 @@ OP *right;
}
}
else {
- if (refcount < 10000) {
+ if (modcount < 10000) {
SV *sv = ((SVOP*)((LISTOP*)right)->op_last)->op_sv;
- if (SvIV(sv) == 0)
- sv_setiv(sv, refcount+1);
+ if (SvIVX(sv) == 0)
+ sv_setiv(sv, modcount+1);
}
}
}
@@ -1589,11 +1704,11 @@ OP *right;
right = newOP(OP_UNDEF, 0);
if (right->op_type == OP_READLINE) {
right->op_flags |= OPf_STACKED;
- return newBINOP(OP_NULL, flags, ref(scalar(left), OP_SASSIGN), scalar(right));
+ return newBINOP(OP_NULL, flags, mod(scalar(left), OP_SASSIGN), scalar(right));
}
else
op = newBINOP(OP_SASSIGN, flags,
- scalar(right), ref(scalar(left), OP_SASSIGN) );
+ scalar(right), mod(scalar(left), OP_SASSIGN) );
return op;
}
@@ -1614,8 +1729,11 @@ OP *op;
cop->op_private = 0;
cop->op_next = (OP*)cop;
- cop->cop_label = label;
- cop->cop_seq = cop_seq++;
+ if (label) {
+ cop->cop_label = label;
+ needblockscope = TRUE;
+ }
+ cop->cop_seq = cop_seqmax++;
if (copline == NOLINE)
cop->cop_line = curcop->cop_line;
@@ -1629,7 +1747,7 @@ OP *op;
if (perldb) {
SV **svp = av_fetch(GvAV(curcop->cop_filegv),(I32)cop->cop_line, FALSE);
if (svp && *svp != &sv_undef && !SvIOK(*svp)) {
- SvIV(*svp) = 1;
+ SvIVX(*svp) = 1;
SvIOK_on(*svp);
SvSTASH(*svp) = (HV*)cop;
}
@@ -1718,6 +1836,8 @@ OP* false;
if (!false)
return newLOGOP(OP_AND, 0, first, true);
+ if (!true)
+ return newLOGOP(OP_OR, 0, first, false);
scalar(first);
if (first->op_type == OP_CONST) {
@@ -1814,21 +1934,28 @@ I32 debuggable;
OP *expr;
OP *block;
{
- OP* listop = append_elem(OP_LINESEQ, block, newOP(OP_UNSTACK, 0));
+ OP* listop;
OP* op;
+ int once = block && block->op_flags & OPf_SPECIAL &&
+ (block->op_type == OP_ENTERSUBR || block->op_type == OP_NULL);
- if (expr && (expr->op_type == OP_READLINE || expr->op_type == OP_GLOB))
- expr = newASSIGNOP(0, newSVREF(newGVOP(OP_GV, 0, defgv)), expr);
+ if (expr) {
+ if (once && expr->op_type == OP_CONST && !SvTRUE(((SVOP*)expr)->op_sv))
+ return block; /* do {} while 0 does once */
+ else if (expr->op_type == OP_READLINE || expr->op_type == OP_GLOB)
+ expr = newASSIGNOP(0, newSVREF(newGVOP(OP_GV, 0, defgv)), expr);
+ }
+ listop = append_elem(OP_LINESEQ, block, newOP(OP_UNSTACK, 0));
op = newLOGOP(OP_AND, 0, expr, listop);
+
((LISTOP*)listop)->op_last->op_next = LINKLIST(op);
- if (block->op_flags & OPf_SPECIAL && /* skip conditional on do {} ? */
- (block->op_type == OP_ENTERSUBR || block->op_type == OP_NULL))
+ if (once && op != listop)
op->op_next = ((LOGOP*)cUNOP->op_first)->op_other;
op->op_flags |= flags;
- return op;
+ return scope(op);
}
OP *
@@ -1857,11 +1984,16 @@ OP *cont;
if (expr)
cont = append_elem(OP_LINESEQ, cont, newOP(OP_UNSTACK, 0));
- listop = append_list(OP_LINESEQ, block, cont);
+ listop = append_list(OP_LINESEQ, (LISTOP*)block, (LISTOP*)cont);
redo = LINKLIST(listop);
if (expr) {
op = newLOGOP(OP_AND, 0, expr, scalar(listop));
+ if (op == expr) { /* oops, it's a while (0) */
+ op_free(expr);
+ op_free((OP*)loop);
+ return Nullop; /* (listop already freed by newLOGOP) */
+ }
((LISTOP*)listop)->op_last->op_next = condop =
(op == listop ? redo : LINKLIST(op));
if (!next)
@@ -1878,7 +2010,7 @@ OP *cont;
loop->op_next = (OP*)loop;
}
- op = newBINOP(OP_LEAVELOOP, 0, loop, op);
+ op = newBINOP(OP_LEAVELOOP, 0, (OP*)loop, op);
loop->op_redoop = redo;
loop->op_lastop = op;
@@ -1914,7 +2046,7 @@ OP*cont;
op_free(op);
}
else
- fatal("Can't use %s for loop variable", op_name[sv->op_type]);
+ croak("Can't use %s for loop variable", op_name[sv->op_type]);
}
else {
sv = newGVOP(OP_GV, 0, defgv);
@@ -1928,7 +2060,7 @@ OP*cont;
}
void
-cv_free(cv)
+cv_clear(cv)
CV *cv;
{
if (!CvUSERSUB(cv) && CvROOT(cv)) {
@@ -1941,12 +2073,11 @@ CV *cv;
while (i > 0) {
SV** svp = av_fetch(CvPADLIST(cv), i--, FALSE);
if (svp)
- av_free(*svp);
+ av_free((AV*)*svp);
}
- av_free(CvPADLIST(cv));
+ av_free((AV*)CvPADLIST(cv));
}
}
- Safefree(cv);
}
void
@@ -1956,11 +2087,12 @@ OP *op;
OP *block;
{
register CV *cv;
- char *name = SvPVnx(cSVOP->op_sv);
- GV *gv = gv_fetchpv(name,TRUE);
+ char *name = SvPVx(cSVOP->op_sv, na);
+ GV *gv = gv_fetchpv(name,2);
AV* av;
- if (cv = GvCV(gv)) {
+ sub_generation++;
+ if ((cv = GvCV(gv)) && !GvCVGEN(gv)) {
if (CvDEPTH(cv))
CvDELETED(cv) = TRUE; /* probably an autoloader */
else {
@@ -1971,12 +2103,14 @@ OP *block;
warn("Subroutine %s redefined",name);
curcop->cop_line = oldline;
}
- cv_free(cv);
+ sv_free((SV*)cv);
}
}
Newz(101,cv,1,CV);
sv_upgrade(cv, SVt_PVCV);
+ SvREFCNT(cv) = 1;
GvCV(gv) = cv;
+ GvCVGEN(gv) = 0;
CvFILEGV(cv) = curcop->cop_filegv;
av = newAV();
@@ -2015,7 +2149,7 @@ OP *block;
rschar = nrschar;
rspara = (nrslen == 2);
calllist(beginav);
- cv_free(cv);
+ sv_free((SV*)cv);
rs = "\n";
rslen = 1;
rschar = '\n';
@@ -2035,13 +2169,13 @@ OP *block;
SV *sv;
SV *tmpstr = sv_mortalcopy(&sv_undef);
- sprintf(buf,"%s:%ld",SvPV(GvSV(curcop->cop_filegv)), subline);
+ sprintf(buf,"%s:%ld",SvPVX(GvSV(curcop->cop_filegv)), subline);
sv = newSVpv(buf,0);
sv_catpv(sv,"-");
sprintf(buf,"%ld",(long)curcop->cop_line);
sv_catpv(sv,buf);
gv_efullname(tmpstr,gv);
- hv_store(GvHV(DBsub), SvPV(tmpstr), SvCUR(tmpstr), sv, 0);
+ hv_store(GvHV(DBsub), SvPVX(tmpstr), SvCUR(tmpstr), sv, 0);
}
op_free(op);
copline = NOLINE;
@@ -2049,18 +2183,17 @@ OP *block;
}
void
-newUSUB(name, ix, subaddr, filename)
+newXSUB(name, ix, subaddr, filename)
char *name;
I32 ix;
I32 (*subaddr)();
char *filename;
{
register CV *cv;
- GV *gv = gv_fetchpv(name,allgvs);
+ GV *gv = gv_fetchpv(name,2);
- if (!gv) /* unused function */
- return;
- if (cv = GvCV(gv)) {
+ sub_generation++;
+ if ((cv = GvCV(gv)) && !GvCVGEN(gv)) {
if (dowarn)
warn("Subroutine %s redefined",name);
if (!CvUSERSUB(cv) && CvROOT(cv)) {
@@ -2071,7 +2204,9 @@ char *filename;
}
Newz(101,cv,1,CV);
sv_upgrade(cv, SVt_PVCV);
+ SvREFCNT(cv) = 1;
GvCV(gv) = cv;
+ GvCVGEN(gv) = 0;
CvFILEGV(cv) = gv_fetchfile(filename);
CvUSERSUB(cv) = subaddr;
CvUSERINDEX(cv) = ix;
@@ -2101,7 +2236,7 @@ OP *block;
AV* av;
if (op)
- name = SvPVnx(cSVOP->op_sv);
+ name = SvPVx(cSVOP->op_sv, na);
else
name = "STDOUT";
gv = gv_fetchpv(name,TRUE);
@@ -2113,10 +2248,11 @@ OP *block;
warn("Format %s redefined",name);
curcop->cop_line = oldline;
}
- cv_free(cv);
+ sv_free((SV*)cv);
}
Newz(101,cv,1,CV);
sv_upgrade(cv, SVt_PVFM);
+ SvREFCNT(cv) = 1;
GvFORM(gv) = cv;
CvFILEGV(cv) = curcop->cop_filegv;
@@ -2232,7 +2368,7 @@ OP *
oopsCV(o)
OP *o;
{
- fatal("NOT IMPL LINE %d",__LINE__);
+ croak("NOT IMPL LINE %d",__LINE__);
/* STUB */
return o;
}
@@ -2278,7 +2414,7 @@ OP *op;
{
if (op->op_flags & OPf_KIDS) {
OP* newop;
- op = refkids(ck_fun(op), op->op_type);
+ op = modkids(ck_fun(op), op->op_type);
if (op->op_private != 1)
return op;
newop = cUNOP->op_first->op_sibling;
@@ -2312,6 +2448,7 @@ OP *
ck_eval(op)
OP *op;
{
+ needblockscope = TRUE;
if (op->op_flags & OPf_KIDS) {
SVOP *kid = (SVOP*)cUNOP->op_first;
@@ -2335,7 +2472,7 @@ OP *op;
/* establish postfix order */
enter->op_next = (OP*)enter;
- op = prepend_elem(OP_LINESEQ, enter, kid);
+ op = prepend_elem(OP_LINESEQ, (OP*)enter, (OP*)kid);
op->op_type = OP_LEAVETRY;
op->op_ppaddr = ppaddr[OP_LEAVETRY];
enter->op_other = op;
@@ -2354,14 +2491,16 @@ ck_exec(op)
OP *op;
{
OP *kid;
- op = ck_fun(op);
if (op->op_flags & OPf_STACKED) {
+ op = ck_fun(op);
kid = cUNOP->op_first->op_sibling;
if (kid->op_type == OP_RV2GV) {
kid->op_type = OP_NULL;
kid->op_ppaddr = ppaddr[OP_NULL];
}
}
+ else
+ op = listkids(op);
return op;
}
@@ -2382,8 +2521,8 @@ register OP *op;
SVOP *kid = (SVOP*)cUNOP->op_first;
if (kid->op_type == OP_CONST) {
kid->op_type = OP_GV;
- kid->op_sv = (SV*)gv_fetchpv(SvPVnx(kid->op_sv),
- 1+(op->op_type==OP_RV2CV));
+ kid->op_sv = sv_ref((SV*)gv_fetchpv(SvPVx(kid->op_sv, na),
+ 1+(op->op_type==OP_RV2CV)));
}
return op;
}
@@ -2409,7 +2548,7 @@ OP *op;
if (kid->op_type == OP_CONST && (kid->op_private & OPpCONST_BARE)) {
OP *newop = newGVOP(type, OPf_SPECIAL,
- gv_fetchpv(SvPVnx(kid->op_sv), TRUE));
+ gv_fetchpv(SvPVx(kid->op_sv, na), TRUE));
op_free(op);
return newop;
}
@@ -2467,26 +2606,34 @@ OP *op;
case OA_AVREF:
if (kid->op_type == OP_CONST &&
(kid->op_private & OPpCONST_BARE)) {
+ char *name = SvPVx(((SVOP*)kid)->op_sv, na);
OP *newop = newAVREF(newGVOP(OP_GV, 0,
- gv_fetchpv(SvPVnx(((SVOP*)kid)->op_sv), TRUE) ));
+ gv_fetchpv(name, TRUE) ));
+ if (dowarn)
+ warn("Array @%s missing the @ in argument %d of %s()",
+ name, numargs, op_name[op->op_type]);
op_free(kid);
kid = newop;
kid->op_sibling = sibl;
*tokid = kid;
}
- ref(kid, op->op_type);
+ mod(kid, op->op_type);
break;
case OA_HVREF:
if (kid->op_type == OP_CONST &&
(kid->op_private & OPpCONST_BARE)) {
+ char *name = SvPVx(((SVOP*)kid)->op_sv, na);
OP *newop = newHVREF(newGVOP(OP_GV, 0,
- gv_fetchpv(SvPVnx(((SVOP*)kid)->op_sv), TRUE) ));
+ gv_fetchpv(name, TRUE) ));
+ if (dowarn)
+ warn("Hash %%%s missing the %% in argument %d of %s()",
+ name, numargs, op_name[op->op_type]);
op_free(kid);
kid = newop;
kid->op_sibling = sibl;
*tokid = kid;
}
- ref(kid, op->op_type);
+ mod(kid, op->op_type);
break;
case OA_CVREF:
{
@@ -2504,7 +2651,7 @@ OP *op;
if (kid->op_type == OP_CONST &&
(kid->op_private & OPpCONST_BARE)) {
OP *newop = newGVOP(OP_GV, 0,
- gv_fetchpv(SvPVnx(((SVOP*)kid)->op_sv), TRUE) );
+ gv_fetchpv(SvPVx(((SVOP*)kid)->op_sv, na), TRUE) );
op_free(kid);
kid = newop;
}
@@ -2518,7 +2665,7 @@ OP *op;
scalar(kid);
break;
case OA_SCALARREF:
- ref(scalar(kid), op->op_type);
+ mod(scalar(kid), op->op_type);
break;
}
oa >>= 4;
@@ -2566,7 +2713,7 @@ OP *op;
return op;
kid = cLISTOP->op_first->op_sibling;
if (kid->op_type != OP_NULL)
- fatal("panic: ck_grep");
+ croak("panic: ck_grep");
kid = kUNOP->op_first;
Newz(1101, gwop, 1, LOGOP);
@@ -2606,7 +2753,7 @@ OP *
ck_lfun(op)
OP *op;
{
- return refkids(ck_fun(op), op->op_type);
+ return modkids(ck_fun(op), op->op_type);
}
OP *
@@ -2673,7 +2820,7 @@ OP *
ck_retarget(op)
OP *op;
{
- fatal("NOT IMPL LINE %d",__LINE__);
+ croak("NOT IMPL LINE %d",__LINE__);
/* STUB */
return op;
}
@@ -2707,7 +2854,7 @@ OP *op;
scalar(newGVOP(OP_GV, 0,
gv_fetchpv((subline ? "_" : "ARGV"), TRUE) )))));
}
- return scalar(refkids(ck_fun(op), type));
+ return scalar(modkids(ck_fun(op), type));
}
OP *
@@ -2716,27 +2863,31 @@ OP *op;
{
if (op->op_flags & OPf_STACKED) {
OP *kid = cLISTOP->op_first->op_sibling; /* get past pushmark */
- kid = kUNOP->op_first; /* get past sv2gv */
- if (kid->op_type == OP_LEAVE) {
- OP *k;
+ OP *k;
+ kid = kUNOP->op_first; /* get past rv2gv */
+ if (kid->op_type == OP_SCOPE || kid->op_type == OP_LEAVE) {
linklist(kid);
- kid->op_type = OP_NULL; /* wipe out leave */
- kid->op_ppaddr = ppaddr[OP_NULL];
- kid->op_next = kid;
-
- for (k = kLISTOP->op_first->op_next; k; k = k->op_next) {
- if (k->op_next == kid)
- k->op_next = 0;
+ if (kid->op_type == OP_SCOPE) {
+ k = kid->op_next;
+ kid->op_next = 0;
+ peep(k);
}
- kid->op_type = OP_NULL; /* wipe out enter */
- kid->op_ppaddr = ppaddr[OP_NULL];
-
- kid = cLISTOP->op_first->op_sibling;
- kid->op_type = OP_NULL; /* wipe out sv2gv */
+ else if (kid->op_type == OP_LEAVE) {
+ kid->op_type = OP_NULL; /* wipe out leave */
+ kid->op_ppaddr = ppaddr[OP_NULL];
+ kid->op_next = kid;
+
+ for (k = kLISTOP->op_first->op_next; k; k = k->op_next) {
+ if (k->op_next == kid)
+ k->op_next = 0;
+ }
+ peep(kLISTOP->op_first);
+ }
+ kid = cLISTOP->op_first->op_sibling; /* get past pushmark */
+ kid->op_type = OP_NULL; /* wipe out rv2gv */
kid->op_ppaddr = ppaddr[OP_NULL];
kid->op_next = kid;
-
op->op_flags |= OPf_SPECIAL;
}
}
@@ -2762,10 +2913,11 @@ OP *op;
kid = cLISTOP->op_first;
if (kid->op_type == OP_PUSHMARK)
- fatal("panic: ck_split");
+ croak("panic: ck_split");
if (kid->op_type != OP_MATCH) {
OP *sibl = kid->op_sibling;
+ kid->op_sibling = 0;
kid = pmruntime( newPMOP(OP_MATCH, OPf_SPECIAL), kid, Nullop);
if (cLISTOP->op_first == cLISTOP->op_last)
cLISTOP->op_last = kid;
@@ -2825,6 +2977,8 @@ OP *op;
return ck_fun(op);
}
+/* A peephole optimizer. We visit the ops in the order they're to execute. */
+
void
peep(op)
register OP* op;
@@ -2839,15 +2993,18 @@ register OP* op;
case OP_NULL:
case OP_SCALAR:
case OP_LINESEQ:
+ case OP_SCOPE:
if (oldop) {
oldop->op_next = op->op_next;
continue;
}
- op->op_seq = ++op_seq;
+ op->op_seq = ++op_seqmax;
break;
case OP_GV:
- if (op->op_next->op_type == OP_RV2SV) {
+ if (op->op_next->op_type == OP_RV2SV &&
+ op->op_next->op_private < OP_RV2GV)
+ {
op->op_next->op_type = OP_NULL;
op->op_next->op_ppaddr = ppaddr[OP_NULL];
op->op_flags |= op->op_next->op_flags & OPf_INTRO;
@@ -2855,24 +3012,24 @@ register OP* op;
op->op_type = OP_GVSV;
op->op_ppaddr = ppaddr[OP_GVSV];
}
- op->op_seq = ++op_seq;
+ op->op_seq = ++op_seqmax;
break;
case OP_GREPWHILE:
case OP_AND:
case OP_OR:
- op->op_seq = ++op_seq;
+ op->op_seq = ++op_seqmax;
peep(cLOGOP->op_other);
break;
case OP_COND_EXPR:
- op->op_seq = ++op_seq;
+ op->op_seq = ++op_seqmax;
peep(cCONDOP->op_true);
peep(cCONDOP->op_false);
break;
case OP_ENTERLOOP:
- op->op_seq = ++op_seq;
+ op->op_seq = ++op_seqmax;
peep(cLOOP->op_redoop);
peep(cLOOP->op_nextop);
peep(cLOOP->op_lastop);
@@ -2880,12 +3037,12 @@ register OP* op;
case OP_MATCH:
case OP_SUBST:
- op->op_seq = ++op_seq;
+ op->op_seq = ++op_seqmax;
peep(cPMOP->op_pmreplroot);
break;
default:
- op->op_seq = ++op_seq;
+ op->op_seq = ++op_seqmax;
break;
}
oldop = op;