X-Git-Url: http://git.vpit.fr/?a=blobdiff_plain;f=Magic.xs;h=b4ff2439b9d5ef70f8e6bd567c6d82c1f77b5360;hb=9e4e50e7055da570fbaa1116842593c536158e77;hp=f4ec2923552d7a56da825e4232a27fa7ff0b7f46;hpb=2b17c3d6ed76489160907e04347b661919f8fe2c;p=perl%2Fmodules%2FVariable-Magic.git diff --git a/Magic.xs b/Magic.xs index f4ec292..b4ff243 100644 --- a/Magic.xs +++ b/Magic.xs @@ -63,6 +63,19 @@ # define MY_CXT_CLONE NOOP #endif +#if VMG_MULTIPLICITY + +STATIC SV *vmg_clone(pTHX_ SV *sv, tTHX owner) { +#define vmg_clone(P, O) vmg_clone(aTHX_ (P), (O)) + CLONE_PARAMS param; + param.stashes = NULL; /* don't need it unless sv is a PVHV */ + param.flags = 0; + param.proto_perl = owner; + return sv_dup(sv, ¶m); +} + +#endif /* VMG_MULTIPLICITY */ + /* --- Compatibility ------------------------------------------------------- */ #ifndef Newx @@ -155,23 +168,6 @@ typedef struct { START_MY_CXT -STATIC void vmg_cxt_init -#if defined(pMY_CXT) && defined(aMY_CXT) - (pTHX_ pMY_CXT) { -# define vmg_cxt_init() vmg_cxt_init(aTHX_ aMY_CXT) -#else - (pTHX) { - dMY_CXT; -# define vmg_cxt_init() vmg_cxt_init(aTHX) -#endif - MY_CXT.wizz = newHV(); -#ifdef USE_ITHREADS - HvSHAREKEYS_off(MY_CXT.wizz); -#endif - MY_CXT.count = 0; - return; -} - /* --- Signatures ---------------------------------------------------------- */ #define SIG_MIN ((U16) (1u << 8)) @@ -214,6 +210,9 @@ typedef struct { #if VMG_UVAR SV *cb_fetch, *cb_store, *cb_exists, *cb_delete; #endif /* VMG_UVAR */ +#if VMG_MULTIPLICITY + tTHX owner; +#endif /* VMG_MULTIPLICITY */ } MGWIZ; #define MGWIZ2SV(W) (newSVuv(PTR2UV(W))) @@ -644,6 +643,11 @@ STATIC int vmg_wizard_free(pTHX_ SV *wiz, MAGIC *mg) { return 0; w = SV2MGWIZ(wiz); +#if VMG_MULTIPLICITY + if (w->owner != aTHX) + return 0; + w->owner = NULL; +#endif /* VMG_MULTIPLICITY */ if (hv_delete(MY_CXT.wizz, buf, sprintf(buf, "%u", w->sig), 0)) { --MY_CXT.count; @@ -672,6 +676,7 @@ STATIC int vmg_wizard_free(pTHX_ SV *wiz, MAGIC *mg) { if (w->cb_exists != NULL) { SvREFCNT_dec(SvRV(w->cb_exists)); } if (w->cb_delete != NULL) { SvREFCNT_dec(SvRV(w->cb_delete)); } #endif /* VMG_UVAR */ + Safefree(w->vtbl); Safefree(w); @@ -721,6 +726,52 @@ STATIC U16 vmg_sv2sig(pTHX_ SV *sv) { return sig; } +STATIC U16 vmg_wizard_sig(pTHX_ SV *wiz) { +#define vmg_wizard_sig(W) vmg_wizard_sig(aTHX_ (W)) + char buf[8]; + U16 sig; + + if (SvROK(wiz)) { + sig = SV2MGWIZ(SvRV(wiz))->sig; + } else if (SvOK(wiz)) { + sig = vmg_sv2sig(wiz); + } else { + croak(vmg_invalid_wiz); + } + + dMY_CXT; + + if (!hv_fetch(MY_CXT.wizz, buf, sprintf(buf, "%u", sig), 0)) + sig = 0; + + return sig; +} + +STATIC SV *vmg_wizard_wiz(pTHX_ SV *wiz) { +#define vmg_wizard_wiz(W) vmg_wizard_wiz(aTHX_ (W)) + char buf[8]; + SV **old; + U16 sig; + + if (SvROK(wiz)) { + wiz = SvRV(wiz); +#if VMG_MULTIPLICITY + if (SV2MGWIZ(wiz)->owner == aTHX) + return wiz; +#endif /* VMG_MULTIPLICITY */ + sig = SV2MGWIZ(wiz)->sig; + } else if (SvOK(wiz)) { + sig = vmg_sv2sig(wiz); + } else { + croak(vmg_invalid_wiz); + } + + dMY_CXT; + + return (old = hv_fetch(MY_CXT.wizz, buf, sprintf(buf, "%u", sig), 0)) + ? *old : NULL; +} + #define VMG_SET_CB(S, N) \ cb = (S); \ w->cb_ ## N = (SvOK(cb) && SvROK(cb)) ? newRV_inc(SvRV(cb)) : NULL; @@ -735,6 +786,52 @@ STATIC U16 vmg_sv2sig(pTHX_ SV *sv) { w->cb_ ## N = NULL; \ } +#if VMG_MULTIPLICITY + +#define VMG_CLONE_CB(N) \ + z->cb_ ## N = (w->cb_ ## N) ? newRV_noinc(vmg_clone(SvRV(w->cb_ ## N), \ + w->owner)) \ + : NULL; + +STATIC MGWIZ *vmg_wizard_clone(pTHX_ const MGWIZ *w) { +#define vmg_wizard_clone(W) vmg_wizard_clone(aTHX_ (W)) + MGVTBL *t; + MGWIZ *z; + + Newx(t, 1, MGVTBL); + Copy(w->vtbl, t, 1, MGVTBL); + + Newx(z, 1, MGWIZ); + VMG_CLONE_CB(data); + VMG_CLONE_CB(get); + VMG_CLONE_CB(set); + VMG_CLONE_CB(len); + VMG_CLONE_CB(clear); + VMG_CLONE_CB(free); +#if MGf_COPY + VMG_CLONE_CB(copy); +#endif /* MGf_COPY */ +#if MGf_DUP + VMG_CLONE_CB(dup); +#endif /* MGf_DUP */ +#if MGf_LOCAL + VMG_CLONE_CB(local); +#endif /* MGf_LOCAL */ +#if VMG_UVAR + VMG_CLONE_CB(fetch); + VMG_CLONE_CB(store); + VMG_CLONE_CB(exists); + VMG_CLONE_CB(delete); +#endif /* VMG_UVAR */ + z->owner = aTHX; + z->vtbl = t; + z->sig = w->sig; + z->uvar = w->uvar; + + return z; +} + +#endif /* VMG_MULTIPLICITY */ /* --- XS ------------------------------------------------------------------ */ @@ -746,7 +843,9 @@ BOOT: { HV *stash; MY_CXT_INIT; - vmg_cxt_init(); + MY_CXT.wizz = newHV(); + hv_iterinit(MY_CXT.wizz); /* Allocate iterator */ + MY_CXT.count = 0; stash = gv_stashpv(__PACKAGE__, 1); newCONSTSUB(stash, "SIG_MIN", newSVuv(SIG_MIN)); newCONSTSUB(stash, "SIG_MAX", newSVuv(SIG_MAX)); @@ -768,11 +867,37 @@ BOOT: void CLONE(...) PROTOTYPE: DISABLE +PREINIT: + HV *hv; + U16 count; CODE: -#ifdef MY_CXT_CLONE - MY_CXT_CLONE; - vmg_cxt_init(); -#endif +#if VMG_THREADSAFE + { + HE *key; + dMY_CXT; + count = MY_CXT.count; + hv = newHV(); + hv_iterinit(hv); /* Allocate iterator */ + hv_iterinit(MY_CXT.wizz); + while (key = hv_iternext(MY_CXT.wizz)) { + STRLEN len; + char *sig = HePV(key, len); + SV *sv; + MAGIC *mg; + MGWIZ *w; + sv = MGWIZ2SV(vmg_wizard_clone(SV2MGWIZ(HeVAL(key)))); + mg = sv_magicext(sv, NULL, PERL_MAGIC_ext, &vmg_wizard_vtbl, NULL, 0); + mg->mg_private = SIG_WIZ; + SvREADONLY_on(sv); + hv_store(hv, sig, len, sv, HeHASH(key)); + } + } + { + MY_CXT_CLONE; + MY_CXT.wizz = hv; + MY_CXT.count = count; + } +#endif /* VMG_THREADSAFE */ SV *_wizard(...) PROTOTYPE: DISABLE @@ -844,6 +969,9 @@ CODE: VMG_SET_CB(ST(i++), exists); VMG_SET_CB(ST(i++), delete); #endif /* VMG_UVAR */ +#if VMG_MULTIPLICITY + w->owner = aTHX; +#endif /* VMG_MULTIPLICITY */ w->vtbl = t; w->sig = sig; @@ -886,21 +1014,9 @@ PREINIT: AV *args = NULL; SV *ret; CODE: - dMY_CXT; - if (SvROK(wiz)) { - wiz = SvRV(wiz); - } else if (SvOK(wiz)) { - char buf[8]; - SV **old; - U16 sig = vmg_sv2sig(wiz); - if ((old = hv_fetch(MY_CXT.wizz, buf, sprintf(buf, "%u", sig), 0))) { - wiz = *old; - } else { - XSRETURN_UNDEF; - } - } else { - croak(vmg_invalid_sig); - } + wiz = vmg_wizard_wiz(wiz); + if (!wiz) + XSRETURN_UNDEF; if (items > 2) { I32 i; args = newAV(); @@ -923,18 +1039,9 @@ PREINIT: SV *data; U16 sig; CODE: - dMY_CXT; - if (SvROK(wiz)) { - sig = SV2MGWIZ(SvRV(wiz))->sig; - } else if (SvOK(wiz)) { - char buf[8]; - sig = vmg_sv2sig(wiz); - if (!hv_fetch(MY_CXT.wizz, buf, sprintf(buf, "%u", sig), 0)) { - XSRETURN_UNDEF; - } - } else { - croak(vmg_invalid_wiz); - } + sig = vmg_wizard_sig(wiz); + if (!sig) + XSRETURN_UNDEF; data = vmg_data_get(SvRV(sv), sig); if (!data) { XSRETURN_UNDEF; } ST(0) = data; @@ -945,18 +1052,9 @@ PROTOTYPE: \[$@%&*]$ PREINIT: U16 sig; CODE: - dMY_CXT; - if (SvROK(wiz)) { - sig = SV2MGWIZ(SvRV(wiz))->sig; - } else if (SvOK(wiz)) { - char buf[8]; - sig = vmg_sv2sig(wiz); - if (!hv_fetch(MY_CXT.wizz, buf, sprintf(buf, "%u", sig), 0)) { - XSRETURN_UNDEF; - } - } else { - croak(vmg_invalid_wiz); - } + sig = vmg_wizard_sig(wiz); + if (!sig) + XSRETURN_UNDEF; RETVAL = newSVuv(vmg_dispell(SvRV(sv), sig)); OUTPUT: RETVAL