#define TC_HAS_PERL(R, V, S) (PERL_REVISION > (R) || (PERL_REVISION == (R) && (PERL_VERSION > (V) || (PERL_VERSION == (V) && (PERL_SUBVERSION >= (S))))))
-#include "reap.h"
-
STATIC void tc_callback(pTHX_ void *ud) {
dSP;
LEAVE;
}
+STATIC int tc_endav_free(pTHX_ SV *sv, MAGIC *mg) {
+ SAVEDESTRUCTOR_X(tc_callback, NULL);
+
+ return 0;
+}
+
+STATIC MGVTBL tc_endav_vtbl = {
+ 0,
+ 0,
+ 0,
+ 0,
+ tc_endav_free
+#if MGf_COPY
+ , 0
+#endif
+#if MGf_DUP
+ , 0
+#endif
+#if MGf_LOCAL
+ , 0
+#endif
+};
+
MODULE = Thread::Cleanup PACKAGE = Thread::Cleanup
PROTOTYPES: DISABLE
void
CLONE(...)
PREINIT:
+ GV *gv;
PPCODE:
- reap(3, tc_callback, NULL);
+ gv = gv_fetchpv(__PACKAGE__ "::_CLEANUP", 0, SVt_PVCV);
+ if (gv) {
+ CV *cv = GvCV(gv);
+ if (!PL_endav)
+ PL_endav = newAV();
+ av_unshift(PL_endav, 1);
+ SvREFCNT_inc(cv);
+ if (!av_store(PL_endav, 0, cv))
+ SvREFCNT_dec(cv);
+ sv_magicext((SV *) PL_endav, NULL, PERL_MAGIC_ext, &tc_endav_vtbl, NULL, 0);
+ }
XSRETURN(0);
+++ /dev/null
-/* This file is part of the Thread::Cleanup Perl module.
- * See http://search.cpan.org/dist/Thread-Cleanup/ */
-
-/* This header provides a specialized version of Scope::Upper::reap that can be
- * called directly from XS.
- * See http://search.cpan.org/dist/Scope-Upper/ for details. */
-
-#ifndef REAP_H
-#define REAP_H 1
-
-#define REAP_DESTRUCTOR_SIZE 3
-
-typedef struct {
- I32 depth;
- I32 *origin;
- void (*cb)(pTHX_ void *);
- void *ud;
- char *dummy;
-} reap_ud;
-
-STATIC void reap_pop(pTHX_ void *);
-
-STATIC void reap_pop(pTHX_ void *ud_) {
- reap_ud *ud = ud_;
- I32 depth, *origin, mark, base;
-
- depth = ud->depth;
- origin = ud->origin;
- mark = origin[depth];
- base = origin[depth - 1];
-
- if (base < mark) {
- PL_savestack_ix = mark;
- leave_scope(base);
- }
- PL_savestack_ix = base;
-
- if ((ud->depth = --depth) > 0) {
- SAVEDESTRUCTOR_X(reap_pop, ud);
- } else {
- void (*cb)(pTHX_ void *) = ud->cb;
- void *cb_ud = ud->ud;
-
- PerlMemShared_free(ud->origin);
- PerlMemShared_free(ud);
-
- SAVEDESTRUCTOR_X(cb, cb_ud);
- }
-}
-
-STATIC void reap(pTHX_ I32 depth, void (*cb)(pTHX_ void *), void *cb_ud) {
-#define reap(D, CB, UD) reap(aTHX_ (D), (CB), (UD))
- reap_ud *ud;
- I32 i;
-
- if (depth > PL_scopestack_ix)
- depth = PL_scopestack_ix;
-
- ud = PerlMemShared_malloc(sizeof *ud);
- ud->depth = depth;
- ud->origin = PerlMemShared_malloc((depth + 1) * sizeof *ud->origin);
- ud->cb = cb;
- ud->ud = cb_ud;
- ud->dummy = NULL;
-
- for (i = depth; i >= 1; --i) {
- I32 j = PL_scopestack_ix - i;
- ud->origin[depth - i] = PL_scopestack[j];
- PL_scopestack[j] += REAP_DESTRUCTOR_SIZE;
- }
- ud->origin[depth] = PL_savestack_ix;
-
- while (PL_savestack_ix + REAP_DESTRUCTOR_SIZE
- <= PL_scopestack[PL_scopestack_ix - 1]) {
- save_pptr(&ud->dummy);
- }
-
- SAVEDESTRUCTOR_X(reap_pop, ud);
-}
-
-#endif /* REAP_H */
--- /dev/null
+#!perl -T
+
+use strict;
+use warnings;
+
+use Config;
+
+use Test::More;
+
+use lib 't/lib';
+use VPIT::TestHelpers;
+
+use Thread::Cleanup;
+
+plan skip_all =>
+ 'perl on Windows with pseudoforks enabled is required for this test'
+ unless $^O eq 'MSWin32' and $Config::Config{d_pseudofork};
+
+my $global_end = 0;
+END { ++$global_end }
+
+my $pid = fork;
+
+plan skip_all => 'could not fork' unless defined $pid;
+
+if ($pid) {
+ waitpid $pid, 0;
+} else {
+ plan tests => 4;
+
+ my $gd = 0;
+ my $immortal = VPIT::TestHelpers::Guard->new(sub { ++$gd });
+ $immortal->{self} = $immortal;
+
+ my $local_end = 0;
+ eval 'END { ++$local_end }';
+
+ Thread::Cleanup::register {
+ pass 'pseudo-fork destructor called';
+ is $local_end, 1, 'pseudo-fork destructor called after local END block';
+ is $global_end, 0, 'pseudo-fork destructor called before global END block';
+ is $gd, 0, 'pseudo-fork destructor called before global destruction';
+ };
+
+ exit;
+}