X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?p=gitmo%2FMouse.git;a=blobdiff_plain;f=xs-src%2FMouseAttribute.xs;h=9bda4c64ec21a564497bfe6892fc413df075b993;hp=f4d4ea467aeff17a663ff258b226606557aad847;hb=0aad026606f2d1d5d27cfb8824dd182f790fa9e2;hpb=d4712779eb600de46ed09b09afd193d3def11786 diff --git a/xs-src/MouseAttribute.xs b/xs-src/MouseAttribute.xs index f4d4ea4..9bda4c6 100644 --- a/xs-src/MouseAttribute.xs +++ b/xs-src/MouseAttribute.xs @@ -1,6 +1,110 @@ -#define NEED_newSVpvn_flags_GLOBAL #include "mouse.h" + +AV* +mouse_get_xa(pTHX_ SV* const attr) { + static MGVTBL mouse_xa_vtbl; /* identity */ + + AV* xa; + MAGIC* mg; + + if(!IsObject(attr)){ + croak("Not a Mouse meta attribute"); + } + + mg = mouse_mg_find(aTHX_ SvRV(attr), &mouse_xa_vtbl, 0x00); + if(!mg){ + SV* slot; + STRLEN len; + const char* pv; + U16 flags = 0x00; + + ENTER; + SAVETMPS; + + xa = newAV(); + + mg = sv_magicext(SvRV(attr), (SV*)xa, PERL_MAGIC_ext, &mouse_xa_vtbl,NULL, 0); + SvREFCNT_dec(xa); /* refcnt++ in sv_magicext */ + + av_extend(xa, MOUSE_XA_last - 1); + + slot = mcall0(attr, mouse_name); + pv = SvPV_const(slot, len); + av_store(xa, MOUSE_XA_SLOT, newSVpvn_share(pv, len, 0U)); + + av_store(xa, MOUSE_XA_ATTRIBUTE, newSVsv(attr)); + + if(predicate_calls(attr, "has_type_constraint")){ + SV* tc; + flags |= MOUSEf_ATTR_HAS_TC; + + tc = mcall0s(attr, "type_constraint"); + av_store(xa, MOUSE_XA_TC, newSVsv(tc)); + + if(predicate_calls(attr, "should_auto_deref")){ + SV* const is_a_type_of = sv_2mortal(newSVpvs_share("is_a_type_of")); + + flags |= MOUSEf_ATTR_SHOULD_AUTO_DEREF; + if( SvTRUEx(mcall1(tc, is_a_type_of, newSVpvs_flags("ArrayRef", SVs_TEMP))) ){ + flags |= MOUSEf_TC_IS_ARRAYREF; + } + else if( SvTRUEx(mcall1(tc, is_a_type_of, newSVpvs_flags("HashRef", SVs_TEMP))) ){ + flags |= MOUSEf_TC_IS_HASHREF; + } + else{ + mouse_throw_error(attr, tc, + "Can not auto de-reference the type constraint '%"SVf"'", + mcall0(tc, mouse_name)); + } + } + + if(predicate_calls(attr, "should_coerce")){ + flags |= MOUSEf_ATTR_SHOULD_COERCE; + } + + } + + if(predicate_calls(attr, "has_trigger")){ + flags |= MOUSEf_ATTR_HAS_TRIGGER; + } + + if(predicate_calls(attr, "is_lazy")){ + flags |= MOUSEf_ATTR_IS_LAZY; + + if(predicate_calls(attr, "has_builder")){ + flags |= MOUSEf_ATTR_HAS_BUILDER; + } + else if(predicate_calls(attr, "has_default")){ + flags |= MOUSEf_ATTR_HAS_DEFAULT; + } + } + + if(predicate_calls(attr, "is_weak_ref")){ + flags |= MOUSEf_ATTR_IS_WEAK_REF; + } + + if(predicate_calls(attr, "is_required")){ + flags |= MOUSEf_ATTR_IS_REQUIRED; + } + + av_store(xa, MOUSE_XA_FLAGS, newSVuv(flags)); + MOUSE_mg_flags(mg) = flags; + + FREETMPS; + LEAVE; + } + else{ + xa = (AV*)MOUSE_mg_obj(mg); + + assert(xa); + assert(SvTYPE(xa) == SVt_PVAV); + } + + return xa; +} + + MODULE = Mouse::Meta::Attribute PACKAGE = Mouse::Meta::Attribute PROTOTYPES: DISABLE