]> git.vpit.fr Git - perl/modules/Variable-Magic.git/commitdiff
Don't call the free callback in global destruction
authorVincent Pit <vince@profvince.com>
Tue, 20 Jan 2009 22:01:34 +0000 (23:01 +0100)
committerVincent Pit <vince@profvince.com>
Tue, 20 Jan 2009 22:01:34 +0000 (23:01 +0100)
Because in this case the wizard may not even exist anymore.

And test for that in t/15-self.t if either PERL_DESTRUCT_LEVEL is set or if
Perl::Destruct::Level is present.

Magic.xs
t/15-self.t

index 820d249a730f7590166a50dccfa39476578e263c..40997edcf1286f054f2ae457fe7f69bd1ebf9d67 100644 (file)
--- a/Magic.xs
+++ b/Magic.xs
@@ -568,17 +568,25 @@ STATIC int vmg_svt_clear(pTHX_ SV *sv, MAGIC *mg) {
 }
 
 STATIC int vmg_svt_free(pTHX_ SV *sv, MAGIC *mg) {
+ SV *wiz = (SV *) mg->mg_ptr;
+
+ /* This may happen in global destruction */
+ if (SvTYPE(wiz) == SVTYPEMASK)
+  return 0;
+
  /* So that it can survive tmp cleanup in vmg_cb_call */
  SvREFCNT_inc(sv);
+
 #if !VMG_HAS_PERL_MAINT(5, 11, 0, 32686)
  /* The previous magic tokens were freed but the magic chain wasn't updated, so
   * if you access the sv from the callback the old deleted magics will trigger
   * and cause memory misreads. Change 32686 solved it that way : */
  SvMAGIC_set(sv, mg);
 #endif
+
  /* Perl_mg_free will get rid of the magic and decrement mg->mg_obj and
   * mg->mg_ptr reference count */
- return vmg_cb_call1e(SV2MGWIZ(mg->mg_ptr)->cb_free, sv, mg->mg_obj);
+ return vmg_cb_call1e(SV2MGWIZ(wiz)->cb_free, sv, mg->mg_obj);
 }
 
 #if MGf_COPY
@@ -670,7 +678,7 @@ STATIC int vmg_wizard_free(pTHX_ SV *wiz, MAGIC *mg) {
  char buf[8];
  MGWIZ *w;
 
- if (PL_dirty) /* during global destruction, the context is already freed */
+ if (PL_dirty) /* During global destruction, the context is already freed */
   return 0;
 
  w = SV2MGWIZ(wiz);
index cee1dbd35390bccc47865f4c011b38064dc35bf3..700cb17d714ff582cc1a850053f31731cef5e244 100644 (file)
@@ -50,4 +50,9 @@ my $c = 0;
  ok($res,   're-re-cast on self is valid');
 }
 
+if ((defined $ENV{PERL_DESTRUCT_LEVEL} and $ENV{PERL_DESTRUCT_LEVEL} >= 3)
+    or eval "use Perl::Destruct::Level level => 3; 1") {
+ diag 'Test global destruction';
+}
+
 # is($c, 0, 'magic destructor is called');