/
invoker.xs
94 lines (81 loc) · 2.34 KB
/
invoker.xs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
#include "EXTERN.h"
#include "perl.h"
#include "embed.h"
#include "XSUB.h"
#define NEED_sv_2pv_flags_GLOBAL
#include "ppport.h"
#include "hook_op_check.h"
#if PERL_REVISION == 5 && PERL_VERSION >= 13
#else
#define op_append_elem(a,b,c) Perl_append_elem(aTHX_ a,b,c)
#if PERL_REVISION == 5 && PERL_VERSION >= 12
#else
#define pad_findmy(a,b,c) Perl_pad_findmy(aTHX_ a)
#endif
#endif
typedef struct userdata_St {
hook_op_check_id eval_hook;
SV *class;
} userdata_t;
static OP *
invoker_ck_entersub(pTHX_ OP *o, void *ud) {
OP *f = ((cUNOPo->op_first->op_sibling)
? cUNOPo : ((UNOP*)cUNOPo->op_first))->op_first; // pushmark
OP *arg = f->op_sibling; // the actual first argument
if (arg->op_type == OP_RV2SV) {
GV *gv;
OP *gvop = cUNOPx(arg)->op_first;
if (gvop->op_type == OP_GV &&
(gv = cGVOPx_gv(gvop)) &&
!strcmp(GvNAME_get(gv), "-")) {
const PADOFFSET tmp = pad_findmy("$self", 5, 0);
if (tmp == -1) {
gv = gv_fetchpvn_flags("self", 4, GV_NOINIT, SVt_PV);
if (SvOK(gv) && SvTYPE(gv) == SVt_PVGV) {
// "$self" was defined as a package variable -- use it
cUNOPx(arg)->op_first = newGVOP(
gvop->op_type,
gvop->op_flags,
gv
);
}
else {
croak("$self not found");
}
}
else {
OP * const self = newOP(OP_PADSV, 0);
self->op_targ = tmp;
f->op_sibling = self;
self->op_sibling = arg->op_sibling;
op_free(arg);
}
}
}
return o;
}
MODULE = invoker PACKAGE = invoker
PROTOTYPES: ENABLE
hook_op_check_id
setup (class)
SV *class;
PREINIT:
userdata_t *ud;
INIT:
Newx (ud, 1, userdata_t);
CODE:
ud->class = newSVsv (class);
RETVAL = hook_op_check (OP_ENTERSUB, invoker_ck_entersub, ud);
OUTPUT:
RETVAL
void
teardown (class, hook)
hook_op_check_id hook
PREINIT:
userdata_t *ud;
CODE:
ud = (userdata_t *)hook_op_check_remove (OP_ENTERSUB, hook);
if (ud) {
SvREFCNT_dec (ud->class);
Safefree (ud);
}