Blame parts/inc/cop

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