line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
#include |
2
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
#define EX_CROAK_NOSUB(hvname,subname) croak("Panda::Export: can't export unexisting symbol '%s::%s'", hvname, subname) |
4
|
|
|
|
|
|
|
#define EX_CROAK_EXISTS(hvname,subname) croak("Panda::Export: can't create constant '%s::%s' - symbol already exists", hvname, subname) |
5
|
|
|
|
|
|
|
#define EX_CROAK_NONAME(hvname) croak("Panda::Export: can't define a constant with an empty name in '%s'", hvname) |
6
|
|
|
|
|
|
|
#define EX_CROAK_BADNAME(hvname,subname) croak("Panda::Export: can't create constant '%s::%s' - name must be a valid string", hvname, subname) |
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
namespace xs { namespace exp { |
9
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
static thread_local HV* clists; |
11
|
|
|
|
|
|
|
|
12
|
6
|
|
|
|
|
|
AV* constants_list (pTHX_ HV* stash) { |
13
|
6
|
100
|
|
|
|
|
if (!clists) clists = newHV(); |
14
|
6
|
50
|
|
|
|
|
SV* clist = *hv_fetch(clists, HvNAME(stash), HvNAMELEN(stash), 1); |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
15
|
|
|
|
|
|
|
AV* ret; |
16
|
6
|
100
|
|
|
|
|
if (!SvOK(clist)) { |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
17
|
3
|
50
|
|
|
|
|
SvUPGRADE(clist, SVt_RV); |
18
|
3
|
|
|
|
|
|
SvROK_on(clist); |
19
|
3
|
|
|
|
|
|
ret = newAV(); |
20
|
3
|
|
|
|
|
|
SvRV_set(clist, (SV*) ret); |
21
|
|
|
|
|
|
|
} |
22
|
3
|
|
|
|
|
|
else ret = (AV*)SvRV(clist); |
23
|
6
|
|
|
|
|
|
return ret; |
24
|
|
|
|
|
|
|
} |
25
|
|
|
|
|
|
|
|
26
|
13
|
|
|
|
|
|
static SV* push_export (pTHX_ HV* stash, SV* name, AV* stash_constants_list = NULL, bool need_inc = false) { |
27
|
13
|
50
|
|
|
|
|
if (!stash_constants_list) stash_constants_list = constants_list(aTHX_ stash); |
28
|
|
|
|
|
|
|
|
29
|
13
|
100
|
|
|
|
|
if (!SvPOK(name)) EX_CROAK_BADNAME(HvNAME(stash), SvPV_nolen(name)); |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
30
|
12
|
50
|
|
|
|
|
if (!SvCUR(name)) EX_CROAK_NONAME(HvNAME(stash)); |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
31
|
|
|
|
|
|
|
|
32
|
12
|
50
|
|
|
|
|
if (need_inc) { |
33
|
12
|
50
|
|
|
|
|
if (SvIsCOW_shared_hash(name)) SvREFCNT_inc(name); |
|
|
50
|
|
|
|
|
|
34
|
12
|
|
|
|
|
|
else name = newSVpvn_share(SvPVX_const(name), SvCUR(name), 0); |
35
|
|
|
|
|
|
|
} |
36
|
|
|
|
|
|
|
|
37
|
12
|
|
|
|
|
|
av_push(stash_constants_list, name); |
38
|
12
|
|
|
|
|
|
return name; |
39
|
|
|
|
|
|
|
} |
40
|
|
|
|
|
|
|
|
41
|
0
|
|
|
|
|
|
void register_export (pTHX_ HV* stash, CV* sub) { |
42
|
0
|
0
|
|
|
|
|
HEK* hek = CvNAMED(sub) ? CvNAME_HEK(sub) : GvNAME_HEK(CvGV(sub)); |
43
|
0
|
0
|
|
|
|
|
if (!hek) EX_CROAK_NONAME(HvNAME(stash)); |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
44
|
|
|
|
|
|
|
|
45
|
0
|
|
|
|
|
|
SV* name = newSVhek(hek); |
46
|
|
|
|
|
|
|
assert(SvIsCOW_shared_hash(name)); |
47
|
|
|
|
|
|
|
|
48
|
0
|
|
|
|
|
|
push_export(aTHX_ stash, name); |
49
|
0
|
|
|
|
|
|
} |
50
|
|
|
|
|
|
|
|
51
|
0
|
|
|
|
|
|
void register_export (pTHX_ HV* stash, SV* name) { |
52
|
0
|
|
|
|
|
|
push_export(aTHX_ stash, name, NULL, true); |
53
|
0
|
|
|
|
|
|
} |
54
|
|
|
|
|
|
|
|
55
|
0
|
|
|
|
|
|
void register_export (pTHX_ HV* stash, const char* name) { |
56
|
0
|
|
|
|
|
|
SV* sv_name = newSVpvn_share(name, strlen(name), 0); |
57
|
0
|
|
|
|
|
|
push_export(aTHX_ stash, sv_name); |
58
|
0
|
|
|
|
|
|
} |
59
|
|
|
|
|
|
|
|
60
|
14
|
|
|
|
|
|
void create_constant (pTHX_ HV* stash, SV* name, SV* value, AV* stash_constants_list) { |
61
|
14
|
50
|
|
|
|
|
if (!name) EX_CROAK_NONAME(HvNAME(stash)); |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
62
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
// check that we won't redefine any subroutine |
64
|
14
|
|
|
|
|
|
HE* sym_he = hv_fetch_ent(stash, name, 0, 0); |
65
|
14
|
100
|
|
|
|
|
if (sym_he && HeVAL(sym_he) && isGV(HeVAL(sym_he)) && GvCV(HeVAL(sym_he))) EX_CROAK_EXISTS(HvNAME(stash), SvPV_nolen(name)); |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
66
|
|
|
|
|
|
|
|
67
|
|
|
|
|
|
|
/* gets name as a shared PV */ |
68
|
13
|
|
|
|
|
|
name = push_export(aTHX_ stash, name, stash_constants_list, true); |
69
|
|
|
|
|
|
|
|
70
|
12
|
50
|
|
|
|
|
if (value) SvREFCNT_inc(value); |
71
|
0
|
|
|
|
|
|
else value = newSV(0); |
72
|
12
|
|
|
|
|
|
SvREADONLY_on(value); |
73
|
12
|
|
|
|
|
|
newCONSTSUB(stash, SvPVX_const(name), value); |
74
|
12
|
|
|
|
|
|
} |
75
|
|
|
|
|
|
|
|
76
|
0
|
|
|
|
|
|
void create_constant (pTHX_ HV* stash, const char* name, const char* value, AV* stash_constants_list) { |
77
|
0
|
|
|
|
|
|
SV* namesv = newSVpvn_share(name, strlen(name), 0); |
78
|
0
|
|
|
|
|
|
SV* valuesv = newSVpv(value, 0); |
79
|
0
|
|
|
|
|
|
create_constant(aTHX_ stash, namesv, valuesv, stash_constants_list); |
80
|
0
|
|
|
|
|
|
SvREFCNT_dec_NN(namesv); |
81
|
0
|
|
|
|
|
|
SvREFCNT_dec_NN(valuesv); |
82
|
0
|
|
|
|
|
|
} |
83
|
|
|
|
|
|
|
|
84
|
0
|
|
|
|
|
|
void create_constant (pTHX_ HV* stash, const char* name, int64_t value, AV* stash_constants_list) { |
85
|
0
|
|
|
|
|
|
SV* namesv = newSVpvn_share(name, strlen(name), 0); |
86
|
0
|
|
|
|
|
|
SV* valuesv = newSViv(value); |
87
|
0
|
|
|
|
|
|
create_constant(aTHX_ stash, namesv, valuesv, stash_constants_list); |
88
|
0
|
|
|
|
|
|
SvREFCNT_dec_NN(namesv); |
89
|
0
|
|
|
|
|
|
SvREFCNT_dec_NN(valuesv); |
90
|
0
|
|
|
|
|
|
} |
91
|
|
|
|
|
|
|
|
92
|
0
|
|
|
|
|
|
void create_constant (pTHX_ HV* stash, constant_t constant, AV* stash_constants_list) { |
93
|
0
|
0
|
|
|
|
|
if (constant.svalue) create_constant(aTHX_ stash, constant.name, constant.svalue, stash_constants_list); |
94
|
0
|
|
|
|
|
|
else create_constant(aTHX_ stash, constant.name, constant.value, stash_constants_list); |
95
|
0
|
|
|
|
|
|
} |
96
|
|
|
|
|
|
|
|
97
|
3
|
|
|
|
|
|
void create_constants (pTHX_ HV* stash, HV* constants) { |
98
|
3
|
|
|
|
|
|
AV* clist = constants_list(aTHX_ stash); |
99
|
45
|
50
|
|
|
|
|
XS_HV_ITER(constants, { |
|
|
50
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
100
|
|
|
|
|
|
|
SV* name = newSVpvn_share(HeKEY(he), HeKLEN(he), HeHASH(he)); |
101
|
|
|
|
|
|
|
create_constant(aTHX_ stash, name, HeVAL(he), clist); |
102
|
|
|
|
|
|
|
SvREFCNT_dec_NN(name); |
103
|
|
|
|
|
|
|
}); |
104
|
2
|
|
|
|
|
|
} |
105
|
|
|
|
|
|
|
|
106
|
1
|
|
|
|
|
|
void create_constants (pTHX_ HV* stash, SV** list, size_t items) { |
107
|
1
|
50
|
|
|
|
|
if (!list || !items) return; |
|
|
50
|
|
|
|
|
|
108
|
1
|
|
|
|
|
|
AV* clist = constants_list(aTHX_ stash); |
109
|
1
|
50
|
|
|
|
|
for (size_t i = 0; i < items - 1; i += 2) { |
110
|
1
|
|
|
|
|
|
SV* name = *list++; |
111
|
1
|
|
|
|
|
|
SV* value = *list++; |
112
|
1
|
|
|
|
|
|
create_constant(aTHX_ stash, name, value, clist); |
113
|
|
|
|
|
|
|
} |
114
|
|
|
|
|
|
|
} |
115
|
|
|
|
|
|
|
|
116
|
0
|
|
|
|
|
|
void create_constants (pTHX_ HV* stash, constant_t* list, size_t items) { |
117
|
0
|
0
|
|
|
|
|
if (!list || !items) return; |
|
|
0
|
|
|
|
|
|
118
|
0
|
|
|
|
|
|
AV* clist = constants_list(aTHX_ stash); |
119
|
0
|
0
|
|
|
|
|
while (items--) { |
120
|
0
|
|
|
|
|
|
constant_t constant = *list++; |
121
|
0
|
0
|
|
|
|
|
if (!constant.name) break; |
122
|
0
|
0
|
|
|
|
|
SV* namesv = newSVpvn_share(constant.name, strlen(constant.name), 0); |
123
|
0
|
0
|
|
|
|
|
SV* valuesv = constant.svalue ? newSVpv(constant.svalue, 0) : newSViv(constant.value); |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
124
|
0
|
0
|
|
|
|
|
create_constant(aTHX_ stash, namesv, valuesv, clist); |
125
|
0
|
0
|
|
|
|
|
SvREFCNT_dec_NN(namesv); |
126
|
0
|
0
|
|
|
|
|
SvREFCNT_dec_NN(valuesv); |
127
|
|
|
|
|
|
|
} |
128
|
|
|
|
|
|
|
} |
129
|
|
|
|
|
|
|
|
130
|
27
|
|
|
|
|
|
static inline void _export_sub (pTHX_ HV* from, HV* to, SV* name) { |
131
|
27
|
|
|
|
|
|
HE* symentry_ent = hv_fetch_ent(from, name, 0, 0); |
132
|
27
|
100
|
|
|
|
|
GV* symentry = symentry_ent ? (GV*)HeVAL(symentry_ent) : NULL; |
133
|
27
|
100
|
|
|
|
|
if (!symentry || (isGV(symentry) && !GvCV(symentry))) EX_CROAK_NOSUB(HvNAME(from), SvPV_nolen(name)); |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
134
|
26
|
|
|
|
|
|
SvREFCNT_inc_simple_void_NN((SV*)symentry); |
135
|
26
|
|
|
|
|
|
hv_store_ent(to, name, (SV*)symentry, 0); |
136
|
26
|
|
|
|
|
|
} |
137
|
|
|
|
|
|
|
|
138
|
0
|
|
|
|
|
|
static inline void _export_sub (pTHX_ HV* from, HV* to, const char* name) { |
139
|
0
|
|
|
|
|
|
size_t namelen = strlen(name); |
140
|
0
|
|
|
|
|
|
SV** symentry_ref = hv_fetch(from, name, namelen, 0); |
141
|
0
|
0
|
|
|
|
|
GV* symentry = symentry_ref ? (GV*)(*symentry_ref) : NULL; |
142
|
0
|
0
|
|
|
|
|
if (!symentry || (isGV(symentry) && !GvCV(symentry))) EX_CROAK_NOSUB(HvNAME(from), name); |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
143
|
0
|
|
|
|
|
|
SvREFCNT_inc_simple_void_NN((SV*)symentry); |
144
|
0
|
|
|
|
|
|
hv_store(to, name, namelen, (SV*)symentry, 0); |
145
|
0
|
|
|
|
|
|
} |
146
|
|
|
|
|
|
|
|
147
|
0
|
|
|
|
|
|
void export_sub (pTHX_ HV* from, HV* to, SV* name) { _export_sub(aTHX_ from, to, name); } |
148
|
0
|
|
|
|
|
|
void export_sub (pTHX_ HV* from, HV* to, const char* name) { _export_sub(aTHX_ from, to, name); } |
149
|
|
|
|
|
|
|
|
150
|
1
|
|
|
|
|
|
void export_constants (pTHX_ HV* from, HV* to) { |
151
|
1
|
|
|
|
|
|
AV* clist = constants_list(aTHX_ from); |
152
|
1
|
|
|
|
|
|
export_subs(aTHX_ from, to, AvARRAY(clist), AvFILLp(clist)+1); |
153
|
1
|
|
|
|
|
|
} |
154
|
|
|
|
|
|
|
|
155
|
6
|
|
|
|
|
|
void export_subs (pTHX_ HV* from, HV* to, SV** list, size_t items) { |
156
|
33
|
100
|
|
|
|
|
while (items--) { |
157
|
28
|
|
|
|
|
|
SV* name = *list++; |
158
|
28
|
50
|
|
|
|
|
if (!name) continue; |
159
|
28
|
|
|
|
|
|
const char* name_str = SvPVX_const(name); |
160
|
28
|
100
|
|
|
|
|
if (name_str[0] == ':' && strEQ(name_str, ":const")) { |
|
|
50
|
|
|
|
|
|
161
|
1
|
|
|
|
|
|
AV* clist = constants_list(aTHX_ from); |
162
|
|
|
|
|
|
|
// this check prevents infinite loop if someone created constant with name ":const" |
163
|
1
|
50
|
|
|
|
|
if (AvARRAY(clist) != list) export_subs(aTHX_ from, to, AvARRAY(clist), AvFILLp(clist)+1); |
164
|
1
|
|
|
|
|
|
continue; |
165
|
|
|
|
|
|
|
} |
166
|
27
|
|
|
|
|
|
_export_sub(aTHX_ from, to, name); |
167
|
|
|
|
|
|
|
} |
168
|
5
|
|
|
|
|
|
} |
169
|
|
|
|
|
|
|
|
170
|
0
|
|
|
|
|
|
void export_subs (pTHX_ HV* from, HV* to, const char** list, size_t items) { |
171
|
0
|
0
|
|
|
|
|
while (items--) { |
172
|
0
|
|
|
|
|
|
const char* name = *list++; |
173
|
0
|
0
|
|
|
|
|
if (!name) break; |
174
|
0
|
0
|
|
|
|
|
if (name[0] == ':' && strEQ(name, ":const")) { |
|
|
0
|
|
|
|
|
|
175
|
0
|
|
|
|
|
|
AV* clist = constants_list(aTHX_ from); |
176
|
0
|
|
|
|
|
|
export_subs(aTHX_ from, to, AvARRAY(clist), AvFILLp(clist)+1); |
177
|
0
|
|
|
|
|
|
continue; |
178
|
|
|
|
|
|
|
} |
179
|
0
|
|
|
|
|
|
_export_sub(aTHX_ from, to, name); |
180
|
|
|
|
|
|
|
} |
181
|
0
|
|
|
|
|
|
} |
182
|
|
|
|
|
|
|
|
183
|
8
|
50
|
|
|
|
|
}} |
|
|
50
|
|
|
|
|
|