Add clone_instance() method to mop_instance_vtbl
[gitmo/Class-MOP.git] / mop.h
CommitLineData
d846ade3 1#ifndef __MOP_H__
2#define __MOP_H__
3
e989c0df 4#define PERL_NO_GET_CONTEXT
d846ade3 5#include "EXTERN.h"
6#include "perl.h"
7#include "XSUB.h"
8
d846ade3 9#include "ppport.h"
10
c79ae27a 11#define MOP_CALL_BOOT(name) STMT_START { \
12 EXTERN_C XS(CAT2(boot_, name)); \
13 PUSHMARK(SP); \
14 CALL_FPTR(CAT2(boot_, name))(aTHX_ cv); \
15 } STMT_END
d846ade3 16
d846ade3 17
1bc0cb6b 18#define MAKE_KEYSV(name) newSVpvn_share(#name, sizeof(#name)-1, 0U)
19
206860b8 20XS(mop_xs_simple_accessor);
7ec7b950 21XS(mop_xs_simple_reader);
206860b8 22XS(mop_xs_simple_writer);
23XS(mop_xs_simple_predicate);
24XS(mop_xs_simple_predicate_for_metaclass);
25XS(mop_xs_simple_clearer);
7ec7b950 26
e1f52a8a 27extern SV *mop_method_metaclass;
28extern SV *mop_associated_metaclass;
daf9799b 29extern SV *mop_associated_attribute;
e1f52a8a 30extern SV *mop_wrap;
1bc0cb6b 31extern SV *mop_methods;
32extern SV *mop_name;
33extern SV *mop_body;
34extern SV *mop_package;
35extern SV *mop_package_name;
36extern SV *mop_package_cache_flag;
87cfe982 37extern SV *mop_initialize;
38extern SV *mop_can;
39extern SV *mop_Class;
1bc0cb6b 40extern SV *mop_VERSION;
41extern SV *mop_ISA;
d846ade3 42
5bd0f967 43/* MOP utilities */
44
d846ade3 45UV mop_check_package_cache_flag(pTHX_ HV *stash);
e1f52a8a 46int mop_get_code_info (SV *coderef, char **pkg, char **name);
d846ade3 47SV *mop_call0(pTHX_ SV *const self, SV *const method);
e989c0df 48SV *mop_call1(pTHX_ SV *const self, SV *const method, SV *const arg1);
49
50#define mop_call0_pvs(o, m) mop_call0(aTHX_ o, newSVpvs_flags(m, SVs_TEMP))
51#define mop_call1_pvs(o, m, a) mop_call1(aTHX_ o, newSVpvs_flags(m, SVs_TEMP), a)
52
eaca1141 53bool mop_is_class_loaded(pTHX_ SV*);
54#define is_class_loaded(klass) mop_is_class_loaded(aTHX_ klass)
d846ade3 55
00370481 56bool mop_is_instance_of(pTHX_ SV*, SV*);
57#define is_instance_of(sv, klass) mop_is_instance_of(aTHX_ sv, klass)
58#define is_instance_of_pvs(sv, klass) mop_is_instance_of(aTHX_ sv, newSVpvs_flags(klass, SVs_TEMP))
59
d846ade3 60typedef enum {
61 TYPE_FILTER_NONE,
62 TYPE_FILTER_CODE,
63 TYPE_FILTER_ARRAY,
64 TYPE_FILTER_IO,
65 TYPE_FILTER_HASH,
66 TYPE_FILTER_SCALAR,
67} type_filter_t;
68
69typedef bool (*get_package_symbols_cb_t) (const char *, STRLEN, SV *, void *);
70
e1f52a8a 71void mop_get_package_symbols(HV *stash, type_filter_t filter, get_package_symbols_cb_t cb, void *ud);
e989c0df 72HV *mop_get_all_package_symbols (HV *stash, type_filter_t filter);
73
74
5bd0f967 75#define MOPf_DIE_ON_FAIL 0x01
76MAGIC* mop_mg_find(pTHX_ SV* const sv, const MGVTBL* const vtbl, I32 const flags);
77
78#ifdef DEBUGGING
79#define MOP_av_at(av, ix) *mop_av_at_safe(aTHX_ (av) , (ix))
80SV** mop_av_at_safe(pTHX_ AV* const mi, I32 const ix);
81#else
82#define MOP_av_at(av, ix) AvARRAY(av)[ix]
83#endif
84
85#define IsObject(sv) (SvROK(sv) && SvOBJECT(SvRV(sv)))
86
87#define newSVsv_share(sv) mop_newSVsv_share(aTHX_ sv)
88SV* mop_newSVsv_share(pTHX_ SV*);
89
90SV* mop_class_of(pTHX_ SV* const sv);
e989c0df 91
68b9c541 92/* Class::MOP::Class */
93
94AV* mop_class_get_all_attributes(pTHX_ SV* const metaclass);
5bd0f967 95
96/* Class::MOP Magic stuff */
d83eddd0 97
98/* All the MOP_mg_* macros require MAGIC* mg for the first argument */
d83eddd0 99
e989c0df 100typedef struct {
87cfe982 101 SV* (*create_instance)(pTHX_ HV* const stash);
e2e260f6 102 SV* (*clone_instance) (pTHX_ SV* const instance);
acdbc948 103 bool (*has_slot) (pTHX_ SV* const mi, SV* const instance);
104 SV* (*get_slot) (pTHX_ SV* const mi, SV* const instance);
105 SV* (*set_slot) (pTHX_ SV* const mi, SV* const instance, SV* const value);
106 SV* (*delete_slot) (pTHX_ SV* const mi, SV* const instance);
107 void (*weaken_slot) (pTHX_ SV* const mi, SV* const instance);
e989c0df 108} mop_instance_vtbl;
109
5bd0f967 110/* Class::MOP::Instance stuff */
111
87cfe982 112SV* mop_instance_create (pTHX_ HV* const stash);
e2e260f6 113SV* mop_instance_clone (pTHX_ SV* const instance);
87cfe982 114bool mop_instance_has_slot (pTHX_ SV* const instance, SV* const slot);
115SV* mop_instance_get_slot (pTHX_ SV* const instance, SV* const slot);
116SV* mop_instance_set_slot (pTHX_ SV* const instance, SV* const slot, SV* const value);
e192207e 117SV* mop_instance_delete_slot(pTHX_ SV* const instance, SV* const slot);
118void mop_instance_weaken_slot(pTHX_ SV* const instance, SV* const slot);
119
e989c0df 120const mop_instance_vtbl* mop_get_default_instance_vtbl(pTHX);
121
19b618cb 122#define MOP_mg_obj(mg) ((mg)->mg_obj)
3ac9bef6 123#define MOP_mg_ptr(mg) ((mg)->mg_ptr)
124#define MOP_mg_vtbl(mg) ((const mop_instance_vtbl*)MOP_mg_ptr(mg))
d83eddd0 125#define MOP_mg_flags(mg) ((mg)->mg_private)
22b82ca0 126#define MOP_mg_virtual(mg) ((mg)->mg_virtual)
d83eddd0 127
3ac9bef6 128#define MOP_mg_obj_refcounted_on(mg) (void)((mg)->mg_flags |= MGf_REFCOUNTED);
129
19b618cb 130#define MOP_mg_create_instance(mg, stash) MOP_mg_vtbl(mg)->create_instance (aTHX_ (stash))
e2e260f6 131#define MOP_mg_clone_instance(mg, o) MOP_mg_vtbl(mg)->clone_instance (aTHX_ (o))
87cfe982 132#define MOP_mg_has_slot(mg, o, slot) MOP_mg_vtbl(mg)->has_slot (aTHX_ (o), (slot))
133#define MOP_mg_get_slot(mg, o, slot) MOP_mg_vtbl(mg)->get_slot (aTHX_ (o), (slot))
134#define MOP_mg_set_slot(mg, o, slot, v) MOP_mg_vtbl(mg)->set_slot (aTHX_ (o), (slot), (v))
135#define MOP_mg_delete_slot(mg, o, slot) MOP_mg_vtbl(mg)->delete_slot (aTHX_ (o), (slot))
136#define MOP_mg_weaken_slot(mg, o, slot) MOP_mg_vtbl(mg)->weaken_slot (aTHX_ (o), (slot))
d83eddd0 137
3ac9bef6 138/* Class::MOP::Attribute stuff */
139
68b9c541 140
141#define MOP_attr_slot(meta) MOP_av_at(meta, MOP_ATTR_SLOT)
142#define MOP_attr_init_arg(meta) MOP_av_at(meta, MOP_ATTR_INIT_ARG)
143#define MOP_attr_default(meta) MOP_av_at(meta, MOP_ATTR_DEFAULT)
144#define MOP_attr_builder(meta) MOP_av_at(meta, MOP_ATTR_BUILDER)
145
146enum mop_attr_ix_t{
147 MOP_ATTR_SLOT,
148
149 MOP_ATTR_INIT_ARG,
150 MOP_ATTR_DEFAULT,
151 MOP_ATTR_BUILDER,
152
153 MOP_ATTR_last,
154};
155
156enum mop_attr_flags_t{ /* keep 16 bits */
157 MOP_ATTRf_HAS_INIT_ARG = 0x0001,
158 MOP_ATTRf_HAS_DEFAULT = 0x0002,
159 MOP_ATTRf_IS_DEFAULT_A_CODEREF = 0x0004,
160 MOP_ATTRf_HAS_BUILDER = 0x0008,
161 MOP_ATTRf_HAS_INITIALIZER = 0x0010,
162
163 MOP_ATTRf_DEBUG = 0x8000
164};
165
166MAGIC* mop_attr_mg(pTHX_ SV* const attr, SV* const instance);
167void mop_attr_initialize_instance_slot(pTHX_ SV* const attr, const mop_instance_vtbl* const vtbl, SV* const instance, HV* const args);
e989c0df 168
169/* Class::MOP::Method::Accessor stuff */
170
d83eddd0 171#define dMOP_self SV* const self = mop_accessor_get_self(aTHX_ ax, items, cv)
172#define dMOP_mg(xsub) MAGIC* mg = (MAGIC*)CvXSUBANY(xsub).any_ptr
173#define dMOP_METHOD_COMMON dMOP_self; dMOP_mg(cv)
e989c0df 174
175
176SV* mop_accessor_get_self(pTHX_ I32 const ax, I32 const items, CV* const cv);
e989c0df 177
22b82ca0 178CV* mop_install_accessor(pTHX_ const char* const fq_name, const char* const key, I32 const keylen, XSUBADDR_t const accessor_impl, const mop_instance_vtbl* vtbl);
179CV* mop_instantiate_xs_accessor(pTHX_ SV* const accessor, XSUBADDR_t const accessor_impl, mop_instance_vtbl* const vtbl);
e989c0df 180
181#define INSTALL_SIMPLE_READER(klass, name) INSTALL_SIMPLE_READER_WITH_KEY(klass, name, name)
182#define INSTALL_SIMPLE_READER_WITH_KEY(klass, name, key) (void)mop_install_accessor(aTHX_ "Class::MOP::" #klass "::" #name, #key, sizeof(#key)-1, mop_xs_simple_reader, NULL)
183
184#define INSTALL_SIMPLE_WRITER(klass, name) INSTALL_SIMPLE_WRITER_WITH_KEY(klass, name, name)
185#define INSTALL_SIMPLE_WRITER_WITH_KEY(klass, name, key) (void)mop_install_accessor(aTHX_ "Class::MOP::" #klass "::" #name, #key, sizeof(#key)-1, mop_xs_simple_writer, NULL)
186
187#define INSTALL_SIMPLE_PREDICATE(klass, name) INSTALL_SIMPLE_PREDICATE_WITH_KEY(klass, name, name)
188#define INSTALL_SIMPLE_PREDICATE_WITH_KEY(klass, name, key) (void)mop_install_accessor(aTHX_ "Class::MOP::" #klass "::has_" #name, #key, sizeof(#key)-1, mop_xs_simple_predicate_for_metaclass, NULL)
189
d846ade3 190#endif