summaryrefslogtreecommitdiff
path: root/pp.c
diff options
context:
space:
mode:
Diffstat (limited to 'pp.c')
-rw-r--r--pp.c1052
1 files changed, 659 insertions, 393 deletions
diff --git a/pp.c b/pp.c
index c819f38c69..d5c33d189a 100644
--- a/pp.c
+++ b/pp.c
@@ -58,8 +58,12 @@ extern int h_errno;
#include <sys/file.h>
#endif
-#ifdef I_VARARGS
-# include <varargs.h>
+#ifdef STANDARD_C
+# include <stdarg.h>
+#else
+# ifdef I_VARARGS
+# include <varargs.h>
+# endif
#endif
static I32 dopoptosub P((I32 startingblock));
@@ -175,8 +179,8 @@ PP(pp_padsv)
{
dSP; dTARGET;
XPUSHs(TARG);
- if ((op->op_flags & (OPf_INTRO|OPf_SPECIAL)) == OPf_INTRO)
- SvOK_off(TARG);
+ if (op->op_flags & OPf_INTRO)
+ SAVECLEARSV(curpad[op->op_targ]);
RETURN;
}
@@ -184,8 +188,8 @@ PP(pp_padav)
{
dSP; dTARGET;
XPUSHs(TARG);
- if ((op->op_flags & (OPf_INTRO|OPf_SPECIAL)) == OPf_INTRO)
- av_clear((AV*)TARG);
+ if (op->op_flags & OPf_INTRO)
+ SAVECLEARSV(curpad[op->op_targ]);
if (op->op_flags & OPf_LVAL)
RETURN;
PUTBACK;
@@ -196,8 +200,8 @@ PP(pp_padhv)
{
dSP; dTARGET;
XPUSHs(TARG);
- if ((op->op_flags & (OPf_INTRO|OPf_SPECIAL)) == OPf_INTRO)
- hv_clear((HV*)TARG);
+ if (op->op_flags & OPf_INTRO)
+ SAVECLEARSV(curpad[op->op_targ]);
if (op->op_flags & OPf_LVAL)
RETURN;
PUTBACK;
@@ -229,7 +233,7 @@ PP(pp_rv2gv)
else {
if (SvTYPE(sv) != SVt_PVGV) {
if (!SvOK(sv))
- DIE(no_usym);
+ DIE(no_usym, "a glob");
sv = (SV*)gv_fetchpv(SvPV(sv, na), TRUE);
}
}
@@ -282,26 +286,28 @@ PP(pp_rv2sv)
GV *gv = sv;
if (SvTYPE(gv) != SVt_PVGV) {
if (!SvOK(sv))
- DIE(no_usym);
+ DIE(no_usym, "a scalar");
gv = (SV*)gv_fetchpv(SvPV(sv, na), TRUE);
}
sv = GvSV(gv);
if (op->op_private == OP_RV2HV &&
(!SvROK(sv) || SvTYPE(SvRV(sv)) != SVt_PVHV)) {
- sv_free(sv);
+ SvREFCNT_dec(sv);
sv = NEWSV(0,0);
sv_upgrade(sv, SVt_RV);
- SvRV(sv) = sv_ref((SV*)newHV());
+ SvRV(sv) = SvREFCNT_inc(newHV());
SvROK_on(sv);
+ ++sv_rvcount;
GvSV(gv) = sv;
}
else if (op->op_private == OP_RV2AV &&
(!SvROK(sv) || SvTYPE(SvRV(sv)) != SVt_PVAV)) {
- sv_free(sv);
+ SvREFCNT_dec(sv);
sv = NEWSV(0,0);
sv_upgrade(sv, SVt_RV);
- SvRV(sv) = sv_ref((SV*)newAV());
+ SvRV(sv) = SvREFCNT_inc(newAV());
SvROK_on(sv);
+ ++sv_rvcount;
GvSV(gv) = sv;
}
}
@@ -332,7 +338,9 @@ PP(pp_rv2cv)
SV *sv;
GV *gv;
HV *stash;
- CV *cv = sv_2cv(TOPs, &stash, &gv, 0);
+
+ /* We always try to add a non-existent subroutine in case of AUTOLOAD. */
+ CV *cv = sv_2cv(TOPs, &stash, &gv, TRUE);
SETs((SV*)cv);
RETURN;
@@ -344,10 +352,11 @@ PP(pp_refgen)
SV* rv;
if (!sv)
RETSETUNDEF;
- rv = sv_mortalcopy(&sv_undef);
+ rv = sv_newmortal();
sv_upgrade(rv, SVt_RV);
- SvRV(rv) = sv_ref(sv);
+ SvRV(rv) = SvREFCNT_inc(sv);
SvROK_on(rv);
+ ++sv_rvcount;
SETs(rv);
RETURN;
}
@@ -417,7 +426,7 @@ PP(pp_bless)
ref = SvRV(sv);
SvOBJECT_on(ref);
SvUPGRADE(ref, SVt_PVMG);
- SvSTASH(ref) = stash;
+ SvSTASH(ref) = (HV*)SvREFCNT_inc(stash);
RETURN;
}
@@ -444,7 +453,7 @@ PP(pp_backtick)
for (;;) {
sv = NEWSV(56, 80);
if (sv_gets(sv, fp, 0) == Nullch) {
- sv_free(sv);
+ SvREFCNT_dec(sv);
break;
}
XPUSHs(sv_2mortal(sv));
@@ -478,26 +487,28 @@ do_readline()
fp = Nullfp;
if (io) {
- fp = io->ifp;
+ fp = IoIFP(io);
if (!fp) {
- if (io->flags & IOf_ARGV) {
- if (io->flags & IOf_START) {
- io->flags &= ~IOf_START;
- io->lines = 0;
+ if (IoFLAGS(io) & IOf_ARGV) {
+ if (IoFLAGS(io) & IOf_START) {
+ IoFLAGS(io) &= ~IOf_START;
+ IoLINES(io) = 0;
if (av_len(GvAVn(last_in_gv)) < 0) {
SV *tmpstr = newSVpv("-", 1); /* assume stdin */
(void)av_push(GvAVn(last_in_gv), tmpstr);
}
}
fp = nextargv(last_in_gv);
- if (!fp) { /* Note: fp != io->ifp */
+ if (!fp) { /* Note: fp != IoIFP(io) */
(void)do_close(last_in_gv, FALSE); /* now it does*/
- io->flags |= IOf_START;
+ IoFLAGS(io) |= IOf_START;
}
}
else if (type == OP_GLOB) {
SV *tmpcmd = NEWSV(55, 0);
SV *tmpglob = POPs;
+ ENTER;
+ SAVEFREESV(tmpcmd);
#ifdef DOSISH
sv_setpv(tmpcmd, "perlglob ");
sv_catsv(tmpcmd, tmpglob);
@@ -515,8 +526,8 @@ do_readline()
#endif /* !CSH */
#endif /* !MSDOS */
(void)do_open(last_in_gv, SvPVX(tmpcmd), SvCUR(tmpcmd));
- fp = io->ifp;
- sv_free(tmpcmd);
+ fp = IoIFP(io);
+ LEAVE;
}
}
else if (type == OP_GLOB)
@@ -547,12 +558,12 @@ do_readline()
for (;;) {
if (!sv_gets(sv, fp, offset)) {
clearerr(fp);
- if (io->flags & IOf_ARGV) {
+ if (IoFLAGS(io) & IOf_ARGV) {
fp = nextargv(last_in_gv);
if (fp)
continue;
(void)do_close(last_in_gv, FALSE);
- io->flags |= IOf_START;
+ IoFLAGS(io) |= IOf_START;
}
else if (type == OP_GLOB) {
(void)do_close(last_in_gv, FALSE);
@@ -561,7 +572,7 @@ do_readline()
RETPUSHUNDEF;
RETURN;
}
- io->lines++;
+ IoLINES(io)++;
XPUSHs(sv);
if (tainting) {
tainted = TRUE;
@@ -777,7 +788,7 @@ play_it_again:
}
}
if (--BmUSEFUL(pm->op_pmshort) < 0) {
- sv_free(pm->op_pmshort);
+ SvREFCNT_dec(pm->op_pmshort);
pm->op_pmshort = Nullsv; /* opt is being useless */
}
}
@@ -808,7 +819,7 @@ play_it_again:
i = 0;
EXTEND(SP, iters + i);
for (i = !i; i <= iters; i++) {
- PUSHs(sv_mortalcopy(&sv_no));
+ PUSHs(sv_newmortal());
/*SUPPRESS 560*/
if (s = rx->startp[i]) {
len = rx->endp[i] - s;
@@ -955,7 +966,7 @@ PP(pp_subst)
}
}
if (--BmUSEFUL(pm->op_pmshort) < 0) {
- sv_free(pm->op_pmshort);
+ SvREFCNT_dec(pm->op_pmshort);
pm->op_pmshort = Nullsv; /* opt is being useless */
}
}
@@ -1049,7 +1060,7 @@ PP(pp_subst)
}
SvPOK_only(TARG);
SvSETMAGIC(TARG);
- PUSHs(sv_2mortal(newSVnv((double)iters)));
+ PUSHs(sv_2mortal(newSViv((I32)iters)));
RETURN;
}
PUSHs(&sv_no);
@@ -1092,7 +1103,7 @@ PP(pp_subst)
sv_replace(TARG, dstr);
SvPOK_only(TARG);
SvSETMAGIC(TARG);
- PUSHs(sv_2mortal(newSVnv((double)iters)));
+ PUSHs(sv_2mortal(newSViv((I32)iters)));
RETURN;
}
PUSHs(&sv_no);
@@ -1133,7 +1144,7 @@ PP(pp_substcont)
sv_replace(targ, dstr);
SvPOK_only(targ);
SvSETMAGIC(targ);
- PUSHs(sv_2mortal(newSVnv((double)(cx->sb_iters - 1))));
+ PUSHs(sv_2mortal(newSViv((I32)cx->sb_iters - 1)));
POPSUBST(cx);
RETURNOP(pm->op_next);
}
@@ -1175,7 +1186,7 @@ PP(pp_trans)
PP(pp_sassign)
{
dSP; dPOPTOPssrl;
- if (tainting && tainted && (!SvMAGICAL(lstr) || !mg_find(lstr, 't'))) {
+ if (tainting && tainted && (!SvRMAGICAL(lstr) || !mg_find(lstr, 't'))) {
TAINT_NOT;
}
SvSetSV(rstr, lstr);
@@ -1225,7 +1236,7 @@ PP(pp_aassign)
switch (SvTYPE(sv)) {
case SVt_PVAV:
ary = (AV*)sv;
- magic = SvMAGICAL(ary) != 0;
+ magic = SvSMAGICAL(ary) != 0;
AvREAL_on(ary);
AvFILL(ary) = -1;
i = 0;
@@ -1244,7 +1255,7 @@ PP(pp_aassign)
SV *tmpstr;
hash = (HV*)sv;
- magic = SvMAGICAL(hash) != 0;
+ magic = SvSMAGICAL(hash) != 0;
hv_clear(hash);
while (relem < lastrelem) { /* gobble up all the rest */
@@ -1266,7 +1277,7 @@ PP(pp_aassign)
break;
default:
if (SvTHINKFIRST(sv)) {
- if (SvREADONLY(sv)) {
+ if (SvREADONLY(sv) && curcop != &compiling) {
if (sv != &sv_undef && sv != &sv_yes && sv != &sv_no)
DIE(no_modify);
if (relem <= lastrelem)
@@ -1520,7 +1531,7 @@ PP(pp_study)
SvSCREAM_on(TARG);
retval = 1;
ret:
- XPUSHs(sv_2mortal(newSVnv((double)retval)));
+ XPUSHs(sv_2mortal(newSViv((I32)retval)));
RETURN;
}
@@ -1546,6 +1557,8 @@ PP(pp_postinc)
sv_setsv(TARG, TOPs);
sv_inc(TOPs);
SvSETMAGIC(TOPs);
+ if (!SvOK(TARG))
+ sv_setiv(TARG, 0);
SETs(TARG);
return NORMAL;
}
@@ -1651,8 +1664,8 @@ PP(pp_repeat)
char *tmps;
tmpstr = POPs;
- if (SvTHINKFIRST(tmpstr)) {
- if (SvREADONLY(tmpstr))
+ if (TARG == tmpstr && SvTHINKFIRST(tmpstr)) {
+ if (SvREADONLY(tmpstr) && curcop != &compiling)
DIE("Can't x= to readonly value");
if (SvROK(tmpstr))
sv_unref(tmpstr);
@@ -1670,7 +1683,7 @@ PP(pp_repeat)
SvCUR(TARG) *= count;
*SvEND(TARG) = '\0';
SvPOK_only(TARG);
- sv_free(tmpstr);
+ SvREFCNT_dec(tmpstr);
}
else
sv_setsv(TARG, &sv_no);
@@ -2160,7 +2173,7 @@ PP(pp_substr)
sv_setpvn(TARG, tmps, rem);
if (lvalue) { /* it's an lvalue! */
if (SvTHINKFIRST(sv)) {
- if (SvREADONLY(sv))
+ if (SvREADONLY(sv) && curcop != &compiling)
DIE(no_modify);
if (SvROK(sv))
sv_unref(sv);
@@ -2216,7 +2229,7 @@ PP(pp_vec)
if (lvalue) { /* it's an lvalue! */
if (SvTHINKFIRST(src)) {
- if (SvREADONLY(src))
+ if (SvREADONLY(src) && curcop != &compiling)
DIE(no_modify);
if (SvROK(src))
sv_unref(src);
@@ -2506,8 +2519,10 @@ PP(pp_formline)
bool gotsome;
STRLEN len;
- if (!SvCOMPILED(form))
+ if (!SvCOMPILED(form)) {
+ SvREADONLY_off(form);
doparseform(form);
+ }
SvUPGRADE(formtarget, SVt_PV);
SvGROW(formtarget, SvCUR(formtarget) + SvCUR(form) + 1);
@@ -2600,25 +2615,37 @@ PP(pp_formline)
case FF_CHECKCHOP:
s = SvPV(sv, len);
itemsize = len;
- if (itemsize > fieldsize)
- itemsize = fieldsize;
- send = chophere = s + itemsize;
- while (s < send || (s == send && isSPACE(*s))) {
- if (isSPACE(*s)) {
- if (chopspace)
- chophere = s;
- if (*s == '\r')
+ if (itemsize <= fieldsize) {
+ send = chophere = s + itemsize;
+ while (s < send) {
+ if (*s == '\r') {
+ itemsize = s - SvPVX(sv);
break;
- }
- else {
- if (*s & ~31)
+ }
+ if (*s++ & ~31)
gotsome = TRUE;
- if (strchr(chopset, *s))
- chophere = s + 1;
}
- s++;
}
- itemsize = chophere - SvPVX(sv);
+ else {
+ itemsize = fieldsize;
+ send = chophere = s + itemsize;
+ while (s < send || (s == send && isSPACE(*s))) {
+ if (isSPACE(*s)) {
+ if (chopspace)
+ chophere = s;
+ if (*s == '\r')
+ break;
+ }
+ else {
+ if (*s & ~31)
+ gotsome = TRUE;
+ if (strchr(chopset, *s))
+ chophere = s + 1;
+ }
+ s++;
+ }
+ itemsize = chophere - SvPVX(sv);
+ }
break;
case FF_SPACE:
@@ -2935,7 +2962,7 @@ PP(pp_rv2av)
else {
if (SvTYPE(sv) != SVt_PVGV) {
if (!SvOK(sv))
- DIE(no_usym);
+ DIE(no_usym, "an array");
sv = (SV*)gv_fetchpv(SvPV(sv, na), TRUE);
}
av = GvAVn(sv);
@@ -2965,8 +2992,8 @@ PP(pp_rv2av)
PP(pp_aelemfast)
{
dSP;
- AV *av = (AV*)cSVOP->op_sv;
- SV** svp = av_fetch(av, op->op_private - arybase, FALSE);
+ AV *av = GvAV((GV*)cSVOP->op_sv);
+ SV** svp = av_fetch(av, op->op_private - arybase, op->op_flags & OPf_LVAL);
PUSHs(svp ? *svp : &sv_undef);
RETURN;
}
@@ -2986,18 +3013,20 @@ PP(pp_aelem)
save_svref(svp);
else if (!SvOK(*svp)) {
if (op->op_private == OP_RV2HV) {
- sv_free(*svp);
+ SvREFCNT_dec(*svp);
*svp = NEWSV(0,0);
sv_upgrade(*svp, SVt_RV);
- SvRV(*svp) = sv_ref((SV*)newHV());
+ SvRV(*svp) = SvREFCNT_inc(newHV());
SvROK_on(*svp);
+ ++sv_rvcount;
}
else if (op->op_private == OP_RV2AV) {
- sv_free(*svp);
+ SvREFCNT_dec(*svp);
*svp = NEWSV(0,0);
sv_upgrade(*svp, SVt_RV);
- SvRV(*svp) = sv_ref((SV*)newAV());
+ SvRV(*svp) = SvREFCNT_inc(newAV());
SvROK_on(*svp);
+ ++sv_rvcount;
}
}
}
@@ -3047,22 +3076,16 @@ PP(pp_each)
I32 i;
char *tmps;
- if (mystrk) {
- sv_free(mystrk);
- mystrk = Nullsv;
- }
-
EXTEND(SP, 2);
if (entry) {
+ tmps = hv_iterkey(entry, &i);
+ if (!i)
+ tmps = "";
+ PUSHs(sv_2mortal(newSVpv(tmps, i)));
if (GIMME == G_ARRAY) {
- tmps = hv_iterkey(entry, &i);
- if (!i)
- tmps = "";
- mystrk = newSVpv(tmps, i);
- PUSHs(mystrk);
+ sv_setsv(TARG, hv_iterval(hash, entry));
+ PUSHs(TARG);
}
- sv_setsv(TARG, hv_iterval(hash, entry));
- PUSHs(TARG);
}
else if (GIMME == G_SCALAR)
RETPUSHUNDEF;
@@ -3128,7 +3151,7 @@ PP(pp_rv2hv)
else {
if (SvTYPE(sv) != SVt_PVGV) {
if (!SvOK(sv))
- DIE(no_usym);
+ DIE(no_usym, "a hash");
sv = (SV*)gv_fetchpv(SvPV(sv, na), TRUE);
}
hv = GvHVn(sv);
@@ -3147,12 +3170,12 @@ PP(pp_rv2hv)
}
else {
dTARGET;
- if (HvFILL(hv))
- sv_setiv(TARG, 0);
- else {
+ if (HvFILL(hv)) {
sprintf(buf, "%d/%d", HvFILL(hv), HvMAX(hv)+1);
sv_setpv(TARG, buf);
}
+ else
+ sv_setiv(TARG, 0);
SETTARG;
RETURN;
}
@@ -3175,18 +3198,20 @@ PP(pp_helem)
save_svref(svp);
else if (!SvOK(*svp)) {
if (op->op_private == OP_RV2HV) {
- sv_free(*svp);
+ SvREFCNT_dec(*svp);
*svp = NEWSV(0,0);
sv_upgrade(*svp, SVt_RV);
- SvRV(*svp) = sv_ref((SV*)newHV());
+ SvRV(*svp) = SvREFCNT_inc(newHV());
SvROK_on(*svp);
+ ++sv_rvcount;
}
else if (op->op_private == OP_RV2AV) {
- sv_free(*svp);
+ SvREFCNT_dec(*svp);
*svp = NEWSV(0,0);
sv_upgrade(*svp, SVt_RV);
- SvRV(*svp) = sv_ref((SV*)newAV());
+ SvRV(*svp) = SvREFCNT_inc(newAV());
SvROK_on(*svp);
+ ++sv_rvcount;
}
}
}
@@ -4279,6 +4304,8 @@ PP(pp_split)
DIE("panic: do_split");
if (pm->op_pmreplroot)
ary = GvAVn((GV*)pm->op_pmreplroot);
+ else if (gimme != G_ARRAY)
+ ary = GvAVn(defgv);
else
ary = Nullav;
if (ary && (gimme != G_ARRAY || (pm->op_pmflags & PMf_ONCE))) {
@@ -4455,17 +4482,14 @@ PP(pp_join)
PP(pp_list)
{
- dSP;
+ dSP; dMARK;
if (GIMME != G_ARRAY) {
- dMARK;
if (++MARK <= SP)
*MARK = *SP; /* unwanted list, return last item */
else
*MARK = &sv_undef;
SP = MARK;
}
- else if (op->op_private & OPpLIST_GUESSED) /* didn't need that pushmark */
- markstack_ptr--;
RETURN;
}
@@ -4626,7 +4650,7 @@ PP(pp_splice)
if (AvREAL(ary)) {
sv_2mortal(*MARK);
for (i = length - 1, dst = &AvARRAY(ary)[offset]; i > 0; i--)
- sv_free(*dst++); /* free them now */
+ SvREFCNT_dec(*dst++); /* free them now */
}
}
AvFILL(ary) += diff;
@@ -4690,7 +4714,7 @@ PP(pp_splice)
dst = AvARRAY(ary) + AvFILL(ary);
for (i = diff; i > 0; i--) {
if (*dst) /* stuff was hanging around */
- sv_free(*dst); /* after $#foo */
+ SvREFCNT_dec(*dst); /* after $#foo */
dst--;
}
if (after) {
@@ -4724,7 +4748,7 @@ PP(pp_splice)
if (AvREAL(ary)) {
sv_2mortal(*MARK);
while (length-- > 0)
- sv_free(tmparyval[length]);
+ SvREFCNT_dec(tmparyval[length]);
}
Safefree(tmparyval);
}
@@ -4823,7 +4847,7 @@ PP(pp_grepstart)
GvSV(defgv) = src;
}
else
- GvSV(defgv) = sv_mortalcopy(&sv_undef);
+ GvSV(defgv) = sv_newmortal();
RETURNOP(((LOGOP*)op->op_next)->op_other);
}
@@ -4865,7 +4889,7 @@ PP(pp_grepwhile)
GvSV(defgv) = src;
}
else
- GvSV(defgv) = sv_mortalcopy(&sv_undef);
+ GvSV(defgv) = sv_newmortal();
RETURNOP(cLOGOP->op_other);
}
@@ -4903,7 +4927,7 @@ PP(pp_sort)
cv = sv_2cv(*++MARK, &stash, &gv, 0);
if (!(cv && CvROOT(cv))) {
if (gv) {
- SV *tmpstr = sv_mortalcopy(&sv_undef);
+ SV *tmpstr = sv_newmortal();
gv_efullname(tmpstr, gv);
if (CvUSERSUB(cv))
DIE("Usersub \"%s\" called in sort", SvPVX(tmpstr));
@@ -5042,7 +5066,7 @@ PP(pp_flip)
SV *targ = PAD_SV(op->op_targ);
if ((op->op_private & OPpFLIP_LINENUM)
- ? last_in_gv && SvIV(sv) == GvIO(last_in_gv)->lines
+ ? last_in_gv && SvIV(sv) == IoLINES(GvIO(last_in_gv))
: SvTRUE(sv) ) {
sv_setiv(PAD_SV(cUNOP->op_first->op_targ), 1);
if (op->op_flags & OPf_SPECIAL) {
@@ -5104,7 +5128,7 @@ PP(pp_flop)
SV *targ = PAD_SV(cUNOP->op_first->op_targ);
sv_inc(targ);
if ((op->op_private & OPpFLIP_LINENUM)
- ? last_in_gv && SvIV(sv) == GvIO(last_in_gv)->lines
+ ? last_in_gv && SvIV(sv) == IoLINES(GvIO(last_in_gv))
: SvTRUE(sv) ) {
sv_setiv(PAD_SV(((UNOP*)cUNOP->op_first)->op_first->op_targ), 0);
sv_catpv(targ, "E0");
@@ -5230,8 +5254,8 @@ I32 cxix;
while (cxstack_ix > cxix) {
cx = &cxstack[cxstack_ix--];
- DEBUG_l(fprintf(stderr, "Unwinding block %d, type %d\n", cxstack_ix+1,
- cx->cx_type));
+ DEBUG_l(fprintf(stderr, "Unwinding block %d, type %s\n", cxstack_ix+1,
+ block_type[cx->cx_type]));
/* Note: we don't need to restore the base context info till the end. */
switch (cx->cx_type) {
case CXt_SUB:
@@ -5249,13 +5273,15 @@ I32 cxix;
}
}
-/*VARARGS0*/
+#ifdef STANDARD_C
OP *
-#ifdef __STDC__
-die(char* pat,...)
+die(char* pat, ...)
#else
-die(va_alist)
-va_dcl
+/*VARARGS0*/
+OP *
+die(pat, va_alist)
+ char *pat;
+ va_dcl
#endif
{
va_list args;
@@ -5263,8 +5289,12 @@ va_dcl
char *message;
OP *retop;
+#ifdef STANDARD_C
+ va_start(args, pat);
+#else
va_start(args);
- message = mess(args);
+#endif
+ message = mess(pat, args);
va_end(args);
restartop = die_where(message);
if (stack != mainstack)
@@ -5382,11 +5412,11 @@ PP(pp_method)
if (!SvOK(sv) ||
!(iogv = gv_fetchpv(SvPVX(sv), FALSE)) ||
- !(io=GvIO(iogv)))
+ !(ob=(SV*)GvIO(iogv)))
{
char *name = SvPVX(((SVOP*)cLOGOP->op_other)->op_sv);
char tmpbuf[256];
- char* packname = SvPVX(sv);
+ char* packname = SvPV(sv, na);
HV *stash;
if (!isALPHA(*packname))
DIE("Can't call method \"%s\" without a package or object reference", name);
@@ -5401,14 +5431,6 @@ DIE("Can't call method \"%s\" without a package or object reference", name);
PUSHs(sv);
RETURN;
}
- if (!(ob = io->object)) {
- ob = sv_ref((SV*)newHV());
- SvOBJECT_on(ob);
- SvUPGRADE(ob, SVt_PVMG);
- iogv = gv_fetchpv("FILEHANDLE'flush", TRUE);
- SvSTASH(ob) = GvSTASH(iogv);
- io->object = ob;
- }
}
if (!ob || !SvOBJECT(ob)) {
@@ -5432,30 +5454,74 @@ DIE("Can't call method \"%s\" without a package or object reference", name);
PP(pp_entersubr)
{
dSP; dMARK;
- SV *sv;
+ SV *sv = *++MARK;
GV *gv;
HV *stash;
- register CV *cv = sv_2cv(*++MARK, &stash, &gv, 0);
+ register CV *cv;
register I32 items = SP - MARK;
I32 hasargs = (op->op_flags & OPf_STACKED) != 0;
register CONTEXT *cx;
+ if (!sv)
+ DIE("Not a subroutine reference");
+ switch (SvTYPE(sv)) {
+ default:
+ if (!SvROK(sv)) {
+ if (!SvOK(sv))
+ DIE(no_usym, "a subroutine");
+ gv = gv_fetchpv(SvPV(sv, na), FALSE);
+ if (!gv)
+ cv = 0;
+ else
+ cv = GvCV(gv);
+ break;
+ }
+ /* FALL THROUGH */
+ case SVt_RV:
+ cv = (CV*)SvRV(sv);
+ if (SvTYPE(cv) == SVt_PVCV)
+ break;
+ /* FALL THROUGH */
+ case SVt_PVHV:
+ case SVt_PVAV:
+ DIE("Not a subroutine reference");
+ case SVt_PVCV:
+ cv = (CV*)sv;
+ break;
+ case SVt_PVGV:
+ if (!(cv = GvCV((GV*)sv)))
+ cv = sv_2cv(sv, &stash, &gv, TRUE);
+ break;
+ }
+
ENTER;
SAVETMPS;
- if (!(cv && (CvROOT(cv) || CvUSERSUB(cv)))) {
- if (gv) {
- SV *tmpstr = sv_mortalcopy(&sv_undef);
+ retry:
+ if (!cv)
+ DIE("Not a subroutine reference");
+
+ if (!CvROOT(cv) && !CvUSERSUB(cv)) {
+ if (gv = CvGV(cv)) {
+ SV *tmpstr = sv_newmortal();
+ GV *ngv;
gv_efullname(tmpstr, gv);
- DIE("Undefined subroutine \"%s\" called",SvPVX(tmpstr));
+ ngv = gv_fetchmethod(GvESTASH(gv), "AUTOLOAD");
+ if (ngv && ngv != gv && (cv = GvCV(ngv))) { /* One more chance... */
+ gv = ngv;
+ sv_setsv(GvSV(gv), tmpstr);
+ goto retry;
+ }
+ else
+ DIE("Undefined subroutine &%s called",SvPVX(tmpstr));
}
- if (cv)
- DIE("Undefined subroutine called");
- DIE("Not a subroutine reference");
+ DIE("Undefined subroutine called");
}
+
if ((op->op_private & OPpSUBR_DB) && !CvUSERSUB(cv)) {
sv = GvSV(DBsub);
save_item(sv);
+ gv = CvGV(cv);
gv_efullname(sv,gv);
cv = GvCV(DBsub);
if (!cv)
@@ -5475,15 +5541,10 @@ PP(pp_entersubr)
push_return(op->op_next);
PUSHBLOCK(cx, CXt_SUB, MARK - 1);
PUSHSUB(cx);
- if (hasargs) {
- cx->blk_sub.savearray = GvAV(defgv);
- cx->blk_sub.argarray = av_fake(items, ++MARK);
- GvAV(defgv) = cx->blk_sub.argarray;
- }
CvDEPTH(cv)++;
if (CvDEPTH(cv) >= 2) { /* save temporaries on recursion? */
if (CvDEPTH(cv) == 100 && dowarn)
- warn("Deep recursion on subroutine \"%s\"",GvENAME(gv));
+ warn("Deep recursion on subroutine \"%s\"",GvENAME(CvGV(cv)));
if (CvDEPTH(cv) > AvFILL(padlist)) {
AV *newpad = newAV();
I32 ix = AvFILL((AV*)svp[1]);
@@ -5501,6 +5562,13 @@ PP(pp_entersubr)
else
av_store(newpad, ix--, NEWSV(0,0));
}
+ if (hasargs) {
+ AV* av = newAV();
+ av_store(av, 0, Nullsv);
+ av_store(newpad, 0, (SV*)av);
+ SvOK_on(av);
+ AvREAL_off(av);
+ }
av_store(padlist, CvDEPTH(cv), (SV*)newpad);
AvFILL(padlist) = CvDEPTH(cv);
svp = AvARRAY(padlist);
@@ -5508,6 +5576,36 @@ PP(pp_entersubr)
}
SAVESPTR(curpad);
curpad = AvARRAY((AV*)svp[CvDEPTH(cv)]);
+ if (hasargs) {
+ AV* av = (AV*)curpad[0];
+ SV** ary;
+
+ cx->blk_sub.savearray = GvAV(defgv);
+ cx->blk_sub.argarray = av;
+ GvAV(defgv) = cx->blk_sub.argarray;
+ ++MARK;
+
+ if (items >= AvMAX(av)) {
+ ary = AvALLOC(av);
+ if (AvARRAY(av) != ary) {
+ AvMAX(av) += AvARRAY(av) - AvALLOC(av);
+ SvPVX(av) = (char*)ary;
+ }
+ if (items >= AvMAX(av)) {
+ AvMAX(av) = items - 1;
+ Renew(ary,items+1,SV*);
+ AvALLOC(av) = ary;
+ SvPVX(av) = (char*)ary;
+ }
+ }
+ Copy(MARK,AvARRAY(av),items,SV*);
+ AvFILL(av) = items - 1;
+ while (items--) {
+ if (*MARK)
+ SvTEMP_off(*MARK);
+ MARK++;
+ }
+ }
RETURNOP(CvSTART(cv));
}
}
@@ -5526,7 +5624,10 @@ PP(pp_leavesubr)
if (gimme == G_SCALAR) {
MARK = newsp + 1;
if (MARK <= SP)
- *MARK = sv_mortalcopy(TOPs);
+ if (SvFLAGS(TOPs) & (SVs_PADTMP|SVs_TEMP))
+ *MARK = TOPs;
+ else
+ *MARK = sv_mortalcopy(TOPs);
else {
MEXTEND(mark,0);
*MARK = &sv_undef;
@@ -5535,7 +5636,8 @@ PP(pp_leavesubr)
}
else {
for (mark = newsp + 1; mark <= SP; mark++)
- *mark = sv_mortalcopy(*mark);
+ if (!(SvFLAGS(*mark) & (SVs_PADTMP|SVs_TEMP)))
+ *mark = sv_mortalcopy(*mark);
/* in case LEAVE wipes old return values */
}
@@ -5576,11 +5678,6 @@ PP(pp_caller)
cxix = nextcxix;
}
cx = &cxstack[cxix];
- if (cx->blk_oldcop == &compiling) {
- if (GIMME != G_ARRAY)
- RETPUSHUNDEF;
- RETURN;
- }
if (GIMME != G_ARRAY) {
dTARGET;
@@ -5591,19 +5688,29 @@ PP(pp_caller)
PUSHs(sv_2mortal(newSVpv(HvNAME(cx->blk_oldcop->cop_stash), 0)));
PUSHs(sv_2mortal(newSVpv(SvPVX(GvSV(cx->blk_oldcop->cop_filegv)), 0)));
- PUSHs(sv_2mortal(newSVnv((double)cx->blk_oldcop->cop_line)));
+ PUSHs(sv_2mortal(newSViv((I32)cx->blk_oldcop->cop_line)));
if (!MAXARG)
RETURN;
- sv = NEWSV(49, 0);
- gv_efullname(sv, cx->blk_sub.gv);
- PUSHs(sv_2mortal(sv));
- PUSHs(sv_2mortal(newSVnv((double)cx->blk_sub.hasargs)));
- PUSHs(sv_2mortal(newSVnv((double)cx->blk_gimme)));
- if (cx->blk_sub.hasargs) {
+ if (cx->cx_type == CXt_SUB) {
+ sv = NEWSV(49, 0);
+ gv_efullname(sv, CvGV(cx->blk_sub.cv));
+ PUSHs(sv_2mortal(sv));
+ PUSHs(sv_2mortal(newSViv((I32)cx->blk_sub.hasargs)));
+ }
+ else {
+ PUSHs(sv_2mortal(newSVpv("(eval)",0)));
+ PUSHs(sv_2mortal(newSViv(0)));
+ }
+ PUSHs(sv_2mortal(newSViv((I32)cx->blk_gimme)));
+ if (cx->blk_sub.hasargs && curstash == debstash) {
AV *ary = cx->blk_sub.argarray;
- if (!dbargs)
- dbargs = GvAV(gv_AVadd(gv_fetchpv("DB'args", TRUE)));
+ if (!dbargs) {
+ GV* tmpgv;
+ dbargs = GvAV(gv_AVadd(tmpgv = gv_fetchpv("DB::args", TRUE)));
+ SvMULTI_on(tmpgv);
+ AvREAL_off(dbargs);
+ }
if (AvMAX(dbargs) < AvFILL(ary))
av_store(dbargs, AvFILL(ary), Nullsv);
Copy(AvARRAY(ary), AvARRAY(dbargs), AvFILL(ary)+1, SV*);
@@ -5732,7 +5839,7 @@ PP(pp_nextstate)
curcop = (COP*)op;
TAINT_NOT; /* Each statement is presumed innocent */
stack_sp = stack_base + cxstack[cxstack_ix].blk_oldsp;
- free_tmps();
+ FREE_TMPS();
return NORMAL;
}
@@ -5741,7 +5848,7 @@ PP(pp_dbstate)
curcop = (COP*)op;
TAINT_NOT; /* Each statement is presumed innocent */
stack_sp = stack_base + cxstack[cxstack_ix].blk_oldsp;
- free_tmps();
+ FREE_TMPS();
if (op->op_private || SvIV(DBsingle) || SvIV(DBsignal) || SvIV(DBtrace))
{
@@ -5755,6 +5862,8 @@ PP(pp_dbstate)
ENTER;
SAVETMPS;
+ SAVEI32(debug);
+ debug = 0;
hasargs = 0;
gv = DBgv;
cv = GvCV(gv);
@@ -5762,14 +5871,14 @@ PP(pp_dbstate)
*++sp = Nullsv;
if (!cv)
- DIE("No DB'DB routine defined");
+ DIE("No DB::DB routine defined");
+ if (CvDEPTH(cv) >= 1) /* don't do recursive DB::DB call */
+ return NORMAL;
push_return(op->op_next);
PUSHBLOCK(cx, CXt_SUB, sp - 1);
PUSHSUB(cx);
CvDEPTH(cv)++;
- if (CvDEPTH(cv) >= 2)
- DIE("DB'DB called recursively");
SAVESPTR(curpad);
curpad = AvARRAY((AV*)*av_fetch(CvPADLIST(cv),1,FALSE));
RETURNOP(CvSTART(cv));
@@ -5783,10 +5892,9 @@ PP(pp_unstack)
I32 oldsave;
TAINT_NOT; /* Each statement is presumed innocent */
stack_sp = stack_base + cxstack[cxstack_ix].blk_oldsp;
- free_tmps();
+ FREE_TMPS();
oldsave = scopestack[scopestack_ix - 1];
- if (savestack_ix > oldsave)
- leave_scope(oldsave);
+ LEAVE_SCOPE(oldsave);
return NORMAL;
}
@@ -5798,7 +5906,7 @@ PP(pp_enter)
ENTER;
SAVETMPS;
- PUSHBLOCK(cx,CXt_BLOCK,sp);
+ PUSHBLOCK(cx, CXt_BLOCK, sp);
RETURN;
}
@@ -5807,10 +5915,32 @@ PP(pp_leave)
{
dSP;
register CONTEXT *cx;
- I32 gimme;
+ register SV **mark;
SV **newsp;
+ I32 gimme;
POPBLOCK(cx);
+
+ if (GIMME == G_SCALAR) {
+ MARK = newsp + 1;
+ if (MARK <= SP)
+ if (SvFLAGS(TOPs) & (SVs_PADTMP|SVs_TEMP))
+ *MARK = TOPs;
+ else
+ *MARK = sv_mortalcopy(TOPs);
+ else {
+ MEXTEND(mark,0);
+ *MARK = &sv_undef;
+ }
+ SP = MARK;
+ }
+ else {
+ for (mark = newsp + 1; mark <= SP; mark++)
+ if (!(SvFLAGS(*mark) & (SVs_PADTMP|SVs_TEMP)))
+ *mark = sv_mortalcopy(*mark);
+ /* in case LEAVE wipes old return values */
+ }
+
LEAVE;
RETURN;
@@ -5832,7 +5962,7 @@ PP(pp_enteriter)
SAVETMPS;
ENTER;
- PUSHBLOCK(cx,CXt_LOOP,SP);
+ PUSHBLOCK(cx, CXt_LOOP, SP);
PUSHLOOP(cx, svp, MARK);
cx->blk_loop.iterary = stack;
cx->blk_loop.iterix = MARK - stack_base;
@@ -5854,9 +5984,12 @@ PP(pp_iter)
if (cx->blk_loop.iterix >= cx->blk_oldsp)
RETPUSHNO;
- sv = AvARRAY(cx->blk_loop.iterary)[++cx->blk_loop.iterix];
- SvTEMP_off(sv);
- *cx->blk_loop.itervar = sv ? sv : &sv_undef;
+ if (sv = AvARRAY(cx->blk_loop.iterary)[++cx->blk_loop.iterix]) {
+ SvTEMP_off(sv);
+ *cx->blk_loop.itervar = sv;
+ }
+ else
+ *cx->blk_loop.itervar = &sv_undef;
RETPUSHYES;
}
@@ -6041,8 +6174,7 @@ PP(pp_next)
TOPBLOCK(cx);
oldsave = scopestack[scopestack_ix - 1];
- if (savestack_ix > oldsave)
- leave_scope(oldsave);
+ LEAVE_SCOPE(oldsave);
return cx->blk_loop.next_op;
}
@@ -6068,8 +6200,7 @@ PP(pp_redo)
TOPBLOCK(cx);
oldsave = scopestack[scopestack_ix - 1];
- if (savestack_ix > oldsave)
- leave_scope(oldsave);
+ LEAVE_SCOPE(oldsave);
return cx->blk_loop.redo_op;
}
@@ -6093,15 +6224,17 @@ OP **opstack;
if (op->op_flags & OPf_KIDS) {
/* First try all the kids at this level, since that's likeliest. */
for (kid = cUNOP->op_first; kid; kid = kid->op_sibling) {
- if (kid->op_type == OP_NEXTSTATE && kCOP->cop_label &&
- strEQ(kCOP->cop_label, label))
+ if ((kid->op_type == OP_NEXTSTATE || kid->op_type == OP_DBSTATE) &&
+ kCOP->cop_label && strEQ(kCOP->cop_label, label))
return kid;
}
for (kid = cUNOP->op_first; kid; kid = kid->op_sibling) {
if (kid == lastgotoprobe)
continue;
- if (kid->op_type == OP_NEXTSTATE) {
- if (ops > opstack && ops[-1]->op_type == OP_NEXTSTATE)
+ if (kid->op_type == OP_NEXTSTATE || kid->op_type == OP_DBSTATE) {
+ if (ops > opstack &&
+ (ops[-1]->op_type == OP_NEXTSTATE ||
+ ops[-1]->op_type == OP_DBSTATE))
*ops = kid;
else
*ops++ = kid;
@@ -6131,7 +6264,126 @@ PP(pp_goto)
char *label;
label = 0;
- if (op->op_flags & OPf_SPECIAL) {
+ if (op->op_flags & OPf_STACKED) {
+ SV *sv = POPs;
+
+ /* This egregious kludge implements goto &subroutine */
+ if (SvROK(sv) && SvTYPE(SvRV(sv)) == SVt_PVCV) {
+ I32 cxix;
+ register CONTEXT *cx;
+ CV* cv = (CV*)SvRV(sv);
+ SV** mark;
+ I32 items = 0;
+ I32 oldsave;
+
+ /* First do some returnish stuff. */
+ cxix = dopoptosub(cxstack_ix);
+ if (cxix < 0)
+ DIE("Can't goto subroutine outside a subroutine");
+ if (cxix < cxstack_ix)
+ dounwind(cxix);
+ TOPBLOCK(cx);
+ mark = ++stack_sp;
+ *stack_sp = (SV*)cv;
+ if (cx->blk_sub.hasargs) { /* put @_ back onto stack */
+ items = AvFILL(cx->blk_sub.argarray) + 1;
+ Copy(AvARRAY(cx->blk_sub.argarray), ++stack_sp, items, SV*);
+ stack_sp += items;
+ GvAV(defgv) = cx->blk_sub.savearray;
+ }
+ if (!(CvDEPTH(cx->blk_sub.cv) = cx->blk_sub.olddepth)) {
+ if (CvDELETED(cx->blk_sub.cv))
+ SvREFCNT_dec(cx->blk_sub.cv);
+ }
+ oldsave = scopestack[scopestack_ix - 1];
+ LEAVE_SCOPE(oldsave);
+
+ /* Now do some callish stuff. */
+ if (CvUSERSUB(cv)) {
+ items = (*CvUSERSUB(cv))(CvUSERINDEX(cv),
+ mark - stack_base, items);
+ sp = stack_base + items;
+ LEAVE;
+ return pop_return();
+ }
+ else {
+ AV* padlist = CvPADLIST(cv);
+ SV** svp = AvARRAY(padlist);
+ cx->blk_sub.cv = cv;
+ cx->blk_sub.olddepth = CvDEPTH(cv);
+ CvDEPTH(cv)++;
+ if (CvDEPTH(cv) >= 2) { /* save temporaries on recursion? */
+ if (CvDEPTH(cv) == 100 && dowarn)
+ warn("Deep recursion on subroutine \"%s\"",
+ GvENAME(CvGV(cv)));
+ if (CvDEPTH(cv) > AvFILL(padlist)) {
+ AV *newpad = newAV();
+ I32 ix = AvFILL((AV*)svp[1]);
+ svp = AvARRAY(svp[0]);
+ while (ix > 0) {
+ if (svp[ix]) {
+ char *name = SvPVX(svp[ix]); /* XXX */
+ if (*name == '@')
+ av_store(newpad, ix--, (SV*)newAV());
+ else if (*name == '%')
+ av_store(newpad, ix--, (SV*)newHV());
+ else
+ av_store(newpad, ix--, NEWSV(0,0));
+ }
+ else
+ av_store(newpad, ix--, NEWSV(0,0));
+ }
+ if (cx->blk_sub.hasargs) {
+ AV* av = newAV();
+ av_store(av, 0, Nullsv);
+ av_store(newpad, 0, (SV*)av);
+ SvOK_on(av);
+ AvREAL_off(av);
+ }
+ av_store(padlist, CvDEPTH(cv), (SV*)newpad);
+ AvFILL(padlist) = CvDEPTH(cv);
+ svp = AvARRAY(padlist);
+ }
+ }
+ SAVESPTR(curpad);
+ curpad = AvARRAY((AV*)svp[CvDEPTH(cv)]);
+ if (cx->blk_sub.hasargs) {
+ AV* av = (AV*)curpad[0];
+ SV** ary;
+
+ cx->blk_sub.savearray = GvAV(defgv);
+ cx->blk_sub.argarray = av;
+ GvAV(defgv) = cx->blk_sub.argarray;
+ ++mark;
+
+ if (items >= AvMAX(av)) {
+ ary = AvALLOC(av);
+ if (AvARRAY(av) != ary) {
+ AvMAX(av) += AvARRAY(av) - AvALLOC(av);
+ SvPVX(av) = (char*)ary;
+ }
+ if (items >= AvMAX(av)) {
+ AvMAX(av) = items - 1;
+ Renew(ary,items+1,SV*);
+ AvALLOC(av) = ary;
+ SvPVX(av) = (char*)ary;
+ }
+ }
+ Copy(mark,AvARRAY(av),items,SV*);
+ AvFILL(av) = items - 1;
+ while (items--) {
+ if (*mark)
+ SvTEMP_off(*mark);
+ mark++;
+ }
+ }
+ RETURNOP(CvSTART(cv));
+ }
+ }
+ else
+ label = SvPV(sv,na);
+ }
+ else if (op->op_flags & OPf_SPECIAL) {
if (op->op_type != OP_DUMP)
DIE("goto must have label");
}
@@ -6190,8 +6442,7 @@ PP(pp_goto)
dounwind(ix);
TOPBLOCK(cx);
oldsave = scopestack[scopestack_ix - 1];
- if (savestack_ix > oldsave)
- leave_scope(oldsave);
+ LEAVE_SCOPE(oldsave);
}
/* push wanted frames */
@@ -6288,7 +6539,7 @@ PP(pp_open)
gv = (GV*)POPs;
tmps = SvPV(sv, len);
if (do_open(gv, tmps, len)) {
- GvIO(gv)->lines = 0;
+ IoLINES(GvIO(gv)) = 0;
PUSHi( (I32)forkprocess );
}
else if (forkprocess == 0) /* we are a new child */
@@ -6331,24 +6582,24 @@ PP(pp_pipe_op)
rstio = GvIOn(rgv);
wstio = GvIOn(wgv);
- if (rstio->ifp)
+ if (IoIFP(rstio))
do_close(rgv, FALSE);
- if (wstio->ifp)
+ if (IoIFP(wstio))
do_close(wgv, FALSE);
if (pipe(fd) < 0)
goto badexit;
- rstio->ifp = fdopen(fd[0], "r");
- wstio->ofp = fdopen(fd[1], "w");
- wstio->ifp = wstio->ofp;
- rstio->type = '<';
- wstio->type = '>';
+ IoIFP(rstio) = fdopen(fd[0], "r");
+ IoOFP(wstio) = fdopen(fd[1], "w");
+ IoIFP(wstio) = IoOFP(wstio);
+ IoTYPE(rstio) = '<';
+ IoTYPE(wstio) = '>';
- if (!rstio->ifp || !wstio->ofp) {
- if (rstio->ifp) fclose(rstio->ifp);
+ if (!IoIFP(rstio) || !IoOFP(wstio)) {
+ if (IoIFP(rstio)) fclose(IoIFP(rstio));
else close(fd[0]);
- if (wstio->ofp) fclose(wstio->ofp);
+ if (IoOFP(wstio)) fclose(IoOFP(wstio));
else close(fd[1]);
goto badexit;
}
@@ -6371,7 +6622,7 @@ PP(pp_fileno)
if (MAXARG < 1)
RETPUSHUNDEF;
gv = (GV*)POPs;
- if (!gv || !(io = GvIO(gv)) || !(fp = io->ifp))
+ if (!gv || !(io = GvIO(gv)) || !(fp = IoIFP(io)))
RETPUSHUNDEF;
PUSHi(fileno(fp));
RETURN;
@@ -6410,7 +6661,7 @@ PP(pp_binmode)
gv = (GV*)POPs;
EXTEND(SP, 1);
- if (!gv || !(io = GvIO(gv)) || !(fp = io->ifp))
+ if (!gv || !(io = GvIO(gv)) || !(fp = IoIFP(io)))
RETSETUNDEF;
#ifdef DOSISH
@@ -6692,7 +6943,7 @@ PP(pp_getc)
RETPUSHUNDEF;
TAINT_IF(1);
sv_setpv(TARG, " ");
- *SvPVX(TARG) = getc(GvIO(gv)->ifp); /* should never be EOF */
+ *SvPVX(TARG) = getc(IoIFP(GvIO(gv))); /* should never be EOF */
PUSHTARG;
RETURN;
}
@@ -6742,8 +6993,8 @@ PP(pp_enterwrite)
RETPUSHNO;
}
curoutgv = gv;
- if (io->fmt_gv)
- fgv = io->fmt_gv;
+ if (IoFMT_GV(io))
+ fgv = IoFMT_GV(io);
else
fgv = gv;
@@ -6751,7 +7002,7 @@ PP(pp_enterwrite)
if (!cv) {
if (fgv) {
- SV *tmpstr = sv_mortalcopy(&sv_undef);
+ SV *tmpstr = sv_newmortal();
gv_efullname(tmpstr, gv);
DIE("Undefined format \"%s\" called",SvPVX(tmpstr));
}
@@ -6766,7 +7017,7 @@ PP(pp_leavewrite)
dSP;
GV *gv = cxstack[cxstack_ix].blk_sub.gv;
register IO *io = GvIO(gv);
- FILE *ofp = io->ofp;
+ FILE *ofp = IoOFP(io);
FILE *fp;
SV **mark;
SV **newsp;
@@ -6774,37 +7025,37 @@ PP(pp_leavewrite)
register CONTEXT *cx;
DEBUG_f(fprintf(stderr,"left=%ld, todo=%ld\n",
- (long)io->lines_left, (long)FmLINES(formtarget)));
- if (io->lines_left < FmLINES(formtarget) &&
+ (long)IoLINES_LEFT(io), (long)FmLINES(formtarget)));
+ if (IoLINES_LEFT(io) < FmLINES(formtarget) &&
formtarget != toptarget)
{
- if (!io->top_gv) {
+ if (!IoTOP_GV(io)) {
GV *topgv;
char tmpbuf[256];
- if (!io->top_name) {
- if (!io->fmt_name)
- io->fmt_name = savestr(GvNAME(gv));
- sprintf(tmpbuf, "%s_TOP", io->fmt_name);
+ if (!IoTOP_NAME(io)) {
+ if (!IoFMT_NAME(io))
+ IoFMT_NAME(io) = savestr(GvNAME(gv));
+ sprintf(tmpbuf, "%s_TOP", IoFMT_NAME(io));
topgv = gv_fetchpv(tmpbuf,FALSE);
if (topgv && GvFORM(topgv))
- io->top_name = savestr(tmpbuf);
+ IoTOP_NAME(io) = savestr(tmpbuf);
else
- io->top_name = savestr("top");
+ IoTOP_NAME(io) = savestr("top");
}
- topgv = gv_fetchpv(io->top_name,FALSE);
+ topgv = gv_fetchpv(IoTOP_NAME(io),FALSE);
if (!topgv || !GvFORM(topgv)) {
- io->lines_left = 100000000;
+ IoLINES_LEFT(io) = 100000000;
goto forget_top;
}
- io->top_gv = topgv;
+ IoTOP_GV(io) = topgv;
}
- if (io->lines_left >= 0 && io->page > 0)
+ if (IoLINES_LEFT(io) >= 0 && IoPAGE(io) > 0)
fwrite(SvPVX(formfeed), SvCUR(formfeed), 1, ofp);
- io->lines_left = io->page_len;
- io->page++;
+ IoLINES_LEFT(io) = IoPAGE_LEN(io);
+ IoPAGE(io)++;
formtarget = toptarget;
- return doform(GvFORM(io->top_gv),gv,op);
+ return doform(GvFORM(IoTOP_GV(io)),gv,op);
}
forget_top:
@@ -6812,10 +7063,10 @@ PP(pp_leavewrite)
POPFORMAT(cx);
LEAVE;
- fp = io->ofp;
+ fp = IoOFP(io);
if (!fp) {
if (dowarn) {
- if (io->ifp)
+ if (IoIFP(io))
warn("Filehandle only opened for input");
else
warn("Write on closed filehandle");
@@ -6823,7 +7074,7 @@ PP(pp_leavewrite)
PUSHs(&sv_no);
}
else {
- if ((io->lines_left -= FmLINES(formtarget)) < 0) {
+ if ((IoLINES_LEFT(io) -= FmLINES(formtarget)) < 0) {
if (dowarn)
warn("page overflow");
}
@@ -6833,7 +7084,7 @@ PP(pp_leavewrite)
else {
FmLINES(formtarget) = 0;
SvCUR_set(formtarget, 0);
- if (io->flags & IOf_FLUSH)
+ if (IoFLAGS(io) & IOf_FLUSH)
(void)fflush(fp);
PUSHs(&sv_yes);
}
@@ -6861,9 +7112,9 @@ PP(pp_prtf)
errno = EBADF;
goto just_say_no;
}
- else if (!(fp = io->ofp)) {
+ else if (!(fp = IoOFP(io))) {
if (dowarn) {
- if (io->ifp)
+ if (IoIFP(io))
warn("Filehandle opened only for input");
else
warn("printf on closed filehandle");
@@ -6876,17 +7127,17 @@ PP(pp_prtf)
if (!do_print(sv, fp))
goto just_say_no;
- if (io->flags & IOf_FLUSH)
+ if (IoFLAGS(io) & IOf_FLUSH)
if (fflush(fp) == EOF)
goto just_say_no;
}
- sv_free(sv);
+ SvREFCNT_dec(sv);
SP = ORIGMARK;
PUSHs(&sv_yes);
RETURN;
just_say_no:
- sv_free(sv);
+ SvREFCNT_dec(sv);
SP = ORIGMARK;
PUSHs(&sv_undef);
RETURN;
@@ -6909,9 +7160,9 @@ PP(pp_print)
errno = EBADF;
goto just_say_no;
}
- else if (!(fp = io->ofp)) {
+ else if (!(fp = IoOFP(io))) {
if (dowarn) {
- if (io->ifp)
+ if (IoIFP(io))
warn("Filehandle opened only for input");
else
warn("print on closed filehandle");
@@ -6948,7 +7199,7 @@ PP(pp_print)
if (fwrite(ors, 1, orslen, fp) == 0 || ferror(fp))
goto just_say_no;
- if (io->flags & IOf_FLUSH)
+ if (IoFLAGS(io) & IOf_FLUSH)
if (fflush(fp) == EOF)
goto just_say_no;
}
@@ -6982,7 +7233,7 @@ PP(pp_sysread)
buffer = SvPV(bufstr, blen);
length = SvIVx(*++MARK);
if (SvTHINKFIRST(bufstr)) {
- if (SvREADONLY(bufstr))
+ if (SvREADONLY(bufstr) && curcop != &compiling)
DIE(no_modify);
if (SvROK(bufstr))
sv_unref(bufstr);
@@ -6995,13 +7246,13 @@ PP(pp_sysread)
if (MARK < SP)
warn("Too many args on read");
io = GvIO(gv);
- if (!io || !io->ifp)
+ if (!io || !IoIFP(io))
goto say_undef;
#ifdef HAS_SOCKET
if (op->op_type == OP_RECV) {
bufsize = sizeof buf;
SvGROW(bufstr, length+1), (buffer = SvPV(bufstr, blen)); /* sneaky */
- length = recvfrom(fileno(io->ifp), buffer, length, offset,
+ length = recvfrom(fileno(IoIFP(io)), buffer, length, offset,
buf, &bufsize);
if (length < 0)
RETPUSHUNDEF;
@@ -7019,18 +7270,18 @@ PP(pp_sysread)
#endif
SvGROW(bufstr, length+offset+1), (buffer = SvPV(bufstr, blen)); /* sneaky */
if (op->op_type == OP_SYSREAD) {
- length = read(fileno(io->ifp), buffer+offset, length);
+ length = read(fileno(IoIFP(io)), buffer+offset, length);
}
else
#ifdef HAS_SOCKET
- if (io->type == 's') {
+ if (IoTYPE(io) == 's') {
bufsize = sizeof buf;
- length = recvfrom(fileno(io->ifp), buffer+offset, length, 0,
+ length = recvfrom(fileno(IoIFP(io)), buffer+offset, length, 0,
buf, &bufsize);
}
else
#endif
- length = fread(buffer+offset, 1, length, io->ifp);
+ length = fread(buffer+offset, 1, length, IoIFP(io));
if (length < 0)
goto say_undef;
SvCUR_set(bufstr, length+offset);
@@ -7069,7 +7320,7 @@ PP(pp_send)
length = SvIVx(*++MARK);
errno = 0;
io = GvIO(gv);
- if (!io || !io->ifp) {
+ if (!io || !IoIFP(io)) {
length = -1;
if (dowarn) {
if (op->op_type == OP_SYSWRITE)
@@ -7085,7 +7336,7 @@ PP(pp_send)
offset = 0;
if (MARK < SP)
warn("Too many args on syswrite");
- length = write(fileno(io->ifp), buffer+offset, length);
+ length = write(fileno(IoIFP(io)), buffer+offset, length);
}
#ifdef HAS_SOCKET
else if (SP >= MARK) {
@@ -7093,10 +7344,10 @@ PP(pp_send)
if (SP > MARK)
warn("Too many args on send");
buffer = SvPVx(*++MARK, mlen);
- length = sendto(fileno(io->ifp), buffer, blen, length, buffer, mlen);
+ length = sendto(fileno(IoIFP(io)), buffer, blen, length, buffer, mlen);
}
else
- length = send(fileno(io->ifp), buffer, blen, length);
+ length = send(fileno(IoIFP(io)), buffer, blen, length);
#else
else
DIE(no_sock_func, "send");
@@ -7126,7 +7377,7 @@ PP(pp_eof)
gv = last_in_gv;
else
gv = (GV*)POPs;
- PUSHs(do_eof(gv) ? &sv_yes : &sv_no);
+ PUSHs(!gv || do_eof(gv) ? &sv_yes : &sv_no);
RETURN;
}
@@ -7167,8 +7418,8 @@ PP(pp_truncate)
#ifdef HAS_TRUNCATE
if (op->op_flags & OPf_SPECIAL) {
tmpgv = gv_fetchpv(POPp,FALSE);
- if (!tmpgv || !GvIO(tmpgv) || !GvIO(tmpgv)->ifp ||
- ftruncate(fileno(GvIO(tmpgv)->ifp), len) < 0)
+ if (!tmpgv || !GvIO(tmpgv) || !IoIFP(GvIO(tmpgv)) ||
+ ftruncate(fileno(IoIFP(GvIO(tmpgv))), len) < 0)
result = 0;
}
else if (truncate(POPp, len) < 0)
@@ -7176,8 +7427,8 @@ PP(pp_truncate)
#else
if (op->op_flags & OPf_SPECIAL) {
tmpgv = gv_fetchpv(POPp,FALSE);
- if (!tmpgv || !GvIO(tmpgv) || !GvIO(tmpgv)->ifp ||
- chsize(fileno(GvIO(tmpgv)->ifp), len) < 0)
+ if (!tmpgv || !GvIO(tmpgv) || !IoIFP(GvIO(tmpgv)) ||
+ chsize(fileno(IoIFP(GvIO(tmpgv))), len) < 0)
result = 0;
}
else {
@@ -7219,13 +7470,13 @@ PP(pp_ioctl)
GV *gv = (GV*)POPs;
IO *io = GvIOn(gv);
- if (!io || !argstr || !io->ifp) {
+ if (!io || !argstr || !IoIFP(io)) {
errno = EBADF; /* well, sort of... */
RETPUSHUNDEF;
}
if (SvPOK(argstr) || !SvNIOK(argstr)) {
- STRLEN len;
+ STRLEN len = 0;
if (!SvPOK(argstr))
s = SvPV(argstr, len);
retval = IOCPARM_LEN(func);
@@ -7249,13 +7500,13 @@ PP(pp_ioctl)
TAINT_PROPER(optype == OP_IOCTL ? "ioctl" : "fcntl");
if (optype == OP_IOCTL)
- retval = ioctl(fileno(io->ifp), func, s);
+ retval = ioctl(fileno(IoIFP(io)), func, s);
else
#ifdef DOSISH
DIE("fcntl is not implemented");
#else
# ifdef HAS_FCNTL
- retval = fcntl(fileno(io->ifp), func, s);
+ retval = fcntl(fileno(IoIFP(io)), func, s);
# else
DIE("fcntl is not implemented");
# endif
@@ -7293,7 +7544,7 @@ PP(pp_flock)
else
gv = (GV*)POPs;
if (gv && GvIO(gv))
- fp = GvIO(gv)->ifp;
+ fp = IoIFP(GvIO(gv));
else
fp = Nullfp;
if (fp) {
@@ -7329,20 +7580,20 @@ PP(pp_socket)
}
io = GvIOn(gv);
- if (io->ifp)
+ if (IoIFP(io))
do_close(gv, FALSE);
TAINT_PROPER("socket");
fd = socket(domain, type, protocol);
if (fd < 0)
RETPUSHUNDEF;
- io->ifp = fdopen(fd, "r"); /* stdio gets confused about sockets */
- io->ofp = fdopen(fd, "w");
- io->type = 's';
- if (!io->ifp || !io->ofp) {
- if (io->ifp) fclose(io->ifp);
- if (io->ofp) fclose(io->ofp);
- if (!io->ifp && !io->ofp) close(fd);
+ IoIFP(io) = fdopen(fd, "r"); /* stdio gets confused about sockets */
+ IoOFP(io) = fdopen(fd, "w");
+ IoTYPE(io) = 's';
+ if (!IoIFP(io) || !IoOFP(io)) {
+ if (IoIFP(io)) fclose(IoIFP(io));
+ if (IoOFP(io)) fclose(IoOFP(io));
+ if (!IoIFP(io) && !IoOFP(io)) close(fd);
RETPUSHUNDEF;
}
@@ -7372,27 +7623,27 @@ PP(pp_sockpair)
io1 = GvIOn(gv1);
io2 = GvIOn(gv2);
- if (io1->ifp)
+ if (IoIFP(io1))
do_close(gv1, FALSE);
- if (io2->ifp)
+ if (IoIFP(io2))
do_close(gv2, FALSE);
TAINT_PROPER("socketpair");
if (socketpair(domain, type, protocol, fd) < 0)
RETPUSHUNDEF;
- io1->ifp = fdopen(fd[0], "r");
- io1->ofp = fdopen(fd[0], "w");
- io1->type = 's';
- io2->ifp = fdopen(fd[1], "r");
- io2->ofp = fdopen(fd[1], "w");
- io2->type = 's';
- if (!io1->ifp || !io1->ofp || !io2->ifp || !io2->ofp) {
- if (io1->ifp) fclose(io1->ifp);
- if (io1->ofp) fclose(io1->ofp);
- if (!io1->ifp && !io1->ofp) close(fd[0]);
- if (io2->ifp) fclose(io2->ifp);
- if (io2->ofp) fclose(io2->ofp);
- if (!io2->ifp && !io2->ofp) close(fd[1]);
+ IoIFP(io1) = fdopen(fd[0], "r");
+ IoOFP(io1) = fdopen(fd[0], "w");
+ IoTYPE(io1) = 's';
+ IoIFP(io2) = fdopen(fd[1], "r");
+ IoOFP(io2) = fdopen(fd[1], "w");
+ IoTYPE(io2) = 's';
+ if (!IoIFP(io1) || !IoOFP(io1) || !IoIFP(io2) || !IoOFP(io2)) {
+ if (IoIFP(io1)) fclose(IoIFP(io1));
+ if (IoOFP(io1)) fclose(IoOFP(io1));
+ if (!IoIFP(io1) && !IoOFP(io1)) close(fd[0]);
+ if (IoIFP(io2)) fclose(IoIFP(io2));
+ if (IoOFP(io2)) fclose(IoOFP(io2));
+ if (!IoIFP(io2) && !IoOFP(io2)) close(fd[1]);
RETPUSHUNDEF;
}
@@ -7412,12 +7663,12 @@ PP(pp_bind)
register IO *io = GvIOn(gv);
STRLEN len;
- if (!io || !io->ifp)
+ if (!io || !IoIFP(io))
goto nuts;
addr = SvPV(addrstr, len);
TAINT_PROPER("bind");
- if (bind(fileno(io->ifp), addr, len) >= 0)
+ if (bind(fileno(IoIFP(io)), addr, len) >= 0)
RETPUSHYES;
else
RETPUSHUNDEF;
@@ -7442,12 +7693,12 @@ PP(pp_connect)
register IO *io = GvIOn(gv);
STRLEN len;
- if (!io || !io->ifp)
+ if (!io || !IoIFP(io))
goto nuts;
addr = SvPV(addrstr, len);
TAINT_PROPER("connect");
- if (connect(fileno(io->ifp), addr, len) >= 0)
+ if (connect(fileno(IoIFP(io)), addr, len) >= 0)
RETPUSHYES;
else
RETPUSHUNDEF;
@@ -7470,10 +7721,10 @@ PP(pp_listen)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->ifp)
+ if (!io || !IoIFP(io))
goto nuts;
- if (listen(fileno(io->ifp), backlog) >= 0)
+ if (listen(fileno(IoIFP(io)), backlog) >= 0)
RETPUSHYES;
else
RETPUSHUNDEF;
@@ -7508,23 +7759,23 @@ PP(pp_accept)
goto nuts;
gstio = GvIO(ggv);
- if (!gstio || !gstio->ifp)
+ if (!gstio || !IoIFP(gstio))
goto nuts;
nstio = GvIOn(ngv);
- if (nstio->ifp)
+ if (IoIFP(nstio))
do_close(ngv, FALSE);
- fd = accept(fileno(gstio->ifp), (struct sockaddr *)buf, &len);
+ fd = accept(fileno(IoIFP(gstio)), (struct sockaddr *)buf, &len);
if (fd < 0)
goto badexit;
- nstio->ifp = fdopen(fd, "r");
- nstio->ofp = fdopen(fd, "w");
- nstio->type = 's';
- if (!nstio->ifp || !nstio->ofp) {
- if (nstio->ifp) fclose(nstio->ifp);
- if (nstio->ofp) fclose(nstio->ofp);
- if (!nstio->ifp && !nstio->ofp) close(fd);
+ IoIFP(nstio) = fdopen(fd, "r");
+ IoOFP(nstio) = fdopen(fd, "w");
+ IoTYPE(nstio) = 's';
+ if (!IoIFP(nstio) || !IoOFP(nstio)) {
+ if (IoIFP(nstio)) fclose(IoIFP(nstio));
+ if (IoOFP(nstio)) fclose(IoOFP(nstio));
+ if (!IoIFP(nstio) && !IoOFP(nstio)) close(fd);
goto badexit;
}
@@ -7552,10 +7803,10 @@ PP(pp_shutdown)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->ifp)
+ if (!io || !IoIFP(io))
goto nuts;
- PUSHi( shutdown(fileno(io->ifp), how) >= 0 );
+ PUSHi( shutdown(fileno(IoIFP(io)), how) >= 0 );
RETURN;
nuts:
@@ -7598,10 +7849,10 @@ PP(pp_ssockopt)
gv = (GV*)POPs;
io = GvIOn(gv);
- if (!io || !io->ifp)
+ if (!io || !IoIFP(io))
goto nuts;
- fd = fileno(io->ifp);
+ fd = fileno(IoIFP(io));
switch (optype) {
case OP_GSOCKOPT:
SvCUR_set(sv, 256);
@@ -7649,13 +7900,13 @@ PP(pp_getpeername)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->ifp)
+ if (!io || !IoIFP(io))
goto nuts;
sv = sv_2mortal(NEWSV(22, 257));
SvCUR_set(sv, 256);
SvPOK_on(sv);
- fd = fileno(io->ifp);
+ fd = fileno(IoIFP(io));
switch (optype) {
case OP_GETSOCKNAME:
if (getsockname(fd, SvPVX(sv), (int*)&SvCUR(sv)) < 0)
@@ -7700,8 +7951,8 @@ PP(pp_stat)
laststype = OP_STAT;
statgv = tmpgv;
sv_setpv(statname, "");
- if (!GvIO(tmpgv) || !GvIO(tmpgv)->ifp ||
- fstat(fileno(GvIO(tmpgv)->ifp), &statcache) < 0) {
+ if (!GvIO(tmpgv) || !IoIFP(GvIO(tmpgv)) ||
+ fstat(fileno(IoIFP(GvIO(tmpgv))), &statcache) < 0) {
max = 0;
laststatval = -1;
}
@@ -7734,20 +7985,20 @@ PP(pp_stat)
RETPUSHUNDEF;
}
if (max) {
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_dev)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_ino)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_mode)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_nlink)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_uid)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_gid)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_rdev)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_size)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_atime)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_mtime)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_ctime)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_dev)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_ino)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_mode)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_nlink)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_uid)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_gid)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_rdev)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_size)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_atime)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_mtime)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_ctime)));
#ifdef STATBLOCKS
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_blksize)));
- PUSHs(sv_2mortal(newSVnv((double)statcache.st_blocks)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_blksize)));
+ PUSHs(sv_2mortal(newSViv((I32)statcache.st_blocks)));
#else
PUSHs(sv_2mortal(newSVpv("", 0)));
PUSHs(sv_2mortal(newSVpv("", 0)));
@@ -8029,8 +8280,8 @@ PP(pp_fttty)
}
else
gv = gv_fetchpv(tmps = POPp, FALSE);
- if (gv && GvIO(gv) && GvIO(gv)->ifp)
- fd = fileno(GvIO(gv)->ifp);
+ if (gv && GvIO(gv) && IoIFP(GvIO(gv)))
+ fd = fileno(IoIFP(GvIO(gv)));
else if (isDIGIT(*tmps))
fd = atoi(tmps);
else
@@ -8066,23 +8317,23 @@ PP(pp_fttext)
sv_setpv(statname, "");
io = GvIO(statgv);
}
- if (io && io->ifp) {
+ if (io && IoIFP(io)) {
#if defined(STDSTDIO) || defined(atarist) /* this will work with atariST */
- fstat(fileno(io->ifp), &statcache);
+ fstat(fileno(IoIFP(io)), &statcache);
if (S_ISDIR(statcache.st_mode)) /* handle NFS glitch */
if (op->op_type == OP_FTTEXT)
RETPUSHNO;
else
RETPUSHYES;
- if (io->ifp->_cnt <= 0) {
- i = getc(io->ifp);
+ if (IoIFP(io)->_cnt <= 0) {
+ i = getc(IoIFP(io));
if (i != EOF)
- (void)ungetc(i, io->ifp);
+ (void)ungetc(i, IoIFP(io));
}
- if (io->ifp->_cnt <= 0) /* null file is anything */
+ if (IoIFP(io)->_cnt <= 0) /* null file is anything */
RETPUSHYES;
- len = io->ifp->_cnt + (io->ifp->_ptr - io->ifp->_base);
- s = io->ifp->_base;
+ len = IoIFP(io)->_cnt + (IoIFP(io)->_ptr - IoIFP(io)->_base);
+ s = IoIFP(io)->_base;
#else
DIE("-T and -B not implemented on filehandles");
#endif
@@ -8423,9 +8674,9 @@ PP(pp_open_dir)
if (!io)
goto nope;
- if (io->dirp)
- closedir(io->dirp);
- if (!(io->dirp = opendir(dirname)))
+ if (IoDIRP(io))
+ closedir(IoDIRP(io));
+ if (!(IoDIRP(io) = opendir(dirname)))
goto nope;
RETPUSHYES;
@@ -8449,12 +8700,12 @@ PP(pp_readdir)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->dirp)
+ if (!io || !IoDIRP(io))
goto nope;
if (GIMME == G_ARRAY) {
/*SUPPRESS 560*/
- while (dp = readdir(io->dirp)) {
+ while (dp = readdir(IoDIRP(io))) {
#ifdef DIRNAMLEN
XPUSHs(sv_2mortal(newSVpv(dp->d_name, dp->d_namlen)));
#else
@@ -8463,7 +8714,7 @@ PP(pp_readdir)
}
}
else {
- if (!(dp = readdir(io->dirp)))
+ if (!(dp = readdir(IoDIRP(io))))
goto nope;
#ifdef DIRNAMLEN
XPUSHs(sv_2mortal(newSVpv(dp->d_name, dp->d_namlen)));
@@ -8495,10 +8746,10 @@ PP(pp_telldir)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->dirp)
+ if (!io || !IoDIRP(io))
goto nope;
- PUSHi( telldir(io->dirp) );
+ PUSHi( telldir(IoDIRP(io)) );
RETURN;
nope:
if (!errno)
@@ -8517,10 +8768,10 @@ PP(pp_seekdir)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->dirp)
+ if (!io || !IoDIRP(io))
goto nope;
- (void)seekdir(io->dirp, along);
+ (void)seekdir(IoDIRP(io), along);
RETPUSHYES;
nope:
@@ -8539,10 +8790,10 @@ PP(pp_rewinddir)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->dirp)
+ if (!io || !IoDIRP(io))
goto nope;
- (void)rewinddir(io->dirp);
+ (void)rewinddir(IoDIRP(io));
RETPUSHYES;
nope:
if (!errno)
@@ -8560,12 +8811,12 @@ PP(pp_closedir)
GV *gv = (GV*)POPs;
register IO *io = GvIOn(gv);
- if (!io || !io->dirp)
+ if (!io || !IoDIRP(io))
goto nope;
- if (closedir(io->dirp) < 0)
+ if (closedir(IoDIRP(io)) < 0)
goto nope;
- io->dirp = 0;
+ IoDIRP(io) = 0;
RETPUSHYES;
nope:
@@ -8909,15 +9160,15 @@ PP(pp_gmtime)
PUSHp(mybuf, strlen(mybuf));
}
else if (tmbuf) {
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_sec)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_min)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_hour)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_mday)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_mon)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_year)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_wday)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_yday)));
- PUSHs(sv_2mortal(newSVnv((double)tmbuf->tm_isdst)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_sec)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_min)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_hour)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_mday)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_mon)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_year)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_wday)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_yday)));
+ PUSHs(sv_2mortal(newSViv((I32)tmbuf->tm_isdst)));
}
RETURN;
}
@@ -9123,11 +9374,14 @@ doeval()
SAVEINT(padix);
SAVESPTR(curpad);
SAVESPTR(comppad);
- SAVESPTR(comppadname);
- SAVEINT(comppadnamefill);
+ SAVESPTR(comppad_name);
+ SAVEINT(comppad_name_fill);
+ SAVEINT(min_intro_pending);
+ SAVEINT(max_intro_pending);
comppad = newAV();
- comppadname = newAV();
- comppadnamefill = -1;
+ comppad_name = newAV();
+ comppad_name_fill = 0;
+ min_intro_pending = 0;
av_push(comppad, Nullsv);
curpad = AvARRAY(comppad);
padix = 0;
@@ -9151,23 +9405,22 @@ doeval()
rslen = 1;
rschar = '\n';
rspara = 0;
- lex_start();
if (yyparse() || error_count || !eval_root) {
SV **newsp;
I32 gimme;
CONTEXT *cx;
I32 optype;
- lex_end();
op = saveop;
- POPBLOCK(cx);
- POPEVAL(cx);
- pop_return();
- LEAVE;
if (eval_root) {
op_free(eval_root);
eval_root = Nullop;
}
+ POPBLOCK(cx);
+ POPEVAL(cx);
+ pop_return();
+ lex_end();
+ LEAVE;
if (optype == OP_REQUIRE)
DIE("%s", SvPVx(GvSV(gv_fetchpv("@",TRUE)), na));
rs = nrs;
@@ -9176,22 +9429,19 @@ doeval()
rspara = (nrslen == 2);
RETPUSHUNDEF;
}
- lex_end();
rs = nrs;
rslen = nrslen;
rschar = nrschar;
rspara = (nrslen == 2);
compiling.cop_line = 0;
+ SAVEFREESV(comppad_name);
+ SAVEFREESV(comppad);
+ SAVEFREEOP(eval_root);
DEBUG_x(dump_eval());
/* compiled okay, so do it */
- if (beginav) {
- calllist(beginav);
- av_free(beginav);
- beginav = 0;
- }
sv_setpv(GvSV(gv_fetchpv("@",TRUE)),"");
RETURNOP(eval_start);
}
@@ -9205,6 +9455,7 @@ PP(pp_require)
char *tmpname;
SV** svp;
I32 gimme = G_SCALAR;
+ FILE *tryrsfp = 0;
if (MAXARG < 1) {
sv = GvSV(defgv);
@@ -9212,6 +9463,12 @@ PP(pp_require)
}
else
sv = POPs;
+ if (SvNIOK(sv) && !SvPOKp(sv)) {
+ if (SvNV(sv) > atof(patchlevel) + 0.000999)
+ DIE("Perl %3.3f required--this is only version %s, stopped",
+ SvNV(sv),patchlevel);
+ RETPUSHYES;
+ }
name = SvPV(sv, na);
if (op->op_type == OP_REQUIRE &&
(svp = hv_fetch(GvHVn(incgv), name, SvCUR(sv), 0)) &&
@@ -9220,16 +9477,13 @@ PP(pp_require)
/* prepare to compile file */
- sv_setpv(linestr,"");
-
- SAVESPTR(rsfp); /* in case we're in a BEGIN */
tmpname = savestr(name);
if (*tmpname == '/' ||
(*tmpname == '.' &&
(tmpname[1] == '/' ||
(tmpname[1] == '.' && tmpname[2] == '/'))))
{
- rsfp = fopen(tmpname,"r");
+ tryrsfp = fopen(tmpname,"r");
}
else {
AV *ar = GvAVn(incgv);
@@ -9238,8 +9492,8 @@ PP(pp_require)
for (i = 0; i <= AvFILL(ar); i++) {
(void)sprintf(buf, "%s/%s",
SvPVx(*av_fetch(ar, i, TRUE), na), name);
- rsfp = fopen(buf, "r");
- if (rsfp) {
+ tryrsfp = fopen(buf, "r");
+ if (tryrsfp) {
char *s = buf;
if (*s == '.' && s[1] == '/')
@@ -9253,7 +9507,7 @@ PP(pp_require)
compiling.cop_filegv = gv_fetchfile(tmpname);
Safefree(tmpname);
tmpname = Nullch;
- if (!rsfp) {
+ if (!tryrsfp) {
if (op->op_type == OP_REQUIRE) {
sprintf(tokenbuf,"Can't locate %s in @INC", name);
if (instr(tokenbuf,".h "))
@@ -9268,15 +9522,17 @@ PP(pp_require)
ENTER;
SAVETMPS;
+ lex_start(sv_2mortal(newSVpv("",0)));
+ rsfp = tryrsfp;
+ name = savestr(name);
+ SAVEFREEPV(name);
/* switch to eval mode */
push_return(op->op_next);
- PUSHBLOCK(cx,CXt_EVAL,SP);
- PUSHEVAL(cx,savestr(name));
+ PUSHBLOCK(cx, CXt_EVAL, SP);
+ PUSHEVAL(cx, name, compiling.cop_filegv);
- if (curcop->cop_line == 0) /* don't debug debugger... */
- perldb = FALSE;
compiling.cop_line = 0;
PUTBACK;
@@ -9294,25 +9550,27 @@ PP(pp_entereval)
register CONTEXT *cx;
dPOPss;
I32 gimme = GIMME;
+ char tmpbuf[32];
ENTER;
SAVETMPS;
+ lex_start(sv);
/* switch to eval mode */
+ sprintf(tmpbuf, "_<(eval %d)", ++evalseq);
+ compiling.cop_filegv = gv_fetchfile(tmpbuf+2);
+ compiling.cop_line = 1;
+ SAVEDELETE(defstash, savestr(tmpbuf), strlen(tmpbuf));
+
push_return(op->op_next);
- PUSHBLOCK(cx,CXt_EVAL,SP);
- PUSHEVAL(cx,0);
+ PUSHBLOCK(cx, CXt_EVAL, SP);
+ PUSHEVAL(cx, 0, compiling.cop_filegv);
/* prepare to compile string */
- save_item(linestr);
- sv_setsv(linestr, sv);
- sv_catpv(linestr, "\n;");
- compiling.cop_filegv = gv_fetchfile("(eval)");
- compiling.cop_line = 1;
- if (perldb)
- save_lines(GvAV(curcop->cop_filegv), linestr);
+ if (perldb && curstash != debstash)
+ save_lines(GvAV(compiling.cop_filegv), linestr);
PUTBACK;
return doeval();
}
@@ -9334,8 +9592,12 @@ PP(pp_leaveeval)
if (gimme == G_SCALAR) {
MARK = newsp + 1;
- if (MARK <= SP)
- *MARK = sv_mortalcopy(TOPs);
+ if (MARK <= SP) {
+ if (SvFLAGS(TOPs) & SVs_TEMP)
+ *MARK = TOPs;
+ else
+ *MARK = sv_mortalcopy(TOPs);
+ }
else {
MEXTEND(mark,0);
*MARK = &sv_undef;
@@ -9344,7 +9606,8 @@ PP(pp_leaveeval)
}
else {
for (mark = newsp + 1; mark <= SP; mark++)
- *mark = sv_mortalcopy(*mark);
+ if (!(SvFLAGS(TOPs) & SVs_TEMP))
+ *mark = sv_mortalcopy(*mark);
/* in case LEAVE wipes old return values */
}
@@ -9357,12 +9620,9 @@ PP(pp_leaveeval)
}
else if (optype == OP_REQUIRE)
retop = die("%s did not return a true value", name);
- Safefree(name);
}
- op_free(eroot);
- av_free(comppad);
- av_free(comppadname);
+ lex_end();
LEAVE;
sv_setpv(GvSV(gv_fetchpv("@",TRUE)),"");
@@ -9376,7 +9636,7 @@ PP(pp_evalonce)
SP = do_eval(st[1], OP_EVAL, curcop->cop_stash, TRUE,
GIMME, arglast);
if (eval_root) {
- sv_free(cSVOP->op_sv);
+ SvREFCNT_dec(cSVOP->op_sv);
op[1].arg_ptr.arg_cmd = eval_root;
op[1].op_type = (A_CMD|A_DONT);
op[0].op_type = OP_TRY;
@@ -9397,8 +9657,9 @@ PP(pp_entertry)
SAVETMPS;
push_return(cLOGOP->op_other->op_next);
- PUSHBLOCK(cx,CXt_EVAL,SP);
- PUSHEVAL(cx,0);
+ PUSHBLOCK(cx, CXt_EVAL, SP);
+ PUSHEVAL(cx, 0, 0);
+ eval_root = op; /* Only needed so that goto works right. */
in_eval = 1;
sv_setpv(GvSV(gv_fetchpv("@",TRUE)),"");
@@ -9420,8 +9681,12 @@ PP(pp_leavetry)
if (gimme == G_SCALAR) {
MARK = newsp + 1;
- if (MARK <= SP)
- *MARK = sv_mortalcopy(TOPs);
+ if (MARK <= SP) {
+ if (SvFLAGS(TOPs) & (SVs_PADTMP|SVs_TEMP))
+ *MARK = TOPs;
+ else
+ *MARK = sv_mortalcopy(TOPs);
+ }
else {
MEXTEND(mark,0);
*MARK = &sv_undef;
@@ -9430,7 +9695,8 @@ PP(pp_leavetry)
}
else {
for (mark = newsp + 1; mark <= SP; mark++)
- *mark = sv_mortalcopy(*mark);
+ if (!(SvFLAGS(TOPs) & (SVs_PADTMP|SVs_TEMP)))
+ *mark = sv_mortalcopy(*mark);
/* in case LEAVE wipes old return values */
}
@@ -9498,7 +9764,7 @@ PP(pp_ghostent)
#endif
if (GIMME != G_ARRAY) {
- PUSHs(sv = sv_mortalcopy(&sv_undef));
+ PUSHs(sv = sv_newmortal());
if (hent) {
if (which == OP_GHBYNAME) {
sv_setpvn(sv, hent->h_addr, hent->h_length);
@@ -9581,7 +9847,7 @@ PP(pp_gnetent)
EXTEND(SP, 4);
if (GIMME != G_ARRAY) {
- PUSHs(sv = sv_mortalcopy(&sv_undef));
+ PUSHs(sv = sv_newmortal());
if (nent) {
if (which == OP_GNBYNAME)
sv_setiv(sv, (I32)nent->n_net);
@@ -9651,7 +9917,7 @@ PP(pp_gprotoent)
EXTEND(SP, 3);
if (GIMME != G_ARRAY) {
- PUSHs(sv = sv_mortalcopy(&sv_undef));
+ PUSHs(sv = sv_newmortal());
if (pent) {
if (which == OP_GPBYNAME)
sv_setiv(sv, (I32)pent->p_proto);
@@ -9730,7 +9996,7 @@ PP(pp_gservent)
EXTEND(SP, 4);
if (GIMME != G_ARRAY) {
- PUSHs(sv = sv_mortalcopy(&sv_undef));
+ PUSHs(sv = sv_newmortal());
if (sent) {
if (which == OP_GSBYNAME) {
#ifdef HAS_NTOHS
@@ -9901,7 +10167,7 @@ PP(pp_gpwent)
EXTEND(SP, 10);
if (GIMME != G_ARRAY) {
- PUSHs(sv = sv_mortalcopy(&sv_undef));
+ PUSHs(sv = sv_newmortal());
if (pwent) {
if (which == OP_GPWNAM)
sv_setiv(sv, (I32)pwent->pw_uid);
@@ -10018,7 +10284,7 @@ PP(pp_ggrent)
EXTEND(SP, 4);
if (GIMME != G_ARRAY) {
- PUSHs(sv = sv_mortalcopy(&sv_undef));
+ PUSHs(sv = sv_newmortal());
if (grent) {
if (which == OP_GGRNAM)
sv_setiv(sv, (I32)grent->gr_gid);
@@ -10099,7 +10365,7 @@ PP(pp_syscall)
if (tainting) {
while (++MARK <= SP) {
- if (SvMAGICAL(*MARK) && mg_find(*MARK, 't'))
+ if (SvRMAGICAL(*MARK) && mg_find(*MARK, 't'))
tainted = TRUE;
}
MARK = ORIGMARK;