Freudian slip error in perlsub.pod
[p5sagit/p5-mst-13.2.git] / op.c
diff --git a/op.c b/op.c
index f3e39fb..6a1fa5b 100644 (file)
--- a/op.c
+++ b/op.c
@@ -39,6 +39,7 @@ static I32 list_assignment _((OP *op));
 static OP *bad_type _((I32 n, char *t, char *name, OP *kid));
 static OP *modkids _((OP *op, I32 type));
 static OP *no_fh_allowed _((OP *op));
+static bool scalar_mod_type _((OP *op, I32 type));
 static OP *scalarboolean _((OP *op));
 static OP *too_few_arguments _((OP *op, char* name));
 static OP *too_many_arguments _((OP *op, char* name));
@@ -219,7 +220,11 @@ pad_findlex(char *name, PADOFFSET newoff, U32 seq, CV* startcv, I32 cx_ix)
                    if (CvANON(compcv) || SvTYPE(compcv) == SVt_PVFM) {
                        /* "It's closures all the way down." */
                        CvCLONE_on(compcv);
-                       if (cv != startcv) {
+                       if (cv == startcv) {
+                           if (CvANON(compcv))
+                               oldsv = Nullsv; /* no need to keep ref */
+                       }
+                       else {
                            CV *bcv;
                            for (bcv = startcv;
                                 bcv && bcv != cv && !CvCLONE(bcv);
@@ -293,6 +298,7 @@ pad_findmy(name)
 char *name;
 {
     I32 off;
+    I32 pendoff = 0;
     SV *sv;
     SV **svp = AvARRAY(comppad_name);
     U32 seq = cop_seqmax;
@@ -301,18 +307,25 @@ char *name;
     for (off = AvFILL(comppad_name); off > 0; off--) {
        if ((sv = svp[off]) &&
            sv != &sv_undef &&
-           seq <= SvIVX(sv) &&
-           seq > I_32(SvNVX(sv)) &&
+           (!SvIVX(sv) ||
+            (seq <= SvIVX(sv) &&
+             seq > I_32(SvNVX(sv)))) &&
            strEQ(SvPVX(sv), name))
        {
-           return (PADOFFSET)off;
+           if (SvIVX(sv))
+               return (PADOFFSET)off;
+           pendoff = off;      /* this pending def. will override import */
        }
     }
 
     /* See if it's in a nested scope */
     off = pad_findlex(name, 0, seq, CvOUTSIDE(compcv), cxstack_ix);
-    if (off)
+    if (off) {
+       /* If there is a pending local definition, this new alias must die */
+       if (pendoff)
+           SvIVX(AvARRAY(comppad_name)[off]) = seq;
        return off;
+    }
 
     return 0;
 }
@@ -389,7 +402,7 @@ pad_sv(PADOFFSET po)
 {
     if (!po)
        croak("panic: pad_sv po");
-    DEBUG_X(PerlIO_printf(Perl_debug_log, "Pad sv %d\n", po));
+    DEBUG_X(PerlIO_printf(Perl_debug_log, "Pad sv %lu\n", (unsigned long)po));
     return curpad[po];         /* eventually we'll turn this into a macro */
 }
 
@@ -407,7 +420,7 @@ pad_free(PADOFFSET po)
        croak("panic: pad_free curpad");
     if (!po)
        croak("panic: pad_free po");
-    DEBUG_X(PerlIO_printf(Perl_debug_log, "Pad free %d\n", po));
+    DEBUG_X(PerlIO_printf(Perl_debug_log, "Pad free %lu\n", (unsigned long)po));
     if (curpad[po] && !SvIMMORTAL(curpad[po]))
        SvPADTMP_off(curpad[po]);
     if ((I32)po < padix)
@@ -426,7 +439,7 @@ pad_swipe(PADOFFSET po)
        croak("panic: pad_swipe curpad");
     if (!po)
        croak("panic: pad_swipe po");
-    DEBUG_X(PerlIO_printf(Perl_debug_log, "Pad swipe %d\n", po));
+    DEBUG_X(PerlIO_printf(Perl_debug_log, "Pad swipe %lu\n", (unsigned long)po));
     SvPADTMP_off(curpad[po]);
     curpad[po] = NEWSV(1107,0);
     SvPADTMP_on(curpad[po]);
@@ -595,12 +608,11 @@ OP *op;
     OP *kid;
 
     /* assumes no premature commitment */
-    if (!op || (op->op_flags & OPf_KNOW) || op->op_type == OP_RETURN
-        || error_count)
+    if (!op || (op->op_flags & OPf_WANT) || error_count
+        || op->op_type == OP_RETURN)
        return op;
 
-    op->op_flags &= ~OPf_LIST;
-    op->op_flags |= OPf_KNOW;
+    op->op_flags = (op->op_flags & ~OPf_WANT) | OPf_WANT_SCALAR;
 
     switch (op->op_type) {
     case OP_REPEAT:
@@ -631,8 +643,16 @@ OP *op;
        break;
     case OP_LEAVE:
     case OP_LEAVETRY:
-       scalar(cLISTOP->op_first);
-       /* FALL THROUGH */
+       kid = cLISTOP->op_first;
+       scalar(kid);
+       while (kid = kid->op_sibling) {
+           if (kid->op_sibling)
+               scalarvoid(kid);
+           else
+               scalar(kid);
+       }
+       curcop = &compiling;
+       break;
     case OP_SCOPE:
     case OP_LINESEQ:
     case OP_LIST:
@@ -656,12 +676,12 @@ OP *op;
     char* useless = 0;
     SV* sv;
 
-    if (!op || error_count)
-       return op;
-    if (op->op_flags & OPf_LIST)
+    /* assumes no premature commitment */
+    if (!op || (op->op_flags & OPf_WANT) == OPf_WANT_LIST || error_count
+        || op->op_type == OP_RETURN)
        return op;
 
-    op->op_flags |= OPf_KNOW;
+    op->op_flags = (op->op_flags & ~OPf_WANT) | OPf_WANT_VOID;
 
     switch (op->op_type) {
     default:
@@ -791,37 +811,38 @@ OP *op;
        for (kid = cUNOP->op_first->op_sibling; kid; kid = kid->op_sibling)
            scalarvoid(kid);
        break;
+
     case OP_NULL:
        if (op->op_targ == OP_NEXTSTATE || op->op_targ == OP_DBSTATE)
            curcop = ((COP*)op);                /* for warning below */
        if (op->op_flags & OPf_STACKED)
            break;
+       /* FALL THROUGH */
     case OP_ENTERTRY:
     case OP_ENTER:
     case OP_SCALAR:
        if (!(op->op_flags & OPf_KIDS))
            break;
+       /* FALL THROUGH */
     case OP_SCOPE:
     case OP_LEAVE:
     case OP_LEAVETRY:
     case OP_LEAVELOOP:
-       op->op_private |= OPpLEAVE_VOID;
     case OP_LINESEQ:
     case OP_LIST:
        for (kid = cLISTOP->op_first; kid; kid = kid->op_sibling)
            scalarvoid(kid);
        break;
+    case OP_REQUIRE:
+       /* since all requires must return a value, they're never void */
+       op->op_flags &= ~OPf_WANT;
+       return scalar(op);
     case OP_SPLIT:
        if ((kid = ((LISTOP*)op)->op_first) && kid->op_type == OP_PUSHRE) {
            if (!kPMOP->op_pmreplroot)
                deprecate("implicit split to @_");
        }
        break;
-    case OP_KEYS:
-    case OP_VALUES:
-    case OP_DELETE:
-       op->op_private |= OPpLEAVE_VOID;
-       break;
     }
     if (useless && dowarn)
        warn("Useless use of %s in void context", useless);
@@ -847,11 +868,11 @@ OP *op;
     OP *kid;
 
     /* assumes no premature commitment */
-    if (!op || (op->op_flags & OPf_KNOW) || op->op_type == OP_RETURN
-        || error_count)
+    if (!op || (op->op_flags & OPf_WANT) || error_count
+        || op->op_type == OP_RETURN)
        return op;
 
-    op->op_flags |= (OPf_KNOW | OPf_LIST);
+    op->op_flags = (op->op_flags & ~OPf_WANT) | OPf_WANT_LIST;
 
     switch (op->op_type) {
     case OP_FLOP:
@@ -879,8 +900,16 @@ OP *op;
        break;
     case OP_LEAVE:
     case OP_LEAVETRY:
-       list(cLISTOP->op_first);
-       /* FALL THROUGH */
+       kid = cLISTOP->op_first;
+       list(kid);
+       while (kid = kid->op_sibling) {
+           if (kid->op_sibling)
+               scalarvoid(kid);
+           else
+               list(kid);
+       }
+       curcop = &compiling;
+       break;
     case OP_SCOPE:
     case OP_LINESEQ:
        for (kid = cLISTOP->op_first; kid; kid = kid->op_sibling) {
@@ -950,6 +979,8 @@ I32 type;
        return op;
 
     switch (op->op_type) {
+    case OP_UNDEF:
+       return op;
     case OP_CONST:
        if (!(op->op_private & (OPpCONST_ARYBASE)))
            goto nomod;
@@ -1029,6 +1060,8 @@ I32 type;
        }
        /* FALL THROUGH */
     case OP_RV2GV:
+       if (scalar_mod_type(op, type))
+           goto nomod;
        ref(cUNOP->op_first, op->op_type);
        /* FALL THROUGH */
     case OP_AASSIGN:
@@ -1045,7 +1078,6 @@ I32 type;
            croak("Can't localize a reference");
        ref(cUNOP->op_first, op->op_type); 
        /* FALL THROUGH */
-    case OP_UNDEF:
     case OP_GV:
     case OP_AV2ARYLEN:
     case OP_SASSIGN:
@@ -1058,6 +1090,8 @@ I32 type;
        modcount = 10000;
        if (type == OP_REFGEN && op->op_flags & OPf_PARENS)
            return op;          /* Treat \(@foo) like ordinary list. */
+       if (scalar_mod_type(op, type))
+           goto nomod;
        /* FALL THROUGH */
     case OP_PADSV:
        modcount++;
@@ -1086,6 +1120,9 @@ I32 type;
     case OP_AELEM:
     case OP_HELEM:
        ref(cBINOP->op_first, op->op_type);
+       if (type == OP_ENTERSUB &&
+            !(op->op_private & (OPpLVAL_INTRO | OPpDEREF)))
+           op->op_private |= OPpLVAL_DEFER;
        modcount++;
        break;
 
@@ -1122,6 +1159,52 @@ I32 type;
     return op;
 }
 
+static bool
+scalar_mod_type(op, type)
+OP *op;
+I32 type;
+{
+    switch (type) {
+    case OP_SASSIGN:
+       if (op->op_type == OP_RV2GV)
+           return FALSE;
+       /* FALL THROUGH */
+    case OP_PREINC:
+    case OP_PREDEC:
+    case OP_POSTINC:
+    case OP_POSTDEC:
+    case OP_I_PREINC:
+    case OP_I_PREDEC:
+    case OP_I_POSTINC:
+    case OP_I_POSTDEC:
+    case OP_POW:
+    case OP_MULTIPLY:
+    case OP_DIVIDE:
+    case OP_MODULO:
+    case OP_REPEAT:
+    case OP_ADD:
+    case OP_SUBTRACT:
+    case OP_I_MULTIPLY:
+    case OP_I_DIVIDE:
+    case OP_I_MODULO:
+    case OP_I_ADD:
+    case OP_I_SUBTRACT:
+    case OP_LEFT_SHIFT:
+    case OP_RIGHT_SHIFT:
+    case OP_BIT_AND:
+    case OP_BIT_XOR:
+    case OP_BIT_OR:
+    case OP_CONCAT:
+    case OP_SUBST:
+    case OP_TRANS:
+    case OP_ANDASSIGN: /* may work later */
+    case OP_ORASSIGN:  /* may work later */
+       return TRUE;
+    default:
+       return FALSE;
+    }
+}
+
 OP *
 refkids(op, type)
 OP *op;
@@ -1264,6 +1347,20 @@ OP *right;
 {
     OP *op;
 
+    if (dowarn &&
+       (left->op_type == OP_RV2AV ||
+        left->op_type == OP_RV2HV ||
+        left->op_type == OP_PADAV ||
+        left->op_type == OP_PADHV)) {
+       char *desc = op_desc[(right->op_type == OP_SUBST ||
+                             right->op_type == OP_TRANS)
+                            ? right->op_type : OP_MATCH];
+       char *sample = ((left->op_type == OP_RV2AV ||
+                        left->op_type == OP_PADAV)
+                       ? "@array" : "%hash");
+       warn("Applying %s to %s will act on scalar(%s)", desc, sample, sample);
+    }
+
     if (right->op_type == OP_MATCH ||
        right->op_type == OP_SUBST ||
        right->op_type == OP_TRANS) {
@@ -1554,7 +1651,7 @@ OP* op;
     if (!op || op->op_type != OP_LIST)
        op = newLISTOP(OP_LIST, 0, op, Nullop);
     else
-       op->op_flags &= ~(OPf_KNOW|OPf_LIST);
+       op->op_flags &= ~OPf_WANT;
 
     if (!(opargs[type] & OA_MARK))
        null(cLISTOP->op_first);
@@ -2321,7 +2418,7 @@ OP *right;
                        tmpop->op_sibling = Nullop;     /* don't free split */
                        right->op_next = tmpop->op_next;  /* fix starting loc */
                        op_free(op);                    /* blow off assign */
-                       right->op_flags &= ~(OPf_KNOW|OPf_LIST);
+                       right->op_flags &= ~OPf_WANT;
                                /* "I don't know and I don't care." */
                        return right;
                    }
@@ -2496,16 +2593,20 @@ OP* other;
            break;
 
        case OP_SASSIGN:
-           if (k1->op_type == OP_READDIR || k1->op_type == OP_GLOB)
+           if (k1->op_type == OP_READDIR
+                 || k1->op_type == OP_GLOB
+                 || k1->op_type == OP_EACH)
                warnop = k1->op_type;
            break;
        }
        if (warnop) {
            line_t oldline = curcop->cop_line;
            curcop->cop_line = copline;
-           warn("Value of %s construct can be \"0\"; test with defined()",
-                op_desc[warnop]);
-               curcop->cop_line = oldline;
+           warn("Value of %s%s can be \"0\"; test with defined()",
+                op_desc[warnop],
+                ((warnop == OP_READLINE || warnop == OP_GLOB)
+                 ? " construct" : "() operator"));
+           curcop->cop_line = oldline;
        }
     }
 
@@ -2937,6 +3038,9 @@ CV* outside;
     if (outside)
        CvOUTSIDE(cv)   = (CV*)SvREFCNT_inc(outside);
 
+    if (SvPOK(proto))
+       sv_setpvn((SV*)cv, SvPVX(proto), SvCUR(proto));
+
     comppad = newAV();
 
     comppadlist = newAV();
@@ -3026,27 +3130,69 @@ CV* proto;
     return cv_clone2(proto, CvOUTSIDE(proto));
 }
 
+void
+cv_ckproto(cv, gv, p)
+CV* cv;
+GV* gv;
+char* p;
+{
+    if ((!p != !SvPOK(cv)) || (p && strNE(p, SvPVX(cv)))) {
+       char* buf;
+       SV* name = Nullsv;
+
+       if (gv)
+           gv_efullname3(name = NEWSV(606, 40), gv, Nullch);
+       New(607, buf, ((name ? SvCUR(name) : 0)
+                      + (SvPOK(cv) ? SvCUR(cv) : 0)
+                      + (p ? strlen(p) : 0)
+                      + 60), char);
+       strcpy(buf, "Prototype mismatch:");
+       if (name) {
+           sprintf(buf + strlen(buf), " sub %s", SvPVX(name));
+           SvREFCNT_dec(name);
+       }
+       if (SvPOK(cv))
+           sprintf(buf + strlen(buf), " (%s)", SvPVX(cv));
+       strcat(buf, " vs ");
+       sprintf(buf + strlen(buf), p ? "(%s)" : "none", p);
+       warn("%s", buf);
+       Safefree(buf);
+    }
+}
+
 SV *
 cv_const_sv(cv)
-CV *cv;
+CV* cv;
 {
     OP *o;
-    SV *sv = Nullsv;
+    SV *sv;
     
-    if(cv && SvPOK(cv) && !SvCUR(cv)) {
-       for (o = CvSTART(cv); o; o = o->op_next) {
-           OPCODE type = o->op_type;
-       
-           if (type == OP_NEXTSTATE || type == OP_NULL || type == OP_PUSHMARK)
-               continue;
-           if (type == OP_LEAVESUB || type == OP_RETURN)
-               break;
-           if (type != OP_CONST || sv)
-               return Nullsv;
+    if (!cv || !SvPOK(cv) || SvCUR(cv))
+       return Nullsv;
 
+    sv = Nullsv;
+    for (o = CvSTART(cv); o; o = o->op_next) {
+       OPCODE type = o->op_type;
+       
+       if (type == OP_NEXTSTATE || type == OP_NULL || type == OP_PUSHMARK)
+           continue;
+       if (type == OP_LEAVESUB || type == OP_RETURN)
+           break;
+       if (sv)
+           return Nullsv;
+       if (type == OP_CONST)
            sv = ((SVOP*)o)->op_sv;
+       else if (type == OP_PADSV) {
+           AV* pad = (AV*)(AvARRAY(CvPADLIST(cv))[1]);
+           sv = pad ? AvARRAY(pad)[o->op_targ] : Nullsv;
+           if (!sv || (!SvREADONLY(sv) && SvREFCNT(sv) > 1))
+               return Nullsv;
        }
+       else
+           return Nullsv;
     }
+    if (sv)
+       SvREADONLY_on(sv);
     return sv;
 }
 
@@ -3059,43 +3205,37 @@ OP *block;
 {
     char *name = op ? SvPVx(cSVOP->op_sv, na) : Nullch;
     GV *gv = gv_fetchpv(name ? name : "__ANON__", GV_ADDMULTI, SVt_PVCV);
+    char *ps = proto ? SvPVx(((SVOP*)proto)->op_sv, na) : Nullch;
     register CV *cv;
-    AV *av;
     I32 ix;
 
     if (op)
        SAVEFREEOP(op);
-    if (cv = (name ? GvCV(gv) : Nullcv)) {
-       if (GvCVGEN(gv)) {
-           /* just a cached method */
-           SvREFCNT_dec(cv);
-           cv = 0;
-       }
-       else if (CvROOT(cv) || CvXSUB(cv) || GvASSUMECV(gv)) {
-           /* already defined (or promised) */
-
-           SV* const_sv = cv_const_sv(cv);
-           char *p = proto ? SvPVx(((SVOP*)proto)->op_sv, na) : Nullch;
-
-           if((!proto != !SvPOK(cv)) || (p && strNE(SvPV((SV*)cv,na), p))) {
-               warn("Prototype mismatch: (%s) vs (%s)",
-                       SvPOK(cv) ? SvPV((SV*)cv,na) : "none",
-                       p ? p : "none");
-           }
+    if (proto)
+       SAVEFREEOP(proto);
+
+    if (!name || GvCVGEN(gv))
+       cv = Nullcv;
+    else if (cv = GvCV(gv)) {
+       cv_ckproto(cv, gv, ps);
+       /* already defined (or promised)? */
+       if (CvROOT(cv) || CvXSUB(cv) || GvASSUMECV(gv)) {
+           SV* const_sv;
            if (!block) {
                /* just a "sub foo;" when &foo is already defined */
                SAVEFREESV(compcv);
                goto done;
            }
+           const_sv = cv_const_sv(cv);
            if (const_sv || dowarn) {
                line_t oldline = curcop->cop_line;
                curcop->cop_line = copline;
                warn(const_sv ? "Constant subroutine %s redefined"
-                             : "Subroutine %s redefined",name);
+                    : "Subroutine %s redefined", name);
                curcop->cop_line = oldline;
            }
            SvREFCNT_dec(cv);
-           cv = 0;
+           cv = Nullcv;
        }
     }
     if (cv) {                          /* must reuse cv if autoloaded */
@@ -3121,15 +3261,27 @@ OP *block;
     CvFILEGV(cv) = curcop->cop_filegv;
     CvSTASH(cv) = curstash;
 
-    if (proto) {
-       char *p = SvPVx(((SVOP*)proto)->op_sv, na);
-       sv_setpv((SV*)cv, p);
-       op_free(proto);
-    }
+    if (ps)
+       sv_setpv((SV*)cv, ps);
 
     if (error_count) {
        op_free(block);
        block = Nullop;
+       if (name) {
+           char *s = strrchr(name, ':');
+           s = s ? s+1 : name;
+           if (strEQ(s, "BEGIN")) {
+               char *not_safe =
+                   "BEGIN not safe after errors--compilation aborted";
+               if (in_eval & 4)
+                   croak(not_safe);
+               else {
+                   /* force display of errors found but not reported */
+                   sv_catpv(GvSV(errgv), not_safe);
+                   croak("%s", SvPVx(GvSV(errgv), na));
+               }
+           }
+       }
     }
     if (!block) {
        copline = NOLINE;
@@ -3137,18 +3289,44 @@ OP *block;
        return cv;
     }
 
-    av = newAV();                      /* Will be @_ */
-    av_extend(av, 0);
-    av_store(comppad, 0, (SV*)av);
-    AvFLAGS(av) = AVf_REIFY;
+    if (AvFILL(comppad_name) < AvFILL(comppad))
+       av_store(comppad_name, AvFILL(comppad), Nullsv);
 
-    for (ix = AvFILL(comppad); ix > 0; ix--) {
-       if (!SvPADMY(curpad[ix]) && !SvIMMORTAL(curpad[ix]))
-           SvPADTMP_on(curpad[ix]);
+    if (CvCLONE(cv)) {
+       SV **namep = AvARRAY(comppad_name);
+       for (ix = AvFILL(comppad); ix > 0; ix--) {
+           SV *namesv;
+
+           if (SvIMMORTAL(curpad[ix]))
+               continue;
+           /*
+            * The only things that a clonable function needs in its
+            * pad are references to outer lexicals and anonymous subs.
+            * The rest are created anew during cloning.
+            */
+           if (!((namesv = namep[ix]) != Nullsv &&
+                 namesv != &sv_undef &&
+                 (SvFAKE(namesv) ||
+                  *SvPVX(namesv) == '&')))
+           {
+               SvREFCNT_dec(curpad[ix]);
+               curpad[ix] = Nullsv;
+           }
+       }
     }
+    else {
+       AV *av = newAV();                       /* Will be @_ */
+       av_extend(av, 0);
+       av_store(comppad, 0, (SV*)av);
+       AvFLAGS(av) = AVf_REIFY;
 
-    if (AvFILL(comppad_name) < AvFILL(comppad))
-       av_store(comppad_name, AvFILL(comppad), Nullsv);
+       for (ix = AvFILL(comppad); ix > 0; ix--) {
+           if (SvIMMORTAL(curpad[ix]))
+               continue;
+           if (!SvPADMY(curpad[ix]))
+               SvPADTMP_on(curpad[ix]);
+       }
+    }
 
     CvROOT(cv) = newUNOP(OP_LEAVESUB, 0, scalarseq(block));
     CvSTART(cv) = LINKLIST(CvROOT(cv));
@@ -3191,7 +3369,8 @@ OP *block;
            s++;
        else
            s = name;
-       if (strEQ(s, "BEGIN") && !error_count) {
+       if (strEQ(s, "BEGIN")) {
+           I32 oldscope = scopestack_ix;
            ENTER;
            SAVESPTR(compiling.cop_filegv);
            SAVEI16(compiling.cop_line);
@@ -3204,7 +3383,7 @@ OP *block;
            DEBUG_x( dump_sub(gv) );
            av_push(beginav, (SV *)cv);
            GvCV(gv) = 0;
-           calllist(beginav);
+           call_list(oldscope, beginav);
 
            curcop = &compiling;
            LEAVE;
@@ -4399,6 +4578,7 @@ OP *op;
        }
        else
            list(o);
+       mod(o, OP_ENTERSUB);
        prev = o;
        o = o->op_sibling;
     }
@@ -4469,9 +4649,9 @@ register OP* o;
            o->op_seq = op_seqmax++;
            break;
        case OP_STUB:
-           if ((o->op_flags & (OPf_KNOW|OPf_LIST)) != (OPf_KNOW|OPf_LIST)) {
+           if ((o->op_flags & OPf_WANT) != OPf_WANT_LIST) {
                o->op_seq = op_seqmax++;
-               break;  /* Scalar stub must produce undef.  List stub is noop */
+               break; /* Scalar stub must produce undef.  List stub is noop */
            }
            goto nothin;
        case OP_NULL:
@@ -4505,7 +4685,8 @@ register OP* o;
                if (pop->op_type == OP_CONST &&
                    (op = pop->op_next) &&
                    pop->op_next->op_type == OP_AELEM &&
-                   !(pop->op_next->op_private & (OPpDEREF|OPpLVAL_INTRO)) &&
+                   !(pop->op_next->op_private &
+                     (OPpLVAL_INTRO|OPpLVAL_DEFER|OPpDEREF)) &&
                    (i = SvIV(((SVOP*)pop)->op_sv) - compiling.cop_arybase)
                                <= 255 &&
                    i >= 0)