… | |
… | |
29 | #include <XSUB.h> |
29 | #include <XSUB.h> |
30 | |
30 | |
31 | #undef save_long // clashes with libproto.h |
31 | #undef save_long // clashes with libproto.h |
32 | |
32 | |
33 | #define PLUGIN_NAME "perl" |
33 | #define PLUGIN_NAME "perl" |
34 | #define PLUGIN_VERSION "cfperl 0.1" |
34 | #define PLUGIN_VERSION "cfperl 0.2" |
35 | |
35 | |
36 | #ifndef __CEXTRACT__ |
36 | #ifndef __CEXTRACT__ |
37 | #include <plugin.h> |
37 | #include <plugin.h> |
38 | #endif |
38 | #endif |
39 | |
39 | |
… | |
… | |
58 | static f_plug_api systemDirectory; |
58 | static f_plug_api systemDirectory; |
59 | static f_plug_api object_set_property; |
59 | static f_plug_api object_set_property; |
60 | static f_plug_api map_get_map; |
60 | static f_plug_api map_get_map; |
61 | static f_plug_api object_insert; |
61 | static f_plug_api object_insert; |
62 | |
62 | |
|
|
63 | /* this is a stupid way to do things, and awkward to use for plug-in authors */ |
63 | typedef struct |
64 | typedef struct |
64 | { |
65 | { |
65 | object* who; |
66 | object* who; |
66 | object* activator; |
67 | object* activator; |
67 | object* third; |
68 | object* third; |
|
|
69 | mapstruct* map; |
68 | char message[1024]; |
70 | char message[1024]; |
69 | int fix; |
71 | int fix; // seems to be python-only, and should not be part of the API |
70 | int event_code; |
72 | int event_code; |
71 | char extension[1024]; // name field, should invoke specific perl extension |
73 | char extension[1024]; // name field, should invoke specific perl extension |
72 | char options[1024]; // slaying field of event_connectors |
74 | char options[1024]; // slaying field of event_connectors |
73 | int returnvalue; |
75 | int returnvalue; |
74 | } CFPContext; |
76 | } CFPContext; |
75 | |
|
|
76 | //static int current_command = -999; |
|
|
77 | |
77 | |
78 | static HV *obj_cache; |
78 | static HV *obj_cache; |
79 | static PerlInterpreter *perl; |
79 | static PerlInterpreter *perl; |
80 | |
80 | |
81 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
81 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
… | |
… | |
127 | sv_magic (sv, 0, PERL_MAGIC_ext, (char *)ptr, 0); |
127 | sv_magic (sv, 0, PERL_MAGIC_ext, (char *)ptr, 0); |
128 | return sv_bless (newRV_noinc (sv), gv_stashpv (klass, 1)); |
128 | return sv_bless (newRV_noinc (sv), gv_stashpv (klass, 1)); |
129 | } |
129 | } |
130 | |
130 | |
131 | static SV * |
131 | static SV * |
|
|
132 | SVptr_cached (void *ptr) |
|
|
133 | { |
|
|
134 | SV **he = hv_fetch (obj_cache, (char *)&ptr, sizeof (ptr), 0); |
|
|
135 | |
|
|
136 | return he ? *he : 0; |
|
|
137 | } |
|
|
138 | |
|
|
139 | static SV * |
132 | newSVptr_cached (void *ptr, const char *klass) |
140 | newSVptr_cached (void *ptr, const char *klass) |
133 | { |
141 | { |
134 | SV *sv, **he; |
142 | SV *sv; |
135 | |
143 | |
136 | if (!ptr) |
144 | if (!ptr) |
137 | return &PL_sv_undef; |
145 | return &PL_sv_undef; |
138 | |
146 | |
139 | he = hv_fetch (obj_cache, (char *)&ptr, sizeof (ptr), 0); |
147 | sv = SVptr_cached (ptr); |
140 | |
148 | |
141 | if (he) |
149 | if (!sv) |
142 | sv = *he; |
|
|
143 | else |
|
|
144 | { |
150 | { |
145 | HV *hv = newHV (); |
151 | HV *hv = newHV (); |
146 | sv_magic ((SV *)hv, 0, PERL_MAGIC_ext, (char *)ptr, 0); |
152 | sv_magic ((SV *)hv, 0, PERL_MAGIC_ext, (char *)ptr, 0); |
147 | sv = sv_bless (newRV_noinc ((SV *)hv), gv_stashpv (klass, 1)); |
153 | sv = sv_bless (newRV_noinc ((SV *)hv), gv_stashpv (klass, 1)); |
148 | hv_store (obj_cache, (char *)&ptr, sizeof (ptr), sv, 0); |
154 | hv_store (obj_cache, (char *)&ptr, sizeof (ptr), sv, 0); |
… | |
… | |
192 | va_list args; |
198 | va_list args; |
193 | va_start (args, type); |
199 | va_start (args, type); |
194 | |
200 | |
195 | switch (type) |
201 | switch (type) |
196 | { |
202 | { |
|
|
203 | #if 0 |
|
|
204 | case CFAPI_INT16: |
|
|
205 | sv = newSViv (*va_arg (args, sint16_t *)); |
|
|
206 | break; |
|
|
207 | #endif |
|
|
208 | |
197 | case CFAPI_INT: |
209 | case CFAPI_INT: |
198 | sv = newSViv (*va_arg (args, int *)); |
210 | sv = newSViv (*va_arg (args, int *)); |
199 | break; |
211 | break; |
200 | |
212 | |
201 | case CFAPI_LONG: |
213 | case CFAPI_LONG: |
… | |
… | |
281 | HV *hv = newHV (); |
293 | HV *hv = newHV (); |
282 | #define hv_context(type,addr,expr) hv_store (hv, #expr, sizeof (#expr) - 1, newSVcfapi (type, addr context->expr), 0) |
294 | #define hv_context(type,addr,expr) hv_store (hv, #expr, sizeof (#expr) - 1, newSVcfapi (type, addr context->expr), 0) |
283 | hv_context (CFAPI_POBJECT, ,who); |
295 | hv_context (CFAPI_POBJECT, ,who); |
284 | hv_context (CFAPI_POBJECT, ,activator); |
296 | hv_context (CFAPI_POBJECT, ,activator); |
285 | hv_context (CFAPI_POBJECT, ,third); |
297 | hv_context (CFAPI_POBJECT, ,third); |
|
|
298 | hv_context (CFAPI_PMAP, ,map); |
286 | hv_context (CFAPI_STRING , ,message); |
299 | hv_context (CFAPI_STRING , ,message); |
287 | hv_context (CFAPI_INT ,&,fix); |
300 | hv_context (CFAPI_INT ,&,fix); |
288 | hv_context (CFAPI_INT ,&,event_code); |
301 | hv_context (CFAPI_INT ,&,event_code); |
289 | hv_context (CFAPI_STRING , ,options); |
302 | hv_context (CFAPI_STRING , ,options); |
290 | hv_context (CFAPI_STRING , ,extension); |
303 | hv_context (CFAPI_STRING , ,extension); |
… | |
… | |
426 | registerGlobalEvent (NULL, EVENT_LOGIN, PLUGIN_NAME, globalEventListener); |
439 | registerGlobalEvent (NULL, EVENT_LOGIN, PLUGIN_NAME, globalEventListener); |
427 | registerGlobalEvent (NULL, EVENT_LOGOUT, PLUGIN_NAME, globalEventListener); |
440 | registerGlobalEvent (NULL, EVENT_LOGOUT, PLUGIN_NAME, globalEventListener); |
428 | registerGlobalEvent (NULL, EVENT_MAPENTER, PLUGIN_NAME, globalEventListener); |
441 | registerGlobalEvent (NULL, EVENT_MAPENTER, PLUGIN_NAME, globalEventListener); |
429 | registerGlobalEvent (NULL, EVENT_MAPLEAVE, PLUGIN_NAME, globalEventListener); |
442 | registerGlobalEvent (NULL, EVENT_MAPLEAVE, PLUGIN_NAME, globalEventListener); |
430 | registerGlobalEvent (NULL, EVENT_MAPRESET, PLUGIN_NAME, globalEventListener); |
443 | registerGlobalEvent (NULL, EVENT_MAPRESET, PLUGIN_NAME, globalEventListener); |
|
|
444 | registerGlobalEvent (NULL, EVENT_MAPLOAD, PLUGIN_NAME, globalEventListener); |
|
|
445 | registerGlobalEvent (NULL, EVENT_MAPOUT, PLUGIN_NAME, globalEventListener); |
|
|
446 | registerGlobalEvent (NULL, EVENT_MAPIN, PLUGIN_NAME, globalEventListener); |
431 | registerGlobalEvent (NULL, EVENT_REMOVE, PLUGIN_NAME, globalEventListener); |
447 | registerGlobalEvent (NULL, EVENT_REMOVE, PLUGIN_NAME, globalEventListener); |
432 | registerGlobalEvent (NULL, EVENT_SHOUT, PLUGIN_NAME, globalEventListener); |
448 | registerGlobalEvent (NULL, EVENT_SHOUT, PLUGIN_NAME, globalEventListener); |
433 | registerGlobalEvent (NULL, EVENT_TELL, PLUGIN_NAME, globalEventListener); |
449 | registerGlobalEvent (NULL, EVENT_TELL, PLUGIN_NAME, globalEventListener); |
434 | registerGlobalEvent (NULL, EVENT_MUZZLE, PLUGIN_NAME, globalEventListener); |
450 | registerGlobalEvent (NULL, EVENT_MUZZLE, PLUGIN_NAME, globalEventListener); |
435 | registerGlobalEvent (NULL, EVENT_KICK, PLUGIN_NAME, globalEventListener); |
451 | registerGlobalEvent (NULL, EVENT_KICK, PLUGIN_NAME, globalEventListener); |
… | |
… | |
487 | { |
503 | { |
488 | case EVENT_CRASH: |
504 | case EVENT_CRASH: |
489 | printf ("Unimplemented for now\n"); |
505 | printf ("Unimplemented for now\n"); |
490 | break; |
506 | break; |
491 | |
507 | |
|
|
508 | case EVENT_MAPLOAD: |
|
|
509 | case EVENT_MAPOUT: |
|
|
510 | case EVENT_MAPIN: |
|
|
511 | context.map = va_arg (args, mapstruct *); |
|
|
512 | break; |
|
|
513 | |
492 | case EVENT_MAPENTER: |
514 | case EVENT_MAPENTER: |
493 | case EVENT_MAPLEAVE: |
515 | case EVENT_MAPLEAVE: |
494 | case EVENT_FREE_OB: |
516 | case EVENT_FREE_OB: |
495 | case EVENT_BORN: |
517 | case EVENT_BORN: |
496 | case EVENT_REMOVE: |
518 | case EVENT_REMOVE: |
… | |
… | |
530 | |
552 | |
531 | case EVENT_TELL: |
553 | case EVENT_TELL: |
532 | break; |
554 | break; |
533 | |
555 | |
534 | case EVENT_MAPRESET: |
556 | case EVENT_MAPRESET: |
|
|
557 | /* stupid, should be the map itself, not "message"??? */ |
535 | buf = va_arg (args, char *); |
558 | buf = va_arg (args, char *); |
536 | if (buf != 0) |
559 | if (buf != 0) |
537 | strncpy (context.message, buf, sizeof (context.message)); |
560 | strncpy (context.message, buf, sizeof (context.message)); |
538 | break; |
561 | break; |
539 | } |
562 | } |
… | |
… | |
750 | const_iv (ST_BD_REMOVE) |
773 | const_iv (ST_BD_REMOVE) |
751 | const_iv (ST_MAT_FLOOR) |
774 | const_iv (ST_MAT_FLOOR) |
752 | const_iv (ST_MAT_WALL) |
775 | const_iv (ST_MAT_WALL) |
753 | const_iv (ST_MAT_ITEM) |
776 | const_iv (ST_MAT_ITEM) |
754 | |
777 | |
|
|
778 | const_iv (AT_PHYSICAL) |
|
|
779 | const_iv (AT_MAGIC) |
|
|
780 | const_iv (AT_FIRE) |
|
|
781 | const_iv (AT_ELECTRICITY) |
|
|
782 | const_iv (AT_COLD) |
|
|
783 | const_iv (AT_CONFUSION) |
|
|
784 | const_iv (AT_ACID) |
|
|
785 | const_iv (AT_DRAIN) |
|
|
786 | const_iv (AT_WEAPONMAGIC) |
|
|
787 | const_iv (AT_GHOSTHIT) |
|
|
788 | const_iv (AT_POISON) |
|
|
789 | const_iv (AT_SLOW) |
|
|
790 | const_iv (AT_PARALYZE) |
|
|
791 | const_iv (AT_TURN_UNDEAD) |
|
|
792 | const_iv (AT_FEAR) |
|
|
793 | const_iv (AT_CANCELLATION) |
|
|
794 | const_iv (AT_DEPLETE) |
|
|
795 | const_iv (AT_DEATH) |
|
|
796 | const_iv (AT_CHAOS) |
|
|
797 | const_iv (AT_COUNTERSPELL) |
|
|
798 | const_iv (AT_GODPOWER) |
|
|
799 | const_iv (AT_HOLYWORD) |
|
|
800 | const_iv (AT_BLIND) |
|
|
801 | const_iv (AT_INTERNAL) |
|
|
802 | const_iv (AT_LIFE_STEALING) |
|
|
803 | const_iv (AT_DISEASE) |
|
|
804 | |
755 | const_iv (QUEST_IN_PROGRESS) |
805 | const_iv (QUEST_IN_PROGRESS) |
756 | const_iv (QUEST_DONE_QUEST) |
806 | const_iv (QUEST_DONE_QUEST) |
757 | const_iv (QUEST_DONE_TASK) |
807 | const_iv (QUEST_DONE_TASK) |
758 | const_iv (QUEST_START_QUEST) |
808 | const_iv (QUEST_START_QUEST) |
759 | const_iv (QUEST_END_QUEST) |
809 | const_iv (QUEST_END_QUEST) |
… | |
… | |
961 | const_event (LOGIN) |
1011 | const_event (LOGIN) |
962 | const_event (LOGOUT) |
1012 | const_event (LOGOUT) |
963 | const_event (MAPENTER) |
1013 | const_event (MAPENTER) |
964 | const_event (MAPLEAVE) |
1014 | const_event (MAPLEAVE) |
965 | const_event (MAPRESET) |
1015 | const_event (MAPRESET) |
|
|
1016 | const_event (MAPLOAD) |
|
|
1017 | const_event (MAPOUT) |
|
|
1018 | const_event (MAPIN) |
966 | const_event (REMOVE) |
1019 | const_event (REMOVE) |
967 | const_event (SHOUT) |
1020 | const_event (SHOUT) |
968 | const_event (TELL) |
1021 | const_event (TELL) |
969 | const_event (MUZZLE) |
1022 | const_event (MUZZLE) |
970 | const_event (KICK) |
1023 | const_event (KICK) |
… | |
… | |
1170 | int unused_type; |
1223 | int unused_type; |
1171 | object_set_property (&unused_type, obj, idx, (double)SvNV (newval)); |
1224 | object_set_property (&unused_type, obj, idx, (double)SvNV (newval)); |
1172 | } |
1225 | } |
1173 | break; |
1226 | break; |
1174 | case CFAPI_STRING: |
1227 | case CFAPI_STRING: |
1175 | cf_object_set_string_property (obj, idx, SvPV_nolen (newval)); |
1228 | cf_object_set_string_property (obj, idx, SvOK (newval) ? SvPV_nolen (newval) : 0); |
|
|
1229 | break; |
|
|
1230 | case CFAPI_POBJECT: |
|
|
1231 | { |
|
|
1232 | int unused_type; |
|
|
1233 | object_set_property (&unused_type, obj, idx, (object *)SvPTR_ornull (newval, "cf::object")); |
|
|
1234 | } |
1176 | break; |
1235 | break; |
1177 | default: |
1236 | default: |
1178 | croak ("unhandled type '%d' in set_property '%d'", type, idx); |
1237 | croak ("unhandled type '%d' in set_property '%d'", type, idx); |
1179 | } |
1238 | } |
1180 | |
1239 | |
|
|
1240 | # missing properties |
|
|
1241 | |
|
|
1242 | void |
|
|
1243 | set_attacktype (object *obj, U32 attacktype) |
|
|
1244 | CODE: |
|
|
1245 | obj->attacktype = attacktype; |
|
|
1246 | |
|
|
1247 | U32 |
|
|
1248 | get_attacktype (object *obj) |
|
|
1249 | ALIAS: |
|
|
1250 | attacktype = 0 |
|
|
1251 | CODE: |
|
|
1252 | RETVAL = obj->attacktype; |
|
|
1253 | OUTPUT: RETVAL |
|
|
1254 | |
|
|
1255 | void |
|
|
1256 | set_food (object *obj, int food) |
|
|
1257 | CODE: |
|
|
1258 | obj->stats.food = food; |
|
|
1259 | |
|
|
1260 | int |
|
|
1261 | get_food (object *obj) |
|
|
1262 | ALIAS: |
|
|
1263 | food = 0 |
|
|
1264 | CODE: |
|
|
1265 | RETVAL = obj->stats.food; |
|
|
1266 | OUTPUT: RETVAL |
|
|
1267 | |
1181 | void |
1268 | void |
1182 | inv (object *obj) |
1269 | inv (object *obj) |
1183 | PROTOTYPE: $ |
1270 | PROTOTYPE: $ |
1184 | PPCODE: |
1271 | PPCODE: |
1185 | { |
1272 | { |
… | |
… | |
1216 | |
1303 | |
1217 | int cf_object_pay_item (object *op, object *buyer) |
1304 | int cf_object_pay_item (object *op, object *buyer) |
1218 | |
1305 | |
1219 | int cf_object_pay_amount (object *op, double amount) |
1306 | int cf_object_pay_amount (object *op, double amount) |
1220 | |
1307 | |
1221 | int cf_object_cast_spell (object *caster, object *ctoo, int dir, object *sp_, char *flags) |
1308 | int cf_object_cast_spell (object *caster, object *ctoo, int dir, object *spell_ob, char *stringarg = 0) |
1222 | |
1309 | |
1223 | int cf_object_cast_ability (object *caster, object *ctoo, int dir, object *sp_, char *flags) |
1310 | int cf_object_cast_ability (object *caster, object *ctoo, int dir, object *sp_, char *stringarg = 0) |
1224 | |
1311 | |
1225 | void cf_object_learn_spell (object *op, object *sp) |
1312 | void cf_object_learn_spell (object *op, object *sp) |
1226 | |
1313 | |
1227 | void cf_object_forget_spell (object *op, object *sp) |
1314 | void cf_object_forget_spell (object *op, object *sp) |
1228 | |
1315 | |
… | |
… | |
1264 | char *cf_object_get_key (object *op, char *keyname) |
1351 | char *cf_object_get_key (object *op, char *keyname) |
1265 | ALIAS: key = 0 |
1352 | ALIAS: key = 0 |
1266 | |
1353 | |
1267 | void cf_object_set_key (object *op, char *keyname, char *value) |
1354 | void cf_object_set_key (object *op, char *keyname, char *value) |
1268 | |
1355 | |
|
|
1356 | object *cf_create_object_by_name (const char *name) |
|
|
1357 | |
1269 | MODULE = cf PACKAGE = cf::object PREFIX = cf_ |
1358 | MODULE = cf PACKAGE = cf::object PREFIX = cf_ |
1270 | |
1359 | |
1271 | void cf_fix_object (object *pl) |
1360 | void cf_fix_object (object *pl) |
1272 | ALIAS: fix = 0 |
1361 | ALIAS: fix = 0 |
1273 | |
1362 | |
1274 | object *cf_insert_ob_in_ob (object *ob, object *where) |
1363 | object *cf_insert_ob_in_ob (object *ob, object *where) |
1275 | |
1364 | |
1276 | object *create_object_by_name (const char *name = 0) |
1365 | # no clean way to get an object from an archetype - stupid idiotic |
|
|
1366 | # dumb kludgy misdesigned plug-in api slowly gets on my nerves. |
|
|
1367 | |
|
|
1368 | object *new (const char *archetype = 0) |
1277 | PROTOTYPE: ;$ |
1369 | PROTOTYPE: ;$ |
1278 | ALIAS: |
|
|
1279 | create_object = 0 |
|
|
1280 | new = 0 |
|
|
1281 | CODE: |
1370 | CODE: |
1282 | RETVAL = name ? cf_create_object_by_name (name) : cf_create_object (); |
1371 | RETVAL = archetype ? get_archetype (archetype) : cf_create_object (); |
1283 | OUTPUT: |
1372 | OUTPUT: |
1284 | RETVAL |
1373 | RETVAL |
1285 | |
1374 | |
1286 | object *insert_ob_in_map_at (object *ob, mapstruct *where, object_ornull *orig, int flag, int x, int y) |
1375 | object *insert_ob_in_map_at (object *ob, mapstruct *where, object_ornull *orig, int flag, int x, int y) |
1287 | PROTOTYPE: $$$$$$ |
1376 | PROTOTYPE: $$$$$$ |
… | |
… | |
1442 | INIT: |
1531 | INIT: |
1443 | if (x >= MAP_WIDTH (obj) || y >= MAP_HEIGHT (obj)) XSRETURN_EMPTY; |
1532 | if (x >= MAP_WIDTH (obj) || y >= MAP_HEIGHT (obj)) XSRETURN_EMPTY; |
1444 | PPCODE: |
1533 | PPCODE: |
1445 | { |
1534 | { |
1446 | object *o; |
1535 | object *o; |
|
|
1536 | |
1447 | for (o = GET_MAP_OB (obj, x, y); o; o = o->above) |
1537 | for (o = GET_MAP_OB (obj, x, y); o; o = o->above) |
1448 | XPUSHs (sv_2mortal (newSVcfapi (CFAPI_POBJECT, o))); |
1538 | XPUSHs (sv_2mortal (newSVcfapi (CFAPI_POBJECT, o))); |
1449 | } |
1539 | } |
1450 | |
1540 | |
1451 | SV * |
1541 | SV * |
… | |
… | |
1474 | case 7: RETVAL = newSVuv ( GET_MAP_MOVE_OFF (obj, x, y)); break; |
1564 | case 7: RETVAL = newSVuv ( GET_MAP_MOVE_OFF (obj, x, y)); break; |
1475 | } |
1565 | } |
1476 | OUTPUT: |
1566 | OUTPUT: |
1477 | RETVAL |
1567 | RETVAL |
1478 | |
1568 | |
|
|
1569 | void |
|
|
1570 | _get_obs (mapstruct *map) |
|
|
1571 | PPCODE: |
|
|
1572 | { |
|
|
1573 | object *o; |
|
|
1574 | int x, y; |
|
|
1575 | AV *obs = newAV (); |
|
|
1576 | int nonnull = 0; |
|
|
1577 | |
|
|
1578 | for (y = 0; y < MAP_HEIGHT (map); y++) |
|
|
1579 | for (x = 0; x < MAP_WIDTH (map); x++) |
|
|
1580 | { |
|
|
1581 | AV *av = newAV (); |
|
|
1582 | |
|
|
1583 | for (o = GET_MAP_OB (map, x, y); o; o = o->above) |
|
|
1584 | { |
|
|
1585 | SV *sv = SVptr_cached (o); |
|
|
1586 | |
|
|
1587 | if (sv && HvFILL (SvRV (sv))) |
|
|
1588 | { |
|
|
1589 | nonnull = 1; |
|
|
1590 | sv = newSVsv (sv); |
|
|
1591 | } |
|
|
1592 | else |
|
|
1593 | sv = &PL_sv_undef; |
|
|
1594 | |
|
|
1595 | av_push (av, sv); |
|
|
1596 | } |
|
|
1597 | |
|
|
1598 | av_store (obs, x + y * MAP_HEIGHT (map), newRV_noinc ((SV *)av)); |
|
|
1599 | } |
|
|
1600 | |
|
|
1601 | if (nonnull) |
|
|
1602 | XPUSHs (sv_2mortal (newRV_noinc ((SV *)obs))); |
|
|
1603 | else |
|
|
1604 | SvREFCNT_dec (obs); |
|
|
1605 | } |
1479 | |
1606 | |
1480 | MODULE = cf PACKAGE = cf::arch PREFIX = cf_archetype_ |
1607 | MODULE = cf PACKAGE = cf::arch PREFIX = cf_archetype_ |
1481 | |
1608 | |
1482 | archetype *cf_archetype_get_first() |
1609 | archetype *cf_archetype_get_first() |
1483 | PROTOTYPE: |
1610 | PROTOTYPE: |