| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
/* vi: set ft=c : */ |
|
2
|
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
#define make_croak_op(message) S_make_croak_op(aTHX_ message) |
|
4
|
|
|
|
|
|
|
static OP *S_make_croak_op(pTHX_ SV *message) |
|
5
|
|
|
|
|
|
|
{ |
|
6
|
|
|
|
|
|
|
#if HAVE_PERL_VERSION(5, 22, 0) |
|
7
|
|
|
|
|
|
|
sv_catpvs(message, " at %s line %d.\n"); |
|
8
|
|
|
|
|
|
|
/* die sprintf($message, (caller)[1,2]) */ |
|
9
|
|
|
|
|
|
|
return op_convert_list(OP_DIE, 0, |
|
10
|
|
|
|
|
|
|
op_convert_list(OP_SPRINTF, 0, |
|
11
|
|
|
|
|
|
|
op_append_list(OP_LIST, |
|
12
|
|
|
|
|
|
|
newSVOP(OP_CONST, 0, message), |
|
13
|
|
|
|
|
|
|
newSLICEOP(0, |
|
14
|
|
|
|
|
|
|
op_append_list(OP_LIST, |
|
15
|
|
|
|
|
|
|
newSVOP(OP_CONST, 0, newSViv(1)), |
|
16
|
|
|
|
|
|
|
newSVOP(OP_CONST, 0, newSViv(2))), |
|
17
|
|
|
|
|
|
|
newOP(OP_CALLER, 0))))); |
|
18
|
|
|
|
|
|
|
#else |
|
19
|
|
|
|
|
|
|
/* For some reason I can't work out, the above tree isn't correct. Attempts |
|
20
|
|
|
|
|
|
|
* to correct it still make OP_SPRINTF crash with "Out of memory!". For now |
|
21
|
|
|
|
|
|
|
* lets just avoid the sprintf |
|
22
|
|
|
|
|
|
|
*/ |
|
23
|
|
|
|
|
|
|
sv_catpvs(message, "\n"); |
|
24
|
|
|
|
|
|
|
return newLISTOP(OP_DIE, 0, newOP(OP_PUSHMARK, 0), |
|
25
|
|
|
|
|
|
|
newSVOP(OP_CONST, 0, message)); |
|
26
|
|
|
|
|
|
|
#endif |
|
27
|
|
|
|
|
|
|
} |
|
28
|
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
#if HAVE_PERL_VERSION(5, 26, 0) |
|
30
|
|
|
|
|
|
|
# define HAVE_OP_ARGCHECK |
|
31
|
|
|
|
|
|
|
|
|
32
|
|
|
|
|
|
|
# include "make_argcheck_aux.c.inc" |
|
33
|
|
|
|
|
|
|
#endif |
|
34
|
|
|
|
|
|
|
|
|
35
|
|
|
|
|
|
|
#define make_argcheck_ops(required, optional, slurpy, subname) S_make_argcheck_ops(aTHX_ required, optional, slurpy, subname) |
|
36
|
6
|
|
|
|
|
|
static OP *S_make_argcheck_ops(pTHX_ int required, int optional, char slurpy, SV *subname) |
|
37
|
|
|
|
|
|
|
{ |
|
38
|
6
|
|
|
|
|
|
int params = required + optional; |
|
39
|
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
#ifdef HAVE_OP_ARGCHECK |
|
41
|
|
|
|
|
|
|
UNOP_AUX_item *aux = make_argcheck_aux(params, optional, slurpy); |
|
42
|
|
|
|
|
|
|
|
|
43
|
6
|
|
|
|
|
|
return op_prepend_elem(OP_LINESEQ, newSTATEOP(0, NULL, NULL), |
|
44
|
|
|
|
|
|
|
op_prepend_elem(OP_LINESEQ, newUNOP_AUX(OP_ARGCHECK, 0, NULL, aux), NULL)); |
|
45
|
|
|
|
|
|
|
#else |
|
46
|
|
|
|
|
|
|
/* Older perls lack the convenience of OP_ARGCHECK so we'll have to build an |
|
47
|
|
|
|
|
|
|
* optree ourselves. For now we only support required + optional, no slurpy |
|
48
|
|
|
|
|
|
|
* |
|
49
|
|
|
|
|
|
|
* This code heavily inspired by Perl_parse_subsignature() in toke.c from perl 5.24 |
|
50
|
|
|
|
|
|
|
*/ |
|
51
|
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
OP *ret = NULL; |
|
53
|
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
if(required > 0) { |
|
55
|
|
|
|
|
|
|
SV *message = newSVpvf("Too few arguments for subroutine '%" SVf "'", subname); |
|
56
|
|
|
|
|
|
|
/* @_ >= required or die ... */ |
|
57
|
|
|
|
|
|
|
OP *checkop = |
|
58
|
|
|
|
|
|
|
newSTATEOP(0, NULL, |
|
59
|
|
|
|
|
|
|
newLOGOP(OP_OR, 0, |
|
60
|
|
|
|
|
|
|
newBINOP(OP_GE, 0, |
|
61
|
|
|
|
|
|
|
/* scalar @_ */ |
|
62
|
|
|
|
|
|
|
op_contextualize(newUNOP(OP_RV2AV, 0, newGVOP(OP_GV, 0, PL_defgv)), G_SCALAR), |
|
63
|
|
|
|
|
|
|
newSVOP(OP_CONST, 0, newSViv(required))), |
|
64
|
|
|
|
|
|
|
make_croak_op(message))); |
|
65
|
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
ret = op_append_list(OP_LINESEQ, ret, checkop); |
|
67
|
|
|
|
|
|
|
} |
|
68
|
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
if(!slurpy) { |
|
70
|
|
|
|
|
|
|
SV *message = newSVpvf("Too many arguments for subroutine '%" SVf "'", subname); |
|
71
|
|
|
|
|
|
|
/* @_ <= (required+optional) or die ... */ |
|
72
|
|
|
|
|
|
|
OP *checkop = |
|
73
|
|
|
|
|
|
|
newSTATEOP(0, NULL, |
|
74
|
|
|
|
|
|
|
newLOGOP(OP_OR, 0, |
|
75
|
|
|
|
|
|
|
newBINOP(OP_LE, 0, |
|
76
|
|
|
|
|
|
|
/* scalar @_ */ |
|
77
|
|
|
|
|
|
|
op_contextualize(newUNOP(OP_RV2AV, 0, newGVOP(OP_GV, 0, PL_defgv)), G_SCALAR), |
|
78
|
|
|
|
|
|
|
newSVOP(OP_CONST, 0, newSViv(params))), |
|
79
|
|
|
|
|
|
|
make_croak_op(message))); |
|
80
|
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
ret = op_append_list(OP_LINESEQ, ret, checkop); |
|
82
|
|
|
|
|
|
|
} |
|
83
|
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
/* TODO: If slurpy is % then maybe complain about odd number of leftovers */ |
|
85
|
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
return ret; |
|
87
|
|
|
|
|
|
|
#endif |
|
88
|
|
|
|
|
|
|
} |