No meta programming in Mouse/Meta/Role.pm
[gitmo/Mouse.git] / xs-src / Mouse.xs
CommitLineData
df6dd016 1#include "mouse.h"
2
cccb83de 3SV* mouse_package;
4SV* mouse_namespace;
3e44140b 5SV* mouse_methods;
cccb83de 6
df6dd016 7MODULE = Mouse PACKAGE = Mouse::Util
8
9PROTOTYPES: DISABLE
10
cccb83de 11BOOT:
12 mouse_package = newSVpvs_share("package");
13 mouse_namespace = newSVpvs_share("namespace");
3e44140b 14 mouse_methods = newSVpvs_share("methods");
15
1d5ecd5f 16 MOUSE_CALL_BOOT(Mouse__Util__TypeConstraints);
cccb83de 17
18
df6dd016 19bool
20is_class_loaded(SV* sv = &PL_sv_undef)
21
22void
23get_code_info(CV* code)
24PREINIT:
25 GV* gv;
26 HV* stash;
27PPCODE:
28 if((gv = CvGV(code)) && isGV(gv) && (stash = GvSTASH(gv))){
29 EXTEND(SP, 2);
30 mPUSHs(newSVpvn_share(HvNAME_get(stash), HvNAMELEN_get(stash), 0U));
31 mPUSHs(newSVpvn_share(GvNAME_get(gv), GvNAMELEN_get(gv), 0U));
32 }
33
34SV*
35get_code_package(CV* code)
36PREINIT:
37 HV* stash;
38CODE:
39 if(CvGV(code) && isGV(CvGV(code)) && (stash = GvSTASH(CvGV(code)))){
40 RETVAL = newSVpvn_share(HvNAME_get(stash), HvNAMELEN_get(stash), 0U);
41 }
42 else{
43 RETVAL = &PL_sv_no;
44 }
45OUTPUT:
46 RETVAL
47
cccb83de 48CV*
7d96ae4d 49get_code_ref(SV* package, SV* name)
cccb83de 50CODE:
51{
cccb83de 52 HV* stash;
53 HE* he;
7d96ae4d 54
55 if(!SvOK(package)){
56 croak("You must define a package name");
57 }
58 if(!SvOK(name)){
59 croak("You must define a subroutine name");
60 }
61
62 stash = gv_stashsv(package, FALSE);
63 if(!stash){
64 XSRETURN_UNDEF;
cccb83de 65 }
cccb83de 66 he = hv_fetch_ent(stash, name, FALSE, 0U);
67 if(he){
68 GV* const gv = (GV*)hv_iterval(stash, he);
7d96ae4d 69 if(!isGV(gv)){ /* special constant or stub */
cccb83de 70 STRLEN len;
71 const char* const pv = SvPV_const(name, len);
72 gv_init(gv, stash, pv, len, GV_ADDMULTI);
cccb83de 73 }
7d96ae4d 74 RETVAL = GvCVu(gv);
cccb83de 75 }
76 else{
77 RETVAL = NULL;
78 }
79
80 if(!RETVAL){
81 XSRETURN_UNDEF;
82 }
2591e962 83}
84OUTPUT:
85 RETVAL
86
cccb83de 87
7d96ae4d 88MODULE = Mouse PACKAGE = Mouse::Meta::Module
89
90BOOT:
91 INSTALL_SIMPLE_READER_WITH_KEY(Module, name, package);
92 INSTALL_SIMPLE_READER_WITH_KEY(Module, _method_map, methods);
93 INSTALL_SIMPLE_READER_WITH_KEY(Module, _attribute_map, attributes);
94
95HV*
96namespace(SV* self)
97CODE:
98{
3ea28db6 99 SV* const package = mouse_instance_get_slot(aTHX_ self, mouse_package);
7d96ae4d 100 if(!(package && SvOK(package))){
3e44140b 101 croak("No package name defined");
7d96ae4d 102 }
103 RETVAL = gv_stashsv(package, GV_ADDMULTI);
104}
105OUTPUT:
106 RETVAL
107
3e44140b 108# ignore extra arguments for extensibility
109void
110add_method(SV* self, SV* name, SV* code, ...)
111CODE:
112{
b330ee2d 113 SV* const package = mouse_instance_get_slot(aTHX_ self, mouse_package); /* $self->{package} */
114 SV* const methods = mouse_instance_get_slot(aTHX_ self, mouse_methods); /* $self->{methods} */
3e44140b 115 GV* gv;
116 SV* code_ref;
117
118 if(!(package && SvOK(package))){
119 croak("No package name defined");
120 }
121
122 SvGETMAGIC(name);
123 SvGETMAGIC(code);
124
125 if(!SvOK(name)){
126 mouse_throw_error(self, NULL, "You must define a method name");
127 }
128 if(!SvROK(code)){
129 mouse_throw_error(self, NULL, "You must define a CODE reference");
130 }
131
132 code_ref = code;
133 if(SvTYPE(SvRV(code_ref)) != SVt_PVCV){
134 SV* sv = code_ref; /* used in tryAMAGICunDEREF */
135 SV** sp = &sv; /* used in tryAMAGICunDEREF */
136 tryAMAGICunDEREF(to_cv); /* try \&{$code} */
137 if(SvROK(sv) && SvTYPE(SvRV(sv)) == SVt_PVCV){
138 mouse_throw_error(self, NULL, "Not a CODE reference");
139 }
140 code_ref = sv;
141 }
142
143 /* *{$package . '::' . $name} -> *gv */
144 gv = gv_fetchpv(form("%"SVf"::%"SVf, package, name), GV_ADDMULTI, SVt_PVCV);
145 if(GvCVu(gv)){ /* delete *slot{gv} to work around "redefine" warning */
146 SvREFCNT_dec(GvCV(gv));
147 GvCV(gv) = NULL;
148 }
149 sv_setsv_mg((SV*)gv, code_ref); /* *gv = $code_ref */
150
b330ee2d 151 mouse_instance_set_slot(aTHX_ methods, name, code); /* $self->{methods}{$name} = $code */
3e44140b 152
153 /* TODO: name the CODE ref if it's anonymous */
154 //code_entity = (CV*)SvRV(code_ref);
155 //if(CvANON(code_entity)
156 // && CvGV(code_entity) /* a cv under construction has no gv */ ){
157
158 // CvGV(code_entity) = gv;
159 // CvANON_off(code_entity);
160 //}
161}
162
43165725 163MODULE = Mouse PACKAGE = Mouse::Meta::Class
164
165BOOT:
166 INSTALL_SIMPLE_READER(Class, roles);
167 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Class, is_anon_class, anon_serial_id);
168
cccb83de 169void
170linearized_isa(SV* self)
171PPCODE:
172{
173 SV* const stash_ref = mcall0(self, mouse_namespace); /* $self->namespace */
174 AV* linearized_isa;
175 I32 len;
176 I32 i;
177 if(!(SvROK(stash_ref) && SvTYPE(SvRV(stash_ref)) == SVt_PVHV)){
178 croak("namespace() didn't return a HASH reference");
179 }
180 linearized_isa = mro_get_linear_isa((HV*)SvRV(stash_ref));
181 len = AvFILLp(linearized_isa) + 1;
182 EXTEND(SP, len);
183 for(i = 0; i < len; i++){
184 PUSHs(AvARRAY(linearized_isa)[i]);
185 }
186}
187
43165725 188MODULE = Mouse PACKAGE = Mouse::Meta::Role
189
190BOOT:
191 INSTALL_SIMPLE_READER_WITH_KEY(Role, get_roles, roles);
192 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Role, is_anon_role, anon_serial_id);
193
43165725 194MODULE = Mouse PACKAGE = Mouse::Meta::Attribute
195
196BOOT:
197 /* readers */
198 INSTALL_SIMPLE_READER(Attribute, name);
199 INSTALL_SIMPLE_READER(Attribute, associated_class);
200 INSTALL_SIMPLE_READER(Attribute, accessor);
201 INSTALL_SIMPLE_READER(Attribute, reader);
202 INSTALL_SIMPLE_READER(Attribute, writer);
203 INSTALL_SIMPLE_READER(Attribute, predicate);
204 INSTALL_SIMPLE_READER(Attribute, clearer);
205 INSTALL_SIMPLE_READER(Attribute, handles);
206
207 INSTALL_SIMPLE_READER_WITH_KEY(Attribute, _is_metadata, is);
208 INSTALL_SIMPLE_READER_WITH_KEY(Attribute, is_required, required);
209 INSTALL_SIMPLE_READER(Attribute, default);
210 INSTALL_SIMPLE_READER_WITH_KEY(Attribute, is_lazy, lazy);
211 INSTALL_SIMPLE_READER_WITH_KEY(Attribute, is_lazy_build, lazy_build);
212 INSTALL_SIMPLE_READER_WITH_KEY(Attribute, is_weak_ref, weak_ref);
213 INSTALL_SIMPLE_READER(Attribute, init_arg);
214 INSTALL_SIMPLE_READER(Attribute, type_constraint);
215 INSTALL_SIMPLE_READER(Attribute, trigger);
216 INSTALL_SIMPLE_READER(Attribute, builder);
217 INSTALL_SIMPLE_READER_WITH_KEY(Attribute, should_auto_deref, auto_deref);
218 INSTALL_SIMPLE_READER_WITH_KEY(Attribute, should_coerce, coerce);
219 INSTALL_SIMPLE_READER(Attribute, documentation);
220
221 /* predicates */
222 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_accessor, accessor);
223 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_reader, reader);
224 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_writer, writer);
225 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_predicate, predicate);
226 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_clearer, clearer);
227 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_handles, handles);
228
229 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_default, default);
230 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_type_constraint, type_constraint);
231 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_trigger, trigger);
232 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_builder, builder);
233 INSTALL_SIMPLE_PREDICATE_WITH_KEY(Attribute, has_documentation, documentation);
234
93540011 235 newCONSTSUB(gv_stashpvs("Mouse::Meta::Attribute", TRUE), "accessor_metaclass",
236 newSVpvs("Mouse::Meta::Method::Accessor::XS"));
237
43165725 238MODULE = Mouse PACKAGE = Mouse::Meta::TypeConstraint
239
240BOOT:
241 INSTALL_SIMPLE_READER(TypeConstraint, name);
242 INSTALL_SIMPLE_READER(TypeConstraint, parent);
243 INSTALL_SIMPLE_READER(TypeConstraint, message);
244
245 INSTALL_SIMPLE_READER_WITH_KEY(TypeConstraint, _compiled_type_constraint, compiled_type_constraint);
246 INSTALL_SIMPLE_READER(TypeConstraint, _compiled_type_coercion); /* Mouse specific */
247
248 INSTALL_SIMPLE_PREDICATE_WITH_KEY(TypeConstraint, has_coercion, _compiled_type_coercion);
93540011 249
250
251MODULE = Mouse PACKAGE = Mouse::Meta::Method::Accessor::XS
252
93540011 253CV*
254_generate_accessor(klass, SV* attr, metaclass)
255CODE:
256{
257 RETVAL = mouse_instantiate_xs_accessor(aTHX_ attr, mouse_xs_accessor);
258}
259OUTPUT:
260 RETVAL
261
262CV*
263_generate_reader(klass, SV* attr, metaclass)
264CODE:
265{
266 RETVAL = mouse_instantiate_xs_accessor(aTHX_ attr, mouse_xs_reader);
267}
268OUTPUT:
269 RETVAL
270
271CV*
272_generate_writer(klass, SV* attr, metaclass)
273CODE:
274{
275 RETVAL = mouse_instantiate_xs_accessor(aTHX_ attr, mouse_xs_writer);
276}
277OUTPUT:
278 RETVAL
279
280CV*
281_generate_clearer(klass, SV* attr, metaclass)
282CODE:
283{
284 SV* const slot = mcall0s(attr, "name");
285 STRLEN len;
286 const char* const pv = SvPV_const(slot, len);
287 RETVAL = mouse_install_simple_accessor(aTHX_ NULL, pv, len, mouse_xs_simple_clearer);
288}
289OUTPUT:
290 RETVAL
291
292CV*
293_generate_predicate(klass, SV* attr, metaclass)
294CODE:
295{
296 SV* const slot = mcall0s(attr, "name");
297 STRLEN len;
298 const char* const pv = SvPV_const(slot, len);
299 RETVAL = mouse_install_simple_accessor(aTHX_ NULL, pv, len, mouse_xs_simple_predicate);
300}
301OUTPUT:
302 RETVAL
303