7 #define NULL (null_entity)
13 /** Location entity was spawned from in source */
14 .string sourceLocFile;
17 entity __spawn(string _classname, string _sourceFile, int _sourceLine) {
18 entity this = _spawn();
19 this.classname = _classname;
20 this.sourceLocFile = _sourceFile;
21 this.sourceLocLine = _sourceLine;
27 #define entityclass(...) OVERLOAD(entityclass, __VA_ARGS__)
28 #define entityclass_1(name) entityclass_2(name, Object)
29 #ifndef QCC_SUPPORT_ENTITYCLASS
30 #define entityclass_2(name, base) typedef entity name
32 #define new(class) __spawn(#class, __FILE__, __LINE__)
34 #define entityclass_2(name, base) entityclass name : base {}
35 #define class(name) [[class(name)]]
36 #define new(class) ((class) __spawn(#class, __FILE__, __LINE__))
39 // Classes have a `spawn##cname(entity)` constructor
40 // The parameter is used across [[accumulate]] functions
42 // Macro to hide this implementation detail
43 #define NEW(cname, ...) \
44 OVERLOAD(spawn##cname, new(cname), ##__VA_ARGS__)
46 #define CONSTRUCT(cname, ...) \
47 OVERLOAD(spawn##cname, this, ##__VA_ARGS__)
49 #define CONSTRUCTOR(cname, ...) \
50 cname OVERLOAD(spawn##cname, cname this, __VA_ARGS__)
55 void RegisterClasses() { }
56 ACCUMULATE_FUNCTION(__static_init, RegisterClasses)
58 #define VTBL(cname, base) \
60 entity cname##_vtbl; \
61 void cname##_vtbl_init() { \
62 cname e = new(vtbl); \
63 spawn##cname##_static(e); \
64 e.vtblname = #cname; \
65 /* Top level objects refer to themselves */ \
66 e.vtblbase = base##_vtbl ? base##_vtbl : e; \
69 ACCUMULATE_FUNCTION(RegisterClasses, cname##_vtbl_init)
71 #define INIT_STATIC(cname) [[accumulate]] void spawn##cname##_static(cname this)
72 #define INIT(cname) [[accumulate]] cname spawn##cname##_1(cname this)
74 #define CLASS(cname, base) \
75 entityclass(cname, base); \
76 class(cname) .bool instanceOf##cname; \
78 INIT_STATIC(cname) { \
80 copyentity(cname##_vtbl, this); \
83 spawn##base##_static(this); \
84 this.instanceOf##cname = true; \
87 /* Only statically initialize the current class, it contains everything it inherits */ \
88 if (cname##_vtbl.vtblname == this.classname) { \
89 spawn##cname##_static(this); \
90 this.classname = #cname; \
91 this.vtblname = string_null; \
92 this.vtblbase = cname##_vtbl; \
94 spawn##base##_1(this); \
97 #define METHOD(cname, name, prototype) \
98 class(cname) .prototype name; \
99 prototype cname##_##name; \
100 INIT_STATIC(cname) { this.name = cname##_##name; } \
101 prototype cname##_##name
103 #define ATTRIB(cname, name, type, val) \
104 class(cname) .type name; \
105 INIT(cname) { this.name = val; }
107 #define ATTRIBARRAY(cname, name, type, cnt) \
108 class(cname) .type name[cnt];
110 #define ENDCLASS(cname) \
111 [[last]] INIT(cname) { return this; }
113 #define SUPER(cname) (cname##_vtbl.vtblbase)
115 #define spawn_static(this)
116 #define spawn_1(this)
118 CLASS(Object, ); ENDCLASS(Object)