#define PERL_NO_GET_CONTEXT
#include "EXTERN.h"
#include "perl.h"
#include "perlapi.h"
#include "XSUB.h"
#ifdef PERL_OBJECT
#undef PL_op_name
#undef PL_opargs
#undef PL_op_desc
#define PL_op_name (get_op_names())
#define PL_opargs (get_opargs())
#define PL_op_desc (get_op_descs())
#endif
static char *svclassnames[] = {
"B::NULL",
"B::IV",
"B::NV",
"B::RV",
"B::PV",
"B::PVIV",
"B::PVNV",
"B::PVMG",
"B::BM",
"B::PVLV",
"B::AV",
"B::HV",
"B::CV",
"B::GV",
"B::FM",
"B::IO",
};
typedef enum {
OPc_NULL, /* 0 */
OPc_BASEOP, /* 1 */
OPc_UNOP, /* 2 */
OPc_BINOP, /* 3 */
OPc_LOGOP, /* 4 */
OPc_LISTOP, /* 5 */
OPc_PMOP, /* 6 */
OPc_SVOP, /* 7 */
OPc_PADOP, /* 8 */
OPc_PVOP, /* 9 */
OPc_CVOP, /* 10 */
OPc_LOOP, /* 11 */
OPc_COP /* 12 */
} opclass;
static char *opclassnames[] = {
"B::NULL",
"B::OP",
"B::UNOP",
"B::BINOP",
"B::LOGOP",
"B::LISTOP",
"B::PMOP",
"B::SVOP",
"B::PADOP",
"B::PVOP",
"B::CVOP",
"B::LOOP",
"B::COP"
};
static int walkoptree_debug = 0; /* Flag for walkoptree debug hook */
static SV *specialsv_list[6];
SV** my_current_pad;
SV** tmp_pad;
HV* root_cache;
#define GEN_PAD { set_active_sub(find_cv_by_root((OP*)o));tmp_pad = PL_curpad;PL_curpad = my_current_pad; }
#define OLD_PAD (PL_curpad = tmp_pad)
//#define GEN_PAD
//#define OLD_PAD
void
set_active_sub(SV *sv)
{
AV* padlist;
SV** svp;
//dTHX;
// sv_dump(SvRV(sv));
padlist = CvPADLIST(SvRV(sv));
if(!padlist) {
dTHX;
sv_dump(sv);
sv_dump((SV*)padlist);
}
svp = AvARRAY(padlist);
my_current_pad = AvARRAY((AV*)svp[1]);
}
static SV *
find_cv_by_root(OP* o) {
dTHX;
OP* root = o;
SV* key;
SV* val;
HE* cached;
if(PL_compcv && SvTYPE(PL_compcv) == SVt_PVCV &&
!PL_eval_root) {
// printf("Compcv\n");
if(SvROK(PL_compcv))
sv_dump(SvRV(PL_compcv));
return newRV((SV*)PL_compcv);
}
if(!root_cache)
root_cache = newHV();
while(root->op_next)
root = root->op_next;
key = newSViv(PTR2IV(root));
cached = hv_fetch_ent(root_cache, key, 0, 0);
if(cached) {
return HeVAL(cached);
}
if(PL_main_root == root) {
/* Special case, this is the main root */
cached = hv_store_ent(root_cache, key, newRV((SV*)PL_main_cv), 0);
} else if(PL_eval_root == root && PL_compcv) {
SV* tmpcv = (SV*)NEWSV(1104,0);
sv_upgrade((SV *)tmpcv, SVt_PVCV);
CvPADLIST(tmpcv) = CvPADLIST(PL_compcv);
SvREFCNT_inc(CvPADLIST(tmpcv));
CvROOT(tmpcv) = root;
OP_REFCNT_LOCK;
OpREFCNT_inc(root);
OP_REFCNT_UNLOCK;
cached = hv_store_ent(root_cache, key, newRV((SV*)tmpcv), 0);
} else {
/* Need to walk the symbol table, yay */
CV* cv = 0;
SV* sva;
SV* sv;
register SV* svend;
for (sva = PL_sv_arenaroot; sva; sva = (SV*)SvANY(sva)) {
svend = &sva[SvREFCNT(sva)];
for (sv = sva + 1; sv < svend; ++sv) {
if (SvTYPE(sv) != SVTYPEMASK && SvREFCNT(sv)) {
if(SvTYPE(sv) == SVt_PVCV &&
CvROOT(sv) == root
) {
cv = (CV*) sv;
} else if(SvTYPE(sv) == SVt_PVGV && GvGP(sv) &&
GvCV(sv) && !CvXSUB(GvCV(sv)) &&
CvROOT(GvCV(sv)) == root)
{
cv = (CV*) GvCV(sv);
}
}
}
}
if(!cv) {
Perl_die(aTHX_ "I am sorry but we couldn't find this root!\n");
}
cached = hv_store_ent(root_cache, key, newRV((SV*)cv), 0);
}
return (SV*) HeVAL(cached);
}
static SV *
make_sv_object(pTHX_ SV *arg, SV *sv)
{
char *type = 0;
IV iv;
for (iv = 0; iv < sizeof(specialsv_list)/sizeof(SV*); iv++) {
if (sv == specialsv_list[iv]) {
type = "B::SPECIAL";
break;
}
}
if (!type) {
type = svclassnames[SvTYPE(sv)];
iv = PTR2IV(sv);
}
sv_setiv(newSVrv(arg, type), iv);
return arg;
}
#define PERL_CUSTOM_OPS
static I32
op_name_to_num(SV * name)
{
dTHX;
char *s;
char *wanted = SvPV_nolen(name);
int i =0;
int topop = OP_max;
#ifdef PERL_CUSTOM_OPS
topop--;
#endif
if (SvIOK(name) && SvIV(name) >= 0 && SvIV(name) < topop)
return SvIV(name);
for (s = PL_op_name[i]; s; s = PL_op_name[++i]) {
if (strEQ(s, wanted))
return i;
}
#ifdef PERL_CUSTOM_OPS
if (PL_custom_op_names) {
HE* ent;
SV* value;
/* This is sort of a hv_exists, backwards */
(void)hv_iterinit(PL_custom_op_names);
while ((ent = hv_iternext(PL_custom_op_names))) {
if (strEQ(SvPV_nolen(hv_iterval(PL_custom_op_names,ent)),wanted))
return OP_CUSTOM;
}
}
#endif
croak("No such op \"%s\"", SvPV_nolen(name));
return -1;
}
#ifdef PERL_CUSTOM_OPS
static void*
custom_op_ppaddr(char *name)
{
dTHX;
HE *ent;
SV *value;
if (!PL_custom_op_names)
return 0;
/* This is sort of a hv_fetch, backwards */
(void)hv_iterinit(PL_custom_op_names);
while ((ent = hv_iternext(PL_custom_op_names))) {
if (strEQ(SvPV_nolen(hv_iterval(PL_custom_op_names,ent)),name))
return (void*)SvIV(hv_iterkeysv(ent));
}
return 0;
}
#endif
static opclass
cc_opclass(pTHX_ OP *o)
{
if (!o)
return OPc_NULL;
// op_dump(o);
if (o->op_type == 0)
return (o->op_flags & OPf_KIDS) ? OPc_UNOP : OPc_BASEOP;
if (o->op_type == OP_SASSIGN)
return ((o->op_private & OPpASSIGN_BACKWARDS) ? OPc_UNOP : OPc_BINOP);
#ifdef USE_ITHREADS
if (o->op_type == OP_GV || o->op_type == OP_GVSV || o->op_type == OP_AELEMFAST)
return OPc_PADOP;
#endif
switch (PL_opargs[o->op_type] & OA_CLASS_MASK) {
case OA_BASEOP:
return OPc_BASEOP;
case OA_UNOP:
return OPc_UNOP;
case OA_BINOP:
return OPc_BINOP;
case OA_LOGOP:
return OPc_LOGOP;
case OA_LISTOP:
return OPc_LISTOP;
case OA_PMOP:
return OPc_PMOP;
case OA_SVOP:
return OPc_SVOP;
case OA_PADOP:
return OPc_PADOP;
case OA_PVOP_OR_SVOP:
/*
* Character translations (tr///) are usually a PVOP, keeping a
* pointer to a table of shorts used to look up translations.
* Under utf8, however, a simple table isn't practical; instead,
* the OP is an SVOP, and the SV is a reference to a swash
* (i.e., an RV pointing to an HV).
*/
return (o->op_private & (OPpTRANS_TO_UTF|OPpTRANS_FROM_UTF))
? OPc_SVOP : OPc_PVOP;
case OA_LOOP:
return OPc_LOOP;
case OA_COP:
return OPc_COP;
case OA_BASEOP_OR_UNOP:
/*
* UNI(OP_foo) in toke.c returns token UNI or FUNC1 depending on
* whether parens were seen. perly.y uses OPf_SPECIAL to
* signal whether a BASEOP had empty parens or none.
* Some other UNOPs are created later, though, so the best
* test is OPf_KIDS, which is set in newUNOP.
*/
return (o->op_flags & OPf_KIDS) ? OPc_UNOP : OPc_BASEOP;
case OA_FILESTATOP:
/*
* The file stat OPs are created via UNI(OP_foo) in toke.c but use
* the OPf_REF flag to distinguish between OP types instead of the
* usual OPf_SPECIAL flag. As usual, if OPf_KIDS is set, then we
* return OPc_UNOP so that walkoptree can find our children. If
* OPf_KIDS is not set then we check OPf_REF. Without OPf_REF set
* (no argument to the operator) it's an OP; with OPf_REF set it's
* an SVOP (and op_sv is the GV for the filehandle argument).
*/
return ((o->op_flags & OPf_KIDS) ? OPc_UNOP :
#ifdef USE_ITHREADS
(o->op_flags & OPf_REF) ? OPc_PADOP : OPc_BASEOP);
#else
(o->op_flags & OPf_REF) ? OPc_SVOP : OPc_BASEOP);
#endif
case OA_LOOPEXOP:
/*
* next, last, redo, dump and goto use OPf_SPECIAL to indicate that a
* label was omitted (in which case it's a BASEOP) or else a term was
* seen. In this last case, all except goto are definitely PVOP but
* goto is either a PVOP (with an ordinary constant label), an UNOP
* with OPf_STACKED (with a non-constant non-sub) or an UNOP for
* OP_REFGEN (with goto &sub) in which case OPf_STACKED also seems to
* get set.
*/
if (o->op_flags & OPf_STACKED)
return OPc_UNOP;
else if (o->op_flags & OPf_SPECIAL)
return OPc_BASEOP;
else
return OPc_PVOP;
}
warn("can't determine class of operator %s, assuming BASEOP\n",
PL_op_name[o->op_type]);
return OPc_BASEOP;
}
static char *
cc_opclassname(pTHX_ OP *o)
{
return opclassnames[cc_opclass(aTHX_ o)];
}
static OP *
SVtoO(SV* sv) {
dTHX;
if (SvROK(sv)) {
IV tmp = SvIV((SV*)SvRV(sv));
return INT2PTR(OP*,tmp);
}
else {
return 0;
}
croak("Argument is not a reference");
return 0; /* Not reached */
}
/* Pre-5.7 compatibility */
#ifndef op_clear
void op_clear(OP* o) {
/* Fake it, I'm bored */
croak("This operation requires a newer version of Perl");
}
#endif
#ifndef op_null
#define op_null croak("This operation requires a newer version of Perl");
#endif
#ifndef PM_GETRE
#define PM_GETRE(o) ((o)->op_pmregexp)
#endif
typedef OP *B__OP;
typedef UNOP *B__UNOP;
typedef BINOP *B__BINOP;
typedef LOGOP *B__LOGOP;
typedef LISTOP *B__LISTOP;
typedef PMOP *B__PMOP;
typedef SVOP *B__SVOP;
typedef PADOP *B__PADOP;
typedef PVOP *B__PVOP;
typedef LOOP *B__LOOP;
typedef COP *B__COP;
typedef SV *B__SV;
typedef SV *B__IV;
typedef SV *B__PV;
typedef SV *B__NV;
typedef SV *B__PVMG;
typedef SV *B__PVLV;
typedef SV *B__BM;
typedef SV *B__RV;
typedef AV *B__AV;
typedef HV *B__HV;
typedef CV *B__CV;
typedef GV *B__GV;
typedef IO *B__IO;
typedef MAGIC *B__MAGIC;
MODULE = B::Generate PACKAGE = B PREFIX = B_
void
B_fudge()
CODE:
SSCHECK(2);
SSPUSHPTR((SV*)PL_comppad);
SSPUSHINT(SAVEt_COMPPAD);
B::OP
B_main_root(...)
PROTOTYPE: ;$
CODE:
if (items > 0)
PL_main_root = SVtoO(ST(0));
RETVAL = PL_main_root;
OUTPUT:
RETVAL
B::OP
B_main_start(...)
PROTOTYPE: ;$
CODE:
if (items > 0)
PL_main_start = SVtoO(ST(0));
RETVAL = PL_main_start;
OUTPUT:
RETVAL
#define OP_desc(o) PL_op_desc[o->op_type]
MODULE = B::Generate PACKAGE = B::OP PREFIX = OP_
B::CV
OP_find_cv(o)
B::OP o
CODE:
RETVAL = SvRV(find_cv_by_root((OP*)o));
OUTPUT:
RETVAL
B::OP
OP_next(o, ...)
B::OP o
CODE:
if (items > 1)
o->op_next = SVtoO(ST(1));
RETVAL = o->op_next;
OUTPUT:
RETVAL
B::OP
OP_sibling(o, ...)
B::OP o
CODE:
if (items > 1)
o->op_sibling = SVtoO(ST(1));
RETVAL = o->op_sibling;
OUTPUT:
RETVAL
IV
OP_ppaddr(o, ...)
B::OP o
CODE:
if (items > 1)
o->op_ppaddr = (void*)SvIV(ST(1));
RETVAL = PTR2IV((void*)(o->op_ppaddr));
OUTPUT:
RETVAL
char *
OP_desc(o)
B::OP o
PADOFFSET
OP_targ(o, ...)
B::OP o
CODE:
if (items > 1)
o->op_targ = (PADOFFSET)SvIV(ST(1));
/* begin highly experimental */
if (items > 1 && (SvIV(ST(1)) > 1000 || SvIV(ST(1)) & 0x80000000)) {
int padlist = SvIV(ST(1));
int old_padix = PL_padix;
int old_comppad_name_fill = PL_comppad_name_fill;
int old_min_intro_pending = PL_min_intro_pending;
int old_max_intro_pending = PL_max_intro_pending;
// int old_cv_has_eval = PL_cv_has_eval;
int old_pad_reset_pending = PL_pad_reset_pending;
int old_curpad = PL_curpad;
int old_comppad = PL_comppad;
int old_comppad_name = PL_comppad_name;
// PTR2UV
PL_comppad_name = (AV*)(*av_fetch(padlist, 0, FALSE));
PL_comppad = (AV*)(*av_fetch(padlist, 1, FALSE));
PL_curpad = AvARRAY(PL_comppad);
PL_padix = AvFILLp(PL_comppad_name);
PL_pad_reset_pending = 0;
// <medwards> PL_comppad_name_fill appears irrelevant as long as you stick to pad_alloc, pad_swipe, pad_free.
// PL_comppad_name_fill = 0;
// PL_min_intro_pending = 0;
// PL_cv_has_eval = 0;
o->op_targ = Perl_pad_alloc(pTHX_ 0, SVs_PADTMP);
PL_padix = old_padix;
PL_comppad_name_fill = old_comppad_name_fill;
PL_min_intro_pending = old_min_intro_pending;
PL_max_intro_pending = old_max_intro_pending;
// PL_cv_has_eval = old_cv_has_eval;
PL_pad_reset_pending = old_pad_reset_pending;
PL_curpad = old_curpad;
PL_comppad = old_comppad;
PL_comppad_name = old_comppad_name;
}
/* end highly experimental */
RETVAL = o->op_targ;
OUTPUT:
RETVAL
U16
OP_type(o, ...)
B::OP o
CODE:
if (items > 1) {
o->op_type = (U16)SvIV(ST(1));
o->op_ppaddr = PL_ppaddr[o->op_type];
}
RETVAL = o->op_type;
OUTPUT:
RETVAL
U16
OP_seq(o, ...)
B::OP o
CODE:
if (items > 1)
o->op_seq = (U16)SvIV(ST(1));
RETVAL = o->op_seq;
OUTPUT:
RETVAL
U8
OP_flags(o, ...)
B::OP o
CODE:
if (items > 1)
o->op_flags = (U8)SvIV(ST(1));
RETVAL = o->op_flags;
OUTPUT:
RETVAL
U8
OP_private(o, ...)
B::OP o
CODE:
if (items > 1)
o->op_private = (U8)SvIV(ST(1));
RETVAL = o->op_private;
OUTPUT:
RETVAL
void
OP_dump(o)
B::OP o
CODE:
op_dump(o);
void
OP_clean(o)
B::OP o
CODE:
if (o == PL_main_root)
o->op_next = Nullop;
void
OP_new(class, type, flags)
SV * class
SV * type
I32 flags
SV** sparepad = NO_INIT
OP *o = NO_INIT
OP *saveop = NO_INIT
I32 typenum = NO_INIT
CODE:
sparepad = PL_curpad;
saveop = PL_op;
PL_curpad = AvARRAY(PL_comppad);
typenum = op_name_to_num(type);
o = newOP(typenum, flags);
#ifdef PERL_CUSTOM_OPCODES
if (typenum == OP_CUSTOM)
o->op_ppaddr = custom_op_ppaddr(SvPV_nolen(type));
#endif
PL_curpad = sparepad;
PL_op = saveop;
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::OP"), PTR2IV(o));
void
OP_newstate(class, flags, label, oldo)
SV * class
I32 flags
char * label
B::OP oldo
SV** sparepad = NO_INIT
OP *o = NO_INIT
OP *saveop = NO_INIT
CODE:
sparepad = PL_curpad;
saveop = PL_op;
PL_curpad = AvARRAY(PL_comppad);
o = newSTATEOP(flags, label, oldo);
PL_curpad = sparepad;
PL_op = saveop;
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::LISTOP"), PTR2IV(o));
B::OP
OP_mutate(o, type)
B::OP o
SV* type
I32 rtype = NO_INIT
CODE:
rtype = op_name_to_num(type);
o->op_ppaddr = PL_ppaddr[rtype];
o->op_type = rtype;
OUTPUT:
o
B::OP
OP_convert(o, type, flags)
B::OP o
I32 flags
I32 type
CODE:
if (!o || o->op_type != OP_LIST)
o = newLISTOP(OP_LIST, 0, o, Nullop);
else
o->op_flags &= ~OPf_WANT;
if (!(PL_opargs[type] & OA_MARK) && o->op_type != OP_NULL) {
op_clear(o);
o->op_targ = o->op_type;
}
o->op_type = type;
o->op_ppaddr = PL_ppaddr[type];
o->op_flags |= flags;
o = CALL_FPTR(PL_check[type])(aTHX_ (OP*)o);
if (o->op_type == type)
o = Perl_fold_constants(o);
OUTPUT:
o
MODULE = B::Generate PACKAGE = B::UNOP PREFIX = UNOP_
B::OP
UNOP_first(o, ...)
B::UNOP o
CODE:
if (items > 1)
o->op_first = SVtoO(ST(1));
RETVAL = o->op_first;
OUTPUT:
RETVAL
void
UNOP_new(class, type, flags, sv_first)
SV * class
SV * type
I32 flags
SV * sv_first
OP *first = NO_INIT
OP *o = NO_INIT
I32 typenum = NO_INIT
CODE:
if (SvROK(sv_first)) {
if (!sv_derived_from(sv_first, "B::OP"))
Perl_croak(aTHX_ "Reference 'first' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_first));
first = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_first))
Perl_croak(aTHX_
"'first' argument to B::UNOP->new should be a B::OP object or a false value");
else
first = Nullop;
{
I32 padflag = 0;
SV**sparepad = PL_curpad;
OP* saveop = PL_op;
PL_curpad = AvARRAY(PL_comppad);
typenum = op_name_to_num(type);
o = newUNOP(typenum, flags, first);
#ifdef PERL_CUSTOM_OPCODES
if (typenum == OP_CUSTOM)
o->op_ppaddr = custom_op_ppaddr(SvPV_nolen(type));
#endif
PL_curpad = sparepad;
PL_op = saveop;
}
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::UNOP"), PTR2IV(o));
MODULE = B::Generate PACKAGE = B::BINOP PREFIX = BINOP_
void
BINOP_null(o)
B::BINOP o
CODE:
op_null((OP*)o);
B::OP
BINOP_last(o,...)
B::BINOP o
CODE:
if (items > 1)
o->op_last = SVtoO(ST(1));
RETVAL = o->op_last;
OUTPUT:
RETVAL
void
BINOP_new(class, type, flags, sv_first, sv_last)
SV * class
SV * type
I32 flags
SV * sv_first
SV * sv_last
OP *first = NO_INIT
OP *last = NO_INIT
OP *o = NO_INIT
CODE:
if (SvROK(sv_first)) {
if (!sv_derived_from(sv_first, "B::OP"))
Perl_croak(aTHX_ "Reference 'first' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_first));
first = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_first))
Perl_croak(aTHX_
"'first' argument to B::UNOP->new should be a B::OP object or a false value");
else
first = Nullop;
if (SvROK(sv_last)) {
if (!sv_derived_from(sv_last, "B::OP"))
Perl_croak(aTHX_ "Reference 'last' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_last));
last = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_last))
Perl_croak(aTHX_
"'last' argument to B::BINOP->new should be a B::OP object or a false value");
else
last = Nullop;
{
SV**sparepad = PL_curpad;
OP* saveop = PL_op;
I32 optype = op_name_to_num(type);
PL_curpad = AvARRAY(PL_comppad);
if (optype == OP_SASSIGN || optype == OP_AASSIGN)
o = newASSIGNOP(flags, first, 0, last);
else {
o = newBINOP(optype, flags, first, last);
}
PL_curpad = sparepad;
PL_op = saveop;
}
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::BINOP"), PTR2IV(o));
MODULE = B::Generate PACKAGE = B::LISTOP PREFIX = LISTOP_
void
LISTOP_new(class, type, flags, sv_first, sv_last)
SV * class
SV * type
I32 flags
SV * sv_first
SV * sv_last
OP *first = NO_INIT
OP *last = NO_INIT
OP *o = NO_INIT
CODE:
if (SvROK(sv_first)) {
if (!sv_derived_from(sv_first, "B::OP"))
Perl_croak(aTHX_ "Reference 'first' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_first));
first = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_first))
Perl_croak(aTHX_
"'first' argument to B::UNOP->new should be a B::OP object or a false value");
else
first = Nullop;
if (SvROK(sv_last)) {
if (!sv_derived_from(sv_last, "B::OP"))
Perl_croak(aTHX_ "Reference 'last' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_last));
last = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_last))
Perl_croak(aTHX_
"'last' argument to B::BINOP->new should be a B::OP object or a false value");
else
last = Nullop;
{
SV**sparepad = PL_curpad;
OP* saveop = PL_op;
I32 typenum = op_name_to_num(type);
PL_curpad = AvARRAY(PL_comppad);
o = newLISTOP(typenum, flags, first, last);
#ifdef PERL_CUSTOM_OPCODES
if (typenum == OP_CUSTOM)
o->op_ppaddr = custom_op_ppaddr(SvPV_nolen(type));
#endif
PL_curpad = sparepad;
PL_op = saveop;
}
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::LISTOP"), PTR2IV(o));
MODULE = B::Generate PACKAGE = B::LOGOP PREFIX = LOGOP_
void
LOGOP_new(class, type, flags, sv_first, sv_last)
SV * class
SV * type
I32 flags
SV * sv_first
SV * sv_last
OP *first = NO_INIT
OP *last = NO_INIT
OP *o = NO_INIT
CODE:
if (SvROK(sv_first)) {
if (!sv_derived_from(sv_first, "B::OP"))
Perl_croak(aTHX_ "Reference 'first' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_first));
first = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_first))
Perl_croak(aTHX_
"'first' argument to B::UNOP->new should be a B::OP object or a false value");
else
first = Nullop;
if (SvROK(sv_last)) {
if (!sv_derived_from(sv_last, "B::OP"))
Perl_croak(aTHX_ "Reference 'last' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_last));
last = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_last))
Perl_croak(aTHX_
"'last' argument to B::BINOP->new should be a B::OP object or a false value");
else
last = Nullop;
{
SV**sparepad = PL_curpad;
OP* saveop = PL_op;
I32 typenum = op_name_to_num(type);
PL_curpad = AvARRAY(PL_comppad);
o = newLOGOP(typenum, flags, first, last);
#ifdef PERL_CUSTOM_OPCODES
if (typenum == OP_CUSTOM)
o->op_ppaddr = custom_op_ppaddr(SvPV_nolen(type));
#endif
PL_curpad = sparepad;
PL_op = saveop;
}
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::LOGOP"), PTR2IV(o));
void
LOGOP_newcond(class, flags, sv_first, sv_last, sv_else)
SV * class
I32 flags
SV * sv_first
SV * sv_last
SV * sv_else
OP *first = NO_INIT
OP *last = NO_INIT
OP *elseo = NO_INIT
OP *o = NO_INIT
CODE:
if (SvROK(sv_first)) {
if (!sv_derived_from(sv_first, "B::OP"))
Perl_croak(aTHX_ "Reference 'first' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_first));
first = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_first))
Perl_croak(aTHX_
"'first' argument to B::UNOP->new should be a B::OP object or a false value");
else
first = Nullop;
if (SvROK(sv_last)) {
if (!sv_derived_from(sv_last, "B::OP"))
Perl_croak(aTHX_ "Reference 'last' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_last));
last = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_last))
Perl_croak(aTHX_
"'last' argument to B::BINOP->new should be a B::OP object or a false value");
else
last = Nullop;
if (SvROK(sv_else)) {
if (!sv_derived_from(sv_else, "B::OP"))
Perl_croak(aTHX_ "Reference 'else' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_else));
elseo = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_else))
Perl_croak(aTHX_
"'last' argument to B::BINOP->new should be a B::OP object or a false value");
else
elseo = Nullop;
{
SV**sparepad = PL_curpad;
OP* saveop = PL_op;
PL_curpad = AvARRAY(PL_comppad);
o = newCONDOP(flags, first, last, elseo);
PL_curpad = sparepad;
PL_op = saveop;
}
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::LOGOP"), PTR2IV(o));
B::OP
LOGOP_other(o,...)
B::LOGOP o
CODE:
if (items > 1)
o->op_other = SVtoO(ST(1));
RETVAL = o->op_other;
OUTPUT:
RETVAL
#define PMOP_pmreplroot(o) o->op_pmreplroot
#define PMOP_pmnext(o) o->op_pmnext
#define PMOP_pmregexp(o) o->op_pmregexp
#define PMOP_pmflags(o) o->op_pmflags
#define PMOP_pmpermflags(o) o->op_pmpermflags
MODULE = B::Generate PACKAGE = B::PMOP PREFIX = PMOP_
void
PMOP_pmreplroot(o)
B::PMOP o
OP * root = NO_INIT
CODE:
ST(0) = sv_newmortal();
root = o->op_pmreplroot;
/* OP_PUSHRE stores an SV* instead of an OP* in op_pmreplroot */
if (o->op_type == OP_PUSHRE) {
sv_setiv(newSVrv(ST(0), root ?
svclassnames[SvTYPE((SV*)root)] : "B::SV"),
PTR2IV(root));
}
else {
sv_setiv(newSVrv(ST(0), cc_opclassname(aTHX_ root)), PTR2IV(root));
}
B::OP
PMOP_pmreplstart(o, ...)
B::PMOP o
CODE:
if (items > 1)
o->op_pmreplstart = SVtoO(ST(1));
RETVAL = o->op_pmreplstart;
OUTPUT:
RETVAL
B::PMOP
PMOP_pmnext(o, ...)
B::PMOP o
CODE:
if (items > 1)
o->op_pmnext = (PMOP*)SVtoO(ST(1));
RETVAL = o->op_pmnext;
OUTPUT:
RETVAL
U16
PMOP_pmflags(o)
B::PMOP o
U16
PMOP_pmpermflags(o)
B::PMOP o
void
PMOP_precomp(o)
B::PMOP o
REGEXP * rx = NO_INIT
CODE:
ST(0) = sv_newmortal();
rx = PM_GETRE(o);
if (rx)
sv_setpvn(ST(0), rx->precomp, rx->prelen);
#define SVOP_sv(o) (cSVOPo_sv)
#define SVOP_gv(o) ((GV*)cSVOPo_sv)
MODULE = B::Generate PACKAGE = B::SVOP PREFIX = SVOP_
B::SV
SVOP_sv(o, ...)
B::SVOP o
CODE:
GEN_PAD;
if (items > 1)
cSVOPo_sv = newSVsv(ST(1));
RETVAL = cSVOPo_sv;
OLD_PAD;
OUTPUT:
RETVAL
B::GV
SVOP_gv(o)
B::SVOP o
void
SVOP_new(class, type, flags, sv)
SV * class
SV * type
I32 flags
SV * sv
SV** sparepad = NO_INIT
OP *o = NO_INIT
OP *saveop = NO_INIT
SV* param = NO_INIT
I32 typenum = NO_INIT
CODE:
sparepad = PL_curpad;
PL_curpad = AvARRAY(PL_comppad);
saveop = PL_op;
typenum = op_name_to_num(type); /* XXX More classes here! */
if (typenum == OP_GVSV) {
if (*(SvPV_nolen(sv)) == '$')
param = (SV*)gv_fetchpv(SvPVX(sv)+1, TRUE, SVt_PV);
else
Perl_croak(aTHX_
"First character to GVSV was not dollar");
} else
param = newSVsv(sv);
o = newSVOP(typenum, flags, param);
#ifdef PERL_CUSTOM_OPCODES
if (typenum == OP_CUSTOM)
o->op_ppaddr = custom_op_ppaddr(SvPV_nolen(type));
#endif
//PL_curpad = sparepad;
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::SVOP"), PTR2IV(o));
PL_op = saveop;
#define PADOP_padix(o) o->op_padix
#define PADOP_sv(o) (o->op_padix ? PL_curpad[o->op_padix] : Nullsv)
#define PADOP_gv(o) ((o->op_padix \
&& SvTYPE(PL_curpad[o->op_padix]) == SVt_PVGV) \
? (GV*)PL_curpad[o->op_padix] : Nullgv)
MODULE = B::Generate PACKAGE = B::PADOP PREFIX = PADOP_
PADOFFSET
PADOP_padix(o, ...)
B::PADOP o
CODE:
if (items > 1)
o->op_padix = (PADOFFSET)SvIV(ST(1));
RETVAL = o->op_padix;
OUTPUT:
RETVAL
B::SV
PADOP_sv(o)
B::PADOP o
B::GV
PADOP_gv(o)
B::PADOP o
MODULE = B::Generate PACKAGE = B::PVOP PREFIX = PVOP_
void
PVOP_pv(o)
B::PVOP o
CODE:
/*
* OP_TRANS uses op_pv to point to a table of 256 shorts
* whereas other PVOPs point to a null terminated string.
*/
ST(0) = sv_2mortal(newSVpv(o->op_pv, (o->op_type == OP_TRANS) ?
256 * sizeof(short) : 0));
MODULE = B::Generate PACKAGE = B::LOOP PREFIX = LOOP_
B::OP
LOOP_redoop(o, ...)
B::LOOP o
CODE:
if (items > 1)
o->op_redoop = SVtoO(ST(1));
RETVAL = o->op_redoop;
OUTPUT:
RETVAL
B::OP
LOOP_nextop(o, ...)
B::LOOP o
CODE:
if (items > 1)
o->op_nextop = SVtoO(ST(1));
RETVAL = o->op_nextop;
OUTPUT:
RETVAL
B::OP
LOOP_lastop(o, ...)
B::LOOP o
CODE:
if (items > 1)
o->op_lastop = SVtoO(ST(1));
RETVAL = o->op_lastop;
OUTPUT:
RETVAL
#define COP_label(o) o->cop_label
#define COP_stashpv(o) CopSTASHPV(o)
#define COP_stash(o) CopSTASH(o)
#define COP_file(o) CopFILE(o)
#define COP_cop_seq(o) o->cop_seq
#define COP_arybase(o) o->cop_arybase
#define COP_line(o) CopLINE(o)
#define COP_warnings(o) o->cop_warnings
MODULE = B::Generate PACKAGE = B::COP PREFIX = COP_
char *
COP_label(o)
B::COP o
char *
COP_stashpv(o)
B::COP o
B::HV
COP_stash(o)
B::COP o
char *
COP_file(o)
B::COP o
U32
COP_cop_seq(o)
B::COP o
I32
COP_arybase(o)
B::COP o
U16
COP_line(o)
B::COP o
B::SV
COP_warnings(o)
B::COP o
B::COP
COP_new(class, flags, name, sv_first)
SV * class
char * name
I32 flags
SV * sv_first
OP *first = NO_INIT
OP *o = NO_INIT
CODE:
if (SvROK(sv_first)) {
if (!sv_derived_from(sv_first, "B::OP"))
Perl_croak(aTHX_ "Reference 'first' was not a B::OP object");
else {
IV tmp = SvIV((SV*)SvRV(sv_first));
first = INT2PTR(OP*, tmp);
}
} else if (SvTRUE(sv_first))
Perl_croak(aTHX_
"'first' argument to B::COP->new should be a B::OP object or a false value");
else
first = Nullop;
{
SV**sparepad = PL_curpad;
OP* saveop = PL_op;
PL_curpad = AvARRAY(PL_comppad);
o = newSTATEOP(flags, name, first);
PL_curpad = sparepad;
PL_op = saveop;
}
ST(0) = sv_newmortal();
sv_setiv(newSVrv(ST(0), "B::COP"), PTR2IV(o));
MODULE = B::Generate PACKAGE = B::SV PREFIX = Sv
SV*
Svsv(sv)
B::SV sv
CODE:
RETVAL = newSVsv(sv);
OUTPUT:
RETVAL
void*
Svdump(sv)
B::SV sv
CODE:
sv_dump(sv);
U32
SvFLAGS(sv, ...)
B::SV sv
CODE:
if (items > 1)
sv->sv_flags = SvIV(ST(1));
RETVAL = SvFLAGS(sv);
OUTPUT:
RETVAL
MODULE = B::Generate PACKAGE = B::CV PREFIX = CV_
B::OP
CV_ROOT(cv)
B::CV cv
CODE:
if(cv == PL_main_cv) {
RETVAL = PL_main_root;
} else {
RETVAL = CvROOT(cv);
}
OUTPUT:
RETVAL
B::CV
CV_newsub_simple(class, name, block)
SV* class
SV* name
B::OP block
CV* mycv = NO_INIT
OP* o = NO_INIT
CODE:
o = newSVOP(OP_CONST, 0, name);
mycv = newSUB(start_subparse(FALSE, 0), o, Nullop, block);
/*op_free(o); */
RETVAL = mycv;
OUTPUT:
RETVAL
MODULE = B::Generate PACKAGE = B::PV PREFIX = Sv
void
SvPV(sv,...)
B::PV sv
CODE:
{
if(items > 1) {
sv_setpv(sv, SvPV_nolen(ST(1)));
}
ST(0) = sv_newmortal();
if( SvPOK(sv) ) {
sv_setpvn(ST(0), SvPVX(sv), SvCUR(sv));
SvFLAGS(ST(0)) |= SvUTF8(sv);
}
else {
/* XXX for backward compatibility, but should fail */
/* croak( "argument is not SvPOK" ); */
sv_setpvn(ST(0), NULL, 0);
}
}