# define Newx(v, n, c) New(0, v, n, c)
#endif
+#ifndef NewOp
+# define NewOp(m, var, c, type) Newz(m, var, c, type)
+#endif
+
#ifndef SvMAGIC_set
# define SvMAGIC_set(sv, val) (SvMAGIC(sv) = (val))
#endif
# define PERL_MAGIC_ext '~'
#endif
+#ifndef PERL_MAGIC_tied
+# define PERL_MAGIC_tied 'P'
+#endif
+
#ifndef MGf_COPY
# define MGf_COPY 0
#endif
mg->mg_flags |= MGf_LOCAL;
#endif /* MGf_LOCAL */
+ if (SvTYPE(sv) < SVt_PVHV)
+ goto done;
+
#if VMG_UVAR
- if (w->uvar && SvTYPE(sv) >= SVt_PVHV) {
+ if (w->uvar) {
MAGIC *prevmagic;
int add_uvar = 1;
struct ufuncs uf[2];
uf[1] = *olduf;
vmg_uvar_del(sv, prevmagic, mg, moremagic);
}
- }
+ } else if (w->cb_get)
+ SvGMAGICAL_off(sv);
if (add_uvar) {
vmg_sv_magicuvar(sv, (const char *) &uf, sizeof(uf));
}
}
+#else
+ if (w->cb_get)
+ SvGMAGICAL_off(sv);
#endif /* VMG_UVAR */
+done:
return 1;
}
#endif /* MGf_LOCAL */
#if VMG_UVAR
+STATIC OP *vmg_pp_resetuvar(pTHX) {
+ SvRMAGICAL_on(cSVOP_sv);
+ return NORMAL;
+}
+
STATIC I32 vmg_svt_val(pTHX_ IV action, SV *sv) {
struct ufuncs *uf;
MAGIC *mg, *umg;
SV *key = NULL, *newkey = NULL;
+ int tied = 0;
umg = mg_find(sv, PERL_MAGIC_uvar);
/* umg can't be NULL or we wouldn't be there. */
action &= HV_FETCH_ISSTORE | HV_FETCH_ISEXISTS | HV_FETCH_LVALUE | HV_DELETE;
for (mg = SvMAGIC(sv); mg; mg = mg->mg_moremagic) {
MGWIZ *w;
- if ((mg->mg_type != PERL_MAGIC_ext)
- || (mg->mg_private < SIG_MIN)
- || (mg->mg_private > SIG_MAX)) { continue; }
+ switch (mg->mg_type) {
+ case PERL_MAGIC_ext:
+ break;
+ case PERL_MAGIC_tied:
+ ++tied;
+ continue;
+ default:
+ continue;
+ }
+ if (mg->mg_private < SIG_MIN || mg->mg_private > SIG_MAX)
+ continue;
w = SV2MGWIZ(mg->mg_ptr);
switch (w->uvar) {
case 0:
}
}
+ if (SvRMAGICAL(sv) && !tied) {
+ /* Temporarily hide the RMAGICAL flag of the hash so it isn't wrongly
+ * mistaken for a tied hash by the rest of hv_common. It will be reset by
+ * the op_ppaddr of a new fake op injected between the current and the next
+ * one. */
+ OP *o = PL_op;
+ if (!o->op_next || o->op_next->op_ppaddr != vmg_pp_resetuvar) {
+ SVOP *svop;
+ NewOp(1101, svop, 1, SVOP);
+ svop->op_type = OP_STUB;
+ svop->op_ppaddr = vmg_pp_resetuvar;
+ svop->op_next = o->op_next;
+ svop->op_flags = 0;
+ svop->op_sv = sv;
+ o->op_next = (OP *) svop;
+ }
+ SvRMAGICAL_off(sv);
+ }
+
return 0;
}
#endif /* VMG_UVAR */
use strict;
use warnings;
-use Test::More tests => 2 * 20 + 6 + 1;
+use Test::More tests => 2 * 21 + 7 + 1;
use Variable::Magic qw/cast dispell MGf_COPY VMG_UVAR/;
use Variable::Magic::TestWatcher;
my $wiz = init
- [ qw/get set len free copy dup local fetch store exists delete/ ], # clear
+ [ qw/get set len clear free copy dup local fetch store exists delete/ ],
'hash';
my %n = map { $_ => int rand 1000 } qw/foo bar baz qux/;
check { cast %h, $wiz } { }, 'cast';
my $s = check { $h{foo} } +{ (fetch => 1) x VMG_UVAR },
- # (copy => 1) x MGf_COPY # if clear magic
'assign element to';
is $s, $n{foo}, 'hash: assign element to correctly';
-$s = check { exists $h{foo} } +{ (exists => 1) x VMG_UVAR }, 'exists';
-ok $s, 'hash: exists correctly';
+for (1 .. 2) {
+ $s = check { exists $h{foo} } +{ (exists => 1) x VMG_UVAR }, "exists ($_)";
+ ok $s, "hash: exists correctly ($_)";
+}
my %b;
check { %b = %h } { }, 'assign to';
my @b = check { @h{qw/bar qux/} }
+{ (fetch => 2) x VMG_UVAR }, 'slice';
- # (copy => 2) x MGf_COPY # if clear magic
is_deeply \@b, [ @n{qw/bar qux/} ], 'hash: slice correctly';
-check { %h = () } { }, 'empty in list context'; # clear => 1
+check { %h = () } { clear => 1 }, 'empty in list context';
check { %h = (a => 1, d => 3); () }
- +{ (store => 2, copy => 2) x VMG_UVAR }, # clear => 1
+ +{ (store => 2, copy => 2) x VMG_UVAR, clear => 1 },
'assign from list in void context';
check { %h = map { $_ => 1 } qw/a b d/; }
- +{ (exists => 3, store => 3, copy => 3) x VMG_UVAR }, # clear =>1
+ +{ (exists => 3, store => 3, copy => 3) x VMG_UVAR, clear => 1 },
'assign from map in list context';
check { $h{d} = 2; () } +{ (store => 1) x VMG_UVAR },
check { cast %b, $wiz } { }, 'cast 2';
} { free => 1 }, 'scope end';
-check { undef %h } { }, 'undef'; # clear => 1
+check { undef %h } { clear => 1 }, 'undef';
check { dispell %h, $wiz } { }, 'dispell';