|
Packit |
7d6a7d |
################################################################################
|
|
Packit |
7d6a7d |
##
|
|
Packit |
7d6a7d |
## Version 3.x, Copyright (C) 2004-2013, Marcus Holland-Moritz.
|
|
Packit |
7d6a7d |
## Version 2.x, Copyright (C) 2001, Paul Marquess.
|
|
Packit |
7d6a7d |
## Version 1.x, Copyright (C) 1999, Kenneth Albanowski.
|
|
Packit |
7d6a7d |
##
|
|
Packit |
7d6a7d |
## This program is free software; you can redistribute it and/or
|
|
Packit |
7d6a7d |
## modify it under the same terms as Perl itself.
|
|
Packit |
7d6a7d |
##
|
|
Packit |
7d6a7d |
################################################################################
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
=provides
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
caller_cx
|
|
Packit |
7d6a7d |
__UNDEFINED__
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
=implementation
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
#ifdef USE_ITHREADS
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILE(c) ((c)->cop_file)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILEGV(c) (CopFILE(c) ? gv_fetchfile(CopFILE(c)) : Nullgv)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILE_set(c,pv) ((c)->cop_file = savepv(pv))
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILESV(c) (CopFILE(c) ? GvSV(gv_fetchfile(CopFILE(c))) : Nullsv)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILEAV(c) (CopFILE(c) ? GvAV(gv_fetchfile(CopFILE(c))) : Nullav)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASHPV(c) ((c)->cop_stashpv)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASHPV_set(c,pv) ((c)->cop_stashpv = ((pv) ? savepv(pv) : Nullch))
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASH(c) (CopSTASHPV(c) ? gv_stashpv(CopSTASHPV(c),GV_ADD) : Nullhv)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASH_set(c,hv) CopSTASHPV_set(c, (hv) ? HvNAME(hv) : Nullch)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASH_eq(c,hv) ((hv) && (CopSTASHPV(c) == HvNAME(hv) \
|
|
Packit |
7d6a7d |
|| (CopSTASHPV(c) && HvNAME(hv) \
|
|
Packit |
7d6a7d |
&& strEQ(CopSTASHPV(c), HvNAME(hv)))))
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
#else
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILEGV(c) ((c)->cop_filegv)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILEGV_set(c,gv) ((c)->cop_filegv = (GV*)SvREFCNT_inc(gv))
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILE_set(c,pv) CopFILEGV_set((c), gv_fetchfile(pv))
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILESV(c) (CopFILEGV(c) ? GvSV(CopFILEGV(c)) : Nullsv)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILEAV(c) (CopFILEGV(c) ? GvAV(CopFILEGV(c)) : Nullav)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopFILE(c) (CopFILESV(c) ? SvPVX(CopFILESV(c)) : Nullch)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASH(c) ((c)->cop_stash)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASH_set(c,hv) ((c)->cop_stash = (hv))
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASHPV(c) (CopSTASH(c) ? HvNAME(CopSTASH(c)) : Nullch)
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASHPV_set(c,pv) CopSTASH_set((c), gv_stashpv(pv,GV_ADD))
|
|
Packit |
7d6a7d |
__UNDEFINED__ CopSTASH_eq(c,hv) (CopSTASH(c) == (hv))
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
#endif /* USE_ITHREADS */
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
#if { VERSION >= 5.6.0 }
|
|
Packit |
7d6a7d |
#ifndef caller_cx
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
# if defined(NEED_caller_cx) || defined(NEED_caller_cx_GLOBAL)
|
|
Packit |
7d6a7d |
static I32
|
|
Packit |
7d6a7d |
DPPP_dopoptosub_at(const PERL_CONTEXT *cxstk, I32 startingblock)
|
|
Packit |
7d6a7d |
{
|
|
Packit |
7d6a7d |
I32 i;
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
for (i = startingblock; i >= 0; i--) {
|
|
Packit |
7d6a7d |
register const PERL_CONTEXT * const cx = &cxstk[i];
|
|
Packit |
7d6a7d |
switch (CxTYPE(cx)) {
|
|
Packit |
7d6a7d |
default:
|
|
Packit |
7d6a7d |
continue;
|
|
Packit |
7d6a7d |
case CXt_EVAL:
|
|
Packit |
7d6a7d |
case CXt_SUB:
|
|
Packit |
7d6a7d |
case CXt_FORMAT:
|
|
Packit |
7d6a7d |
return i;
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
return i;
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
# endif
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
# if { NEED caller_cx }
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
const PERL_CONTEXT *
|
|
Packit |
7d6a7d |
caller_cx(pTHX_ I32 count, const PERL_CONTEXT **dbcxp)
|
|
Packit |
7d6a7d |
{
|
|
Packit |
7d6a7d |
register I32 cxix = DPPP_dopoptosub_at(cxstack, cxstack_ix);
|
|
Packit |
7d6a7d |
register const PERL_CONTEXT *cx;
|
|
Packit |
7d6a7d |
register const PERL_CONTEXT *ccstack = cxstack;
|
|
Packit |
7d6a7d |
const PERL_SI *top_si = PL_curstackinfo;
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
for (;;) {
|
|
Packit |
7d6a7d |
/* we may be in a higher stacklevel, so dig down deeper */
|
|
Packit |
7d6a7d |
while (cxix < 0 && top_si->si_type != PERLSI_MAIN) {
|
|
Packit |
7d6a7d |
top_si = top_si->si_prev;
|
|
Packit |
7d6a7d |
ccstack = top_si->si_cxstack;
|
|
Packit |
7d6a7d |
cxix = DPPP_dopoptosub_at(ccstack, top_si->si_cxix);
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
if (cxix < 0)
|
|
Packit |
7d6a7d |
return NULL;
|
|
Packit |
7d6a7d |
/* caller() should not report the automatic calls to &DB::sub */
|
|
Packit |
7d6a7d |
if (PL_DBsub && GvCV(PL_DBsub) && cxix >= 0 &&
|
|
Packit |
7d6a7d |
ccstack[cxix].blk_sub.cv == GvCV(PL_DBsub))
|
|
Packit |
7d6a7d |
count++;
|
|
Packit |
7d6a7d |
if (!count--)
|
|
Packit |
7d6a7d |
break;
|
|
Packit |
7d6a7d |
cxix = DPPP_dopoptosub_at(ccstack, cxix - 1);
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
cx = &ccstack[cxix];
|
|
Packit |
7d6a7d |
if (dbcxp) *dbcxp = cx;
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
if (CxTYPE(cx) == CXt_SUB || CxTYPE(cx) == CXt_FORMAT) {
|
|
Packit |
7d6a7d |
const I32 dbcxix = DPPP_dopoptosub_at(ccstack, cxix - 1);
|
|
Packit |
7d6a7d |
/* We expect that ccstack[dbcxix] is CXt_SUB, anyway, the
|
|
Packit |
7d6a7d |
field below is defined for any cx. */
|
|
Packit |
7d6a7d |
/* caller() should not report the automatic calls to &DB::sub */
|
|
Packit |
7d6a7d |
if (PL_DBsub && GvCV(PL_DBsub) && dbcxix >= 0 && ccstack[dbcxix].blk_sub.cv == GvCV(PL_DBsub))
|
|
Packit |
7d6a7d |
cx = &ccstack[dbcxix];
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
return cx;
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
# endif
|
|
Packit |
7d6a7d |
#endif /* caller_cx */
|
|
Packit |
7d6a7d |
#endif /* 5.6.0 */
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
=xsinit
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
#define NEED_caller_cx
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
=xsubs
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
char *
|
|
Packit |
7d6a7d |
CopSTASHPV()
|
|
Packit |
7d6a7d |
CODE:
|
|
Packit |
7d6a7d |
RETVAL = CopSTASHPV(PL_curcop);
|
|
Packit |
7d6a7d |
OUTPUT:
|
|
Packit |
7d6a7d |
RETVAL
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
char *
|
|
Packit |
7d6a7d |
CopFILE()
|
|
Packit |
7d6a7d |
CODE:
|
|
Packit |
7d6a7d |
RETVAL = CopFILE(PL_curcop);
|
|
Packit |
7d6a7d |
OUTPUT:
|
|
Packit |
7d6a7d |
RETVAL
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
#if { VERSION >= 5.6.0 }
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
void
|
|
Packit |
7d6a7d |
caller_cx(level)
|
|
Packit |
7d6a7d |
I32 level
|
|
Packit |
7d6a7d |
PREINIT:
|
|
Packit |
7d6a7d |
const PERL_CONTEXT *cx, *dbcx;
|
|
Packit |
7d6a7d |
const char *pv;
|
|
Packit |
7d6a7d |
const GV *gv;
|
|
Packit |
7d6a7d |
PPCODE:
|
|
Packit |
7d6a7d |
cx = caller_cx(level, &dbcx);
|
|
Packit |
7d6a7d |
if (!cx) XSRETURN_EMPTY;
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
EXTEND(SP, 4);
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
pv = CopSTASHPV(cx->blk_oldcop);
|
|
Packit |
7d6a7d |
ST(0) = pv ? sv_2mortal(newSVpv(pv, 0)) : &PL_sv_undef;
|
|
Packit |
7d6a7d |
gv = CvGV(cx->blk_sub.cv);
|
|
Packit |
7d6a7d |
ST(1) = isGV(gv) ? sv_2mortal(newSVpv(GvNAME(gv), 0)) : &PL_sv_undef;
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
pv = CopSTASHPV(dbcx->blk_oldcop);
|
|
Packit |
7d6a7d |
ST(2) = pv ? sv_2mortal(newSVpv(pv, 0)) : &PL_sv_undef;
|
|
Packit |
7d6a7d |
gv = CvGV(dbcx->blk_sub.cv);
|
|
Packit |
7d6a7d |
ST(3) = isGV(gv) ? sv_2mortal(newSVpv(GvNAME(gv), 0)) : &PL_sv_undef;
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
XSRETURN(4);
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
#endif /* 5.6.0 */
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
=tests plan => 28
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
my $package;
|
|
Packit |
7d6a7d |
{
|
|
Packit |
7d6a7d |
package MyPackage;
|
|
Packit |
7d6a7d |
$package = &Devel::PPPort::CopSTASHPV();
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
print "# $package\n";
|
|
Packit |
7d6a7d |
ok($package, "MyPackage");
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
my $file = &Devel::PPPort::CopFILE();
|
|
Packit |
7d6a7d |
print "# $file\n";
|
|
Packit |
7d6a7d |
ok($file =~ /cop/i);
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
BEGIN {
|
|
Packit |
7d6a7d |
if ($] < 5.006000) {
|
|
Packit |
7d6a7d |
# Skip
|
|
Packit |
7d6a7d |
for (1..28) {
|
|
Packit |
7d6a7d |
ok(1, 1);
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
exit;
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
BEGIN {
|
|
Packit |
7d6a7d |
package DB;
|
|
Packit |
7d6a7d |
no strict "refs";
|
|
Packit |
7d6a7d |
local $^P = 1;
|
|
Packit |
7d6a7d |
sub sub { &$DB::sub }
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
{ package One; sub one { Devel::PPPort::caller_cx($_[0]) } }
|
|
Packit |
7d6a7d |
{
|
|
Packit |
7d6a7d |
package Two;
|
|
Packit |
7d6a7d |
sub two { One::one(@_) }
|
|
Packit |
7d6a7d |
sub dbtwo {
|
|
Packit |
7d6a7d |
BEGIN { $^P = 1 }
|
|
Packit |
7d6a7d |
One::one(@_);
|
|
Packit |
7d6a7d |
BEGIN { $^P = 0 }
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
|
|
Packit |
7d6a7d |
for (
|
|
Packit |
7d6a7d |
# This is rather confusing. The package is the package the call is
|
|
Packit |
7d6a7d |
# made *from*, the sub name is the sub the call is made *to*. When
|
|
Packit |
7d6a7d |
# DB::sub is involved the first call is to DB::sub from the calling
|
|
Packit |
7d6a7d |
# package, the second is to the real sub from package DB.
|
|
Packit |
7d6a7d |
[\&One::one, 0, qw/main one main one/],
|
|
Packit |
7d6a7d |
[\&One::one, 2, ],
|
|
Packit |
7d6a7d |
[\&Two::two, 0, qw/Two one Two one/],
|
|
Packit |
7d6a7d |
[\&Two::two, 1, qw/main two main two/],
|
|
Packit |
7d6a7d |
[\&Two::dbtwo, 0, qw/Two sub DB one/],
|
|
Packit |
7d6a7d |
[\&Two::dbtwo, 1, qw/main dbtwo main dbtwo/],
|
|
Packit |
7d6a7d |
) {
|
|
Packit |
7d6a7d |
my ($sub, $arg, @want) = @$_;
|
|
Packit |
7d6a7d |
my @got = $sub->($arg);
|
|
Packit |
7d6a7d |
ok(@got, @want);
|
|
Packit |
7d6a7d |
for (0..$#want) {
|
|
Packit |
7d6a7d |
ok($got[$_], $want[$_]);
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
}
|
|
Packit |
7d6a7d |
|