]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blob - qcsrc/lib/spawnfunc.qh
Merge branch 'martin-t/has_origin' into 'master'
[xonotic/xonotic-data.pk3dir.git] / qcsrc / lib / spawnfunc.qh
1 #pragma once
2
3 /** If this global exists, only functions with spawnfunc_ name prefix qualify as spawn functions */
4 noref bool require_spawnfunc_prefix;
5 .bool spawnfunc_checked;
6
7 // Optional type checking; increases compile time too much to be enabled by default
8 #if 0
9         bool entityfieldassignablefromeditor(int i)
10         {
11                 switch (entityfieldtype(i))
12                 {
13                         case FIELD_STRING:
14                         case FIELD_FLOAT:
15                         case FIELD_VECTOR:
16                                 return true;
17                 }
18                 return false;
19         }
20
21         #define _spawnfunc_checktypes(fld) \
22                 if (fieldname == #fld) \
23                         if (!entityfieldassignablefromeditor(i)) LOG_FATALF("Entity field '%s' cannot be whitelisted", fieldname);
24 #else
25         #define _spawnfunc_checktypes(fld)
26 #endif
27         #define _spawnfunc_check(fld) \
28                 if (fieldname == #fld) continue;
29
30         noref int __spawnfunc_expecting;
31         noref entity __spawnfunc_expect;
32         noref bool __spawnfunc_unreachable_workaround = true;
33
34     .void(entity) __spawnfunc_constructor;
35     noref IntrusiveList g_spawn_queue;
36
37     #define SPAWNFUNC_INTERNAL_FIELDS(X) \
38         X(string, classname, "spawnfunc") \
39         X(string, targetname, string_null) \
40         /**/
41
42     #define X(T, fld, def) .T fld, __spawnfunc_##fld;
43     SPAWNFUNC_INTERNAL_FIELDS(X)
44     #undef X
45
46     void __spawnfunc_defer(entity prototype, void(entity) constructor)
47     {
48         IL_PUSH(g_spawn_queue, prototype);
49         #define X(T, fld, def) { prototype.__spawnfunc_##fld = prototype.fld; prototype.fld = def; }
50         SPAWNFUNC_INTERNAL_FIELDS(X);
51         #undef X
52         prototype.__spawnfunc_constructor = constructor;
53     }
54
55     noref IntrusiveList g_map_entities;
56     #define __spawnfunc_spawn_all() MACRO_BEGIN \
57         g_map_entities = IL_NEW(); \
58         IL_EACH(g_spawn_queue, true, __spawnfunc_spawn(it)); \
59     MACRO_END
60 #ifdef SVQC
61     void _SV_OnEntityPreSpawnFunction(entity this);
62 #endif
63     void __spawnfunc_spawn(entity prototype)
64     {
65         entity e = new(clone);
66         copyentity(prototype, e);
67         IL_PUSH(g_map_entities, e);
68         #define X(T, fld, def) { e.fld = e.__spawnfunc_##fld; e.__spawnfunc_##fld = def; }
69         SPAWNFUNC_INTERNAL_FIELDS(X);
70         #undef X
71 #ifdef SVQC
72         _SV_OnEntityPreSpawnFunction(e);
73         if (wasfreed(e)) {
74             return;
75         }
76 #endif
77         e.__spawnfunc_constructor(e);
78     }
79
80         noref bool __spawnfunc_first;
81
82         #define spawnfunc_1(id) spawnfunc_2(id, FIELDS_UNION)
83         #define spawnfunc_2(id, whitelist) \
84                 void __spawnfunc_##id(entity this); \
85                 [[accumulate]] void spawnfunc_##id(entity this) \
86                 { \
87                     if (!__spawnfunc_first) { \
88                 __spawnfunc_first = true; \
89                 static_init_early(); \
90                     } \
91                     bool dospawn = true; \
92                     if (__spawnfunc_expecting > 1) { __spawnfunc_expecting = false; } \
93                         else if (__spawnfunc_expecting) { \
94                                 /* engine call */ \
95                 if (!g_spawn_queue) { g_spawn_queue = IL_NEW(); } \
96                                 __spawnfunc_expecting = false; \
97                                 this = __spawnfunc_expect; \
98                                 __spawnfunc_expect = NULL; \
99                 dospawn = false; \
100                         } else { \
101                             /* userland call */ \
102                                 assert(this); \
103                         } \
104                         if (!this.sourceLoc) { \
105                                 this.sourceLoc = __FILE__ ":" STR(__LINE__); \
106                         } \
107                         if (!this.spawnfunc_checked) { \
108                                 for (int i = 0, n = numentityfields(); i < n; ++i) { \
109                                         string value = getentityfieldstring(i, this); \
110                                         string fieldname = entityfieldname(i); \
111                                         whitelist(_spawnfunc_checktypes) \
112                                         if (value == "") continue; \
113                                         if (fieldname == "") continue; \
114                                         FIELDS_COMMON(_spawnfunc_check) \
115                                         whitelist(_spawnfunc_check) \
116                                         LOG_WARNF(_("Entity field %s.%s (%s) is not whitelisted. If you believe this is an error, please file an issue."), #id, fieldname, value); \
117                                 } \
118                                 this.spawnfunc_checked = true; \
119                                 if (this) { \
120                     /* not worldspawn, delay spawn */ \
121                     __spawnfunc_defer(this, __spawnfunc_##id); \
122                 } else { \
123                     /* world might not be "worldspawn" */ \
124                     this.__spawnfunc_constructor = __spawnfunc_##id; \
125                 } \
126                         } \
127                         if (dospawn) { __spawnfunc_##id(this); } \
128                         if (__spawnfunc_unreachable_workaround) return; \
129                 } \
130                 void __spawnfunc_##id(entity this)
131
132         #define FIELD_SCALAR(fld, n) \
133                 fld(n)
134         #define FIELD_VEC(fld, n) \
135                 fld(n) \
136                 fld(n##_x) \
137                 fld(n##_y) \
138                 fld(n##_z)
139
140         #define FIELDS_NONE(fld)
141         #define FIELDS_ALL(fld) if (false)
142
143         #define FIELDS_COMMON(fld) \
144                 FIELD_SCALAR(fld, classname) \
145                 FIELD_SCALAR(fld, sourceLoc) \
146                 FIELD_SCALAR(fld, spawnfunc_checked) \
147                 FIELD_VEC(fld, origin) \
148                 /**/
149
150         #define FIELDS_UNION(fld) \
151                 FIELD_SCALAR(fld, Version) \
152                 FIELD_SCALAR(fld, ammo_cells) \
153                 FIELD_SCALAR(fld, ammo_nails) \
154                 FIELD_SCALAR(fld, ammo_rockets) \
155                 FIELD_SCALAR(fld, armorvalue) \
156                 FIELD_SCALAR(fld, atten) \
157                 FIELD_SCALAR(fld, bgmscriptdecay) \
158                 FIELD_SCALAR(fld, bgmscriptsustain) \
159                 FIELD_SCALAR(fld, bgmscript) \
160                 FIELD_SCALAR(fld, button0) \
161                 FIELD_SCALAR(fld, cnt) \
162                 FIELD_SCALAR(fld, colormap) \
163                 FIELD_SCALAR(fld, count) \
164                 FIELD_SCALAR(fld, curvetarget) \
165                 FIELD_SCALAR(fld, cvarfilter) \
166                 FIELD_SCALAR(fld, debrisdamageforcescale) \
167                 FIELD_SCALAR(fld, debrisfadetime) \
168                 FIELD_SCALAR(fld, debristimejitter) \
169                 FIELD_SCALAR(fld, debristime) \
170                 FIELD_SCALAR(fld, debris) \
171                 FIELD_SCALAR(fld, delay) \
172                 FIELD_SCALAR(fld, dmgtime) \
173                 FIELD_SCALAR(fld, dmg) \
174                 FIELD_SCALAR(fld, dmg_edge) \
175                 FIELD_SCALAR(fld, dmg_force) \
176                 FIELD_SCALAR(fld, dmg_radius) \
177                 FIELD_SCALAR(fld, effects) \
178                 FIELD_SCALAR(fld, flags) \
179                 FIELD_SCALAR(fld, fog) \
180                 FIELD_SCALAR(fld, frags) \
181                 FIELD_SCALAR(fld, frame) \
182                 FIELD_SCALAR(fld, gametypefilter) \
183                 FIELD_SCALAR(fld, geomtype) \
184                 FIELD_SCALAR(fld, gravity) \
185                 FIELD_SCALAR(fld, health) \
186                 FIELD_SCALAR(fld, height) \
187                 FIELD_SCALAR(fld, impulse) \
188                 FIELD_SCALAR(fld, killtarget) \
189                 FIELD_SCALAR(fld, lerpfrac) \
190                 FIELD_SCALAR(fld, light_lev) \
191                 FIELD_SCALAR(fld, lip) \
192                 FIELD_SCALAR(fld, loddistance1) \
193                 FIELD_SCALAR(fld, lodmodel1) \
194                 FIELD_SCALAR(fld, ltime) \
195                 FIELD_SCALAR(fld, map) \
196                 FIELD_SCALAR(fld, mdl) \
197                 FIELD_SCALAR(fld, message2) \
198                 FIELD_SCALAR(fld, message) \
199                 FIELD_SCALAR(fld, modelindex) \
200                 FIELD_SCALAR(fld, modelscale) \
201                 FIELD_SCALAR(fld, model) \
202                 FIELD_SCALAR(fld, monsterid) \
203                 FIELD_SCALAR(fld, monster_moveflags) \
204                 FIELD_SCALAR(fld, monster_name) \
205                 FIELD_SCALAR(fld, movetype) \
206                 FIELD_SCALAR(fld, move_movetype) \
207                 FIELD_SCALAR(fld, netname) \
208                 FIELD_SCALAR(fld, nextthink) \
209                 FIELD_SCALAR(fld, noalign) \
210                 FIELD_SCALAR(fld, noise1) \
211                 FIELD_SCALAR(fld, noise2) \
212                 FIELD_SCALAR(fld, noise) \
213                 FIELD_SCALAR(fld, phase) \
214                 FIELD_SCALAR(fld, platmovetype) \
215                 FIELD_SCALAR(fld, race_place) \
216                 FIELD_SCALAR(fld, radius) \
217                 FIELD_SCALAR(fld, respawntimestart) \
218                 FIELD_SCALAR(fld, respawntimejitter) \
219                 FIELD_SCALAR(fld, respawntime) \
220                 FIELD_SCALAR(fld, restriction) \
221                 FIELD_SCALAR(fld, scale) \
222                 FIELD_SCALAR(fld, skin) \
223                 FIELD_SCALAR(fld, solid) \
224                 FIELD_SCALAR(fld, sound1) \
225                 FIELD_SCALAR(fld, sounds) \
226                 FIELD_SCALAR(fld, spawnflags) \
227                 FIELD_SCALAR(fld, spawnmob) \
228                 FIELD_SCALAR(fld, speed) \
229                 FIELD_SCALAR(fld, strength) \
230                 FIELD_SCALAR(fld, style) \
231                 FIELD_SCALAR(fld, target2) \
232                 FIELD_SCALAR(fld, target3) \
233                 FIELD_SCALAR(fld, target4) \
234                 FIELD_SCALAR(fld, targetname) \
235                 FIELD_SCALAR(fld, target) \
236                 FIELD_SCALAR(fld, target_random) \
237                 FIELD_SCALAR(fld, target_range) \
238                 FIELD_SCALAR(fld, team) \
239                 FIELD_SCALAR(fld, turret_scale_health) \
240                 FIELD_SCALAR(fld, turret_scale_range) \
241                 FIELD_SCALAR(fld, turret_scale_respawn) \
242                 FIELD_SCALAR(fld, volume) \
243                 FIELD_SCALAR(fld, wait) \
244                 FIELD_SCALAR(fld, warpzone_fadeend) \
245                 FIELD_SCALAR(fld, warpzone_fadestart) \
246                 FIELD_SCALAR(fld, weapon) \
247                 FIELD_SCALAR(fld, worldtype) \
248                 FIELD_VEC(fld, absmax) \
249                 FIELD_VEC(fld, absmin) \
250                 FIELD_VEC(fld, angles) \
251                 FIELD_VEC(fld, avelocity) \
252                 FIELD_VEC(fld, color) \
253                 FIELD_VEC(fld, mangle) \
254                 FIELD_VEC(fld, maxs) \
255                 FIELD_VEC(fld, maxs) \
256                 FIELD_VEC(fld, mins) \
257                 FIELD_VEC(fld, modelscale_vec) \
258                 FIELD_VEC(fld, velocity) \
259                 /**/
260
261         #define spawnfunc(...) EVAL_spawnfunc(OVERLOAD(spawnfunc, __VA_ARGS__))
262         #define EVAL_spawnfunc(...) __VA_ARGS__