| 1 |
/* multicall.h (version 1.0) |
| 2 |
* |
| 3 |
* Implements a poor-man's MULTICALL interface for old versions |
| 4 |
* of perl that don't offer a proper one. Intended to be compatible |
| 5 |
* with 5.6.0 and later. |
| 6 |
* |
| 7 |
*/ |
| 8 |
|
| 9 |
#ifdef dMULTICALL |
| 10 |
#define REAL_MULTICALL |
| 11 |
#else |
| 12 |
#undef REAL_MULTICALL |
| 13 |
|
| 14 |
/* In versions of perl where MULTICALL is not defined (i.e. prior |
| 15 |
* to 5.9.4), Perl_pad_push is not exported either. It also has |
| 16 |
* an extra argument in older versions; certainly in the 5.8 series. |
| 17 |
* So we redefine it here. |
| 18 |
*/ |
| 19 |
|
| 20 |
#ifndef AVf_REIFY |
| 21 |
# ifdef SVpav_REIFY |
| 22 |
# define AVf_REIFY SVpav_REIFY |
| 23 |
# else |
| 24 |
# error Neither AVf_REIFY nor SVpav_REIFY is defined |
| 25 |
# endif |
| 26 |
#endif |
| 27 |
|
| 28 |
#ifndef AvFLAGS |
| 29 |
# define AvFLAGS SvFLAGS |
| 30 |
#endif |
| 31 |
|
| 32 |
static void |
| 33 |
multicall_pad_push(pTHX_ AV *padlist, int depth) |
| 34 |
{ |
| 35 |
if (depth <= AvFILLp(padlist)) |
| 36 |
return; |
| 37 |
|
| 38 |
{ |
| 39 |
SV** const svp = AvARRAY(padlist); |
| 40 |
AV* const newpad = newAV(); |
| 41 |
SV** const oldpad = AvARRAY(svp[depth-1]); |
| 42 |
I32 ix = AvFILLp((AV*)svp[1]); |
| 43 |
const I32 names_fill = AvFILLp((AV*)svp[0]); |
| 44 |
SV** const names = AvARRAY(svp[0]); |
| 45 |
AV *av; |
| 46 |
|
| 47 |
for ( ;ix > 0; ix--) { |
| 48 |
if (names_fill >= ix && names[ix] != &PL_sv_undef) { |
| 49 |
const char sigil = SvPVX(names[ix])[0]; |
| 50 |
if ((SvFLAGS(names[ix]) & SVf_FAKE) || sigil == '&') { |
| 51 |
/* outer lexical or anon code */ |
| 52 |
av_store(newpad, ix, SvREFCNT_inc(oldpad[ix])); |
| 53 |
} |
| 54 |
else { /* our own lexical */ |
| 55 |
SV *sv; |
| 56 |
if (sigil == '@') |
| 57 |
sv = (SV*)newAV(); |
| 58 |
else if (sigil == '%') |
| 59 |
sv = (SV*)newHV(); |
| 60 |
else |
| 61 |
sv = NEWSV(0, 0); |
| 62 |
av_store(newpad, ix, sv); |
| 63 |
SvPADMY_on(sv); |
| 64 |
} |
| 65 |
} |
| 66 |
else if (IS_PADGV(oldpad[ix]) || IS_PADCONST(oldpad[ix])) { |
| 67 |
av_store(newpad, ix, SvREFCNT_inc(oldpad[ix])); |
| 68 |
} |
| 69 |
else { |
| 70 |
/* save temporaries on recursion? */ |
| 71 |
SV * const sv = NEWSV(0, 0); |
| 72 |
av_store(newpad, ix, sv); |
| 73 |
SvPADTMP_on(sv); |
| 74 |
} |
| 75 |
} |
| 76 |
av = newAV(); |
| 77 |
av_extend(av, 0); |
| 78 |
av_store(newpad, 0, (SV*)av); |
| 79 |
AvFLAGS(av) = AVf_REIFY; |
| 80 |
|
| 81 |
av_store(padlist, depth, (SV*)newpad); |
| 82 |
AvFILLp(padlist) = depth; |
| 83 |
} |
| 84 |
} |
| 85 |
|
| 86 |
#define dMULTICALL \ |
| 87 |
SV **newsp; /* set by POPBLOCK */ \ |
| 88 |
PERL_CONTEXT *cx; \ |
| 89 |
CV *multicall_cv; \ |
| 90 |
OP *multicall_cop; \ |
| 91 |
bool multicall_oldcatch; \ |
| 92 |
U8 hasargs = 0 |
| 93 |
|
| 94 |
/* Between 5.9.1 and 5.9.2 the retstack was removed, and the |
| 95 |
return op is now stored on the cxstack. */ |
| 96 |
#define HAS_RETSTACK (\ |
| 97 |
PERL_REVISION < 5 || \ |
| 98 |
(PERL_REVISION == 5 && PERL_VERSION < 9) || \ |
| 99 |
(PERL_REVISION == 5 && PERL_VERSION == 9 && PERL_SUBVERSION < 2) \ |
| 100 |
) |
| 101 |
|
| 102 |
|
| 103 |
/* PUSHSUB is defined so differently on different versions of perl |
| 104 |
* that it's easier to define our own version than code for all the |
| 105 |
* different possibilities. |
| 106 |
*/ |
| 107 |
#if HAS_RETSTACK |
| 108 |
# define PUSHSUB_RETSTACK(cx) |
| 109 |
#else |
| 110 |
# define PUSHSUB_RETSTACK(cx) cx->blk_sub.retop = Nullop; |
| 111 |
#endif |
| 112 |
#define MULTICALL_PUSHSUB(cx, the_cv) \ |
| 113 |
cx->blk_sub.cv = the_cv; \ |
| 114 |
cx->blk_sub.olddepth = CvDEPTH(the_cv); \ |
| 115 |
cx->blk_sub.hasargs = hasargs; \ |
| 116 |
cx->blk_sub.lval = PL_op->op_private & \ |
| 117 |
(OPpLVAL_INTRO|OPpENTERSUB_INARGS); \ |
| 118 |
PUSHSUB_RETSTACK(cx) \ |
| 119 |
if (!CvDEPTH(the_cv)) { \ |
| 120 |
(void)SvREFCNT_inc(the_cv); \ |
| 121 |
(void)SvREFCNT_inc(the_cv); \ |
| 122 |
SAVEFREESV(the_cv); \ |
| 123 |
} |
| 124 |
|
| 125 |
#define PUSH_MULTICALL(the_cv) \ |
| 126 |
STMT_START { \ |
| 127 |
CV *_nOnclAshIngNamE_ = the_cv; \ |
| 128 |
AV* padlist = CvPADLIST(_nOnclAshIngNamE_); \ |
| 129 |
multicall_cv = _nOnclAshIngNamE_; \ |
| 130 |
ENTER; \ |
| 131 |
multicall_oldcatch = CATCH_GET; \ |
| 132 |
SAVESPTR(CvROOT(multicall_cv)->op_ppaddr); \ |
| 133 |
CvROOT(multicall_cv)->op_ppaddr = PL_ppaddr[OP_NULL]; \ |
| 134 |
SAVETMPS; SAVEVPTR(PL_op); \ |
| 135 |
CATCH_SET(TRUE); \ |
| 136 |
PUSHSTACKi(PERLSI_SORT); \ |
| 137 |
PUSHBLOCK(cx, CXt_SUB, PL_stack_sp); \ |
| 138 |
MULTICALL_PUSHSUB(cx, multicall_cv); \ |
| 139 |
if (++CvDEPTH(multicall_cv) >= 2) { \ |
| 140 |
PERL_STACK_OVERFLOW_CHECK(); \ |
| 141 |
multicall_pad_push(aTHX_ padlist, CvDEPTH(multicall_cv)); \ |
| 142 |
} \ |
| 143 |
SAVECOMPPAD(); \ |
| 144 |
PL_comppad = (AV*) (AvARRAY(padlist)[CvDEPTH(multicall_cv)]); \ |
| 145 |
PL_curpad = AvARRAY(PL_comppad); \ |
| 146 |
multicall_cop = CvSTART(multicall_cv); \ |
| 147 |
} STMT_END |
| 148 |
|
| 149 |
#define MULTICALL \ |
| 150 |
STMT_START { \ |
| 151 |
PL_op = multicall_cop; \ |
| 152 |
CALLRUNOPS(aTHX); \ |
| 153 |
} STMT_END |
| 154 |
|
| 155 |
#define POP_MULTICALL \ |
| 156 |
STMT_START { \ |
| 157 |
CvDEPTH(multicall_cv)--; \ |
| 158 |
LEAVESUB(multicall_cv); \ |
| 159 |
POPBLOCK(cx,PL_curpm); \ |
| 160 |
POPSTACK; \ |
| 161 |
CATCH_SET(multicall_oldcatch); \ |
| 162 |
LEAVE; \ |
| 163 |
SPAGAIN; \ |
| 164 |
} STMT_END |
| 165 |
|
| 166 |
#endif |