1 | /*****************************************************************************/ |
1 | /*****************************************************************************/ |
2 | /* CrossFire, A Multiplayer game for the X Window System */ |
2 | /* CrossFire, A Multiplayer game for the X Window System */ |
3 | /* */ |
|
|
4 | /*****************************************************************************/ |
3 | /*****************************************************************************/ |
5 | |
4 | |
6 | /* |
5 | /* |
7 | * This code is placed under the GNU General Public Licence (GPL) |
6 | * This code is placed under the GNU General Public Licence (GPL) |
8 | * |
7 | * |
… | |
… | |
29 | #include <XSUB.h> |
28 | #include <XSUB.h> |
30 | |
29 | |
31 | #undef save_long // clashes with libproto.h |
30 | #undef save_long // clashes with libproto.h |
32 | |
31 | |
33 | #define PLUGIN_NAME "perl" |
32 | #define PLUGIN_NAME "perl" |
34 | #define PLUGIN_VERSION "cfperl 0.2" |
33 | #define PLUGIN_VERSION "cfperl 0.3" |
35 | |
34 | |
36 | #ifndef __CEXTRACT__ |
35 | #ifndef __CEXTRACT__ |
37 | #include <plugin.h> |
36 | #include <plugin.h> |
38 | #endif |
37 | #endif |
39 | |
38 | |
… | |
… | |
42 | #else |
41 | #else |
43 | #define MODULEAPI |
42 | #define MODULEAPI |
44 | #endif |
43 | #endif |
45 | |
44 | |
46 | #include <plugin_common.h> |
45 | #include <plugin_common.h> |
|
|
46 | #include <sounds.h> |
47 | |
47 | |
48 | #include <stdarg.h> |
48 | #include <stdarg.h> |
49 | |
49 | |
|
|
50 | //#include "EventAPI.h" |
50 | #include "perlxsi.c" |
51 | #include "perlxsi.c" |
|
|
52 | |
|
|
53 | extern sint64 *levels; // the experience table |
51 | |
54 | |
52 | typedef object object_ornull; |
55 | typedef object object_ornull; |
53 | typedef mapstruct mapstruct_ornull; |
56 | typedef mapstruct mapstruct_ornull; |
|
|
57 | |
|
|
58 | typedef double val64; |
|
|
59 | #define newSVval64 newSVnv |
|
|
60 | #define SvVAL64 SvNV |
54 | |
61 | |
55 | static f_plug_api gethook; |
62 | static f_plug_api gethook; |
56 | static f_plug_api registerGlobalEvent; |
63 | static f_plug_api registerGlobalEvent; |
57 | static f_plug_api unregisterGlobalEvent; |
64 | static f_plug_api unregisterGlobalEvent; |
58 | static f_plug_api systemDirectory; |
65 | static f_plug_api systemDirectory; |
… | |
… | |
64 | typedef struct |
71 | typedef struct |
65 | { |
72 | { |
66 | object* who; |
73 | object* who; |
67 | object* activator; |
74 | object* activator; |
68 | object* third; |
75 | object* third; |
|
|
76 | object* event; |
69 | mapstruct* map; |
77 | mapstruct* map; |
70 | char message[1024]; |
78 | char message[1024]; |
71 | int fix; // seems to be python-only, and should not be part of the API |
79 | int fix; // seems to be python-only, and should not be part of the API |
72 | int event_code; |
80 | int event_code; |
73 | char extension[1024]; // name field, should invoke specific perl extension |
81 | char extension[1024]; // name field, should invoke specific perl extension |
… | |
… | |
75 | int returnvalue; |
83 | int returnvalue; |
76 | } CFPContext; |
84 | } CFPContext; |
77 | |
85 | |
78 | static HV *obj_cache; |
86 | static HV *obj_cache; |
79 | static PerlInterpreter *perl; |
87 | static PerlInterpreter *perl; |
|
|
88 | |
|
|
89 | #define PUSHcfapi(type,value) PUSHs (sv_2mortal (newSVcfapi (CFAPI_ ## type, (value)))) |
|
|
90 | #define PUSHcfapi_va(type,ctype) PUSHcfapi (type, va_arg (args, ctype)) |
|
|
91 | #define PUSH_OB PUSHcfapi_va(POBJECT, object *) |
|
|
92 | #define PUSH_PL PUSHcfapi_va(PPLAYER, player *) |
|
|
93 | #define PUSH_MAP PUSHcfapi_va(PMAP, mapstruct *) |
|
|
94 | #define PUSH_PV PUSHcfapi_va(STRING, const char *) |
|
|
95 | #define PUSH_IV PUSHs (sv_2mortal (newSViv (va_arg (args, int)))) |
|
|
96 | |
|
|
97 | extern void pay_player(object *op, uint64 amount); |
|
|
98 | extern uint64 pay_player_arch(object *op, const char *arch, uint64 amount); |
80 | |
99 | |
81 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
100 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
82 | |
101 | |
83 | // garbage collect some perl objects, if possible |
102 | // garbage collect some perl objects, if possible |
84 | // all objects no longer referenced and empty are |
103 | // all objects no longer referenced and empty are |
… | |
… | |
216 | case CFAPI_INT: |
235 | case CFAPI_INT: |
217 | sv = newSViv (*va_arg (args, int *)); |
236 | sv = newSViv (*va_arg (args, int *)); |
218 | break; |
237 | break; |
219 | |
238 | |
220 | case CFAPI_LONG: |
239 | case CFAPI_LONG: |
221 | sv = newSViv (*va_arg (args, long *)); |
240 | sv = newSVval64 ((val64)*va_arg (args, sint64 *)); |
222 | break; |
241 | break; |
223 | |
242 | |
224 | case CFAPI_DOUBLE: |
243 | case CFAPI_DOUBLE: |
225 | sv = newSViv (*va_arg (args, double *)); |
244 | sv = newSVnv (*va_arg (args, double *)); |
226 | break; |
245 | break; |
227 | |
246 | |
228 | case CFAPI_STRING: |
247 | case CFAPI_STRING: |
229 | { |
248 | { |
230 | char *str = va_arg (args, char *); |
249 | char *str = va_arg (args, char *); |
… | |
… | |
237 | object *obj = va_arg (args, object *); |
256 | object *obj = va_arg (args, object *); |
238 | |
257 | |
239 | if (!obj) |
258 | if (!obj) |
240 | sv = &PL_sv_undef; |
259 | sv = &PL_sv_undef; |
241 | else |
260 | else |
242 | switch (*(int *)cf_object_get_property (obj, CFAPI_OBJECT_PROP_TYPE)) |
261 | switch (obj->type) |
243 | { |
262 | { |
244 | case MAP: |
263 | case MAP: |
245 | sv = newSVptr_cached (obj, "cf::object::map"); |
264 | sv = newSVptr_cached (obj, "cf::object::map"); |
246 | break; |
265 | break; |
247 | |
266 | |
… | |
… | |
283 | va_end (args); |
302 | va_end (args); |
284 | |
303 | |
285 | return sv; |
304 | return sv; |
286 | } |
305 | } |
287 | |
306 | |
288 | ///////////////////////////////////////////////////////////////////////////// |
|
|
289 | |
|
|
290 | void |
|
|
291 | inject_event (const char *func, CFPContext *context) |
|
|
292 | { |
|
|
293 | dSP; |
|
|
294 | |
|
|
295 | ENTER; |
|
|
296 | SAVETMPS; |
|
|
297 | |
|
|
298 | PUSHMARK (SP); |
|
|
299 | |
|
|
300 | HV *hv = newHV (); |
|
|
301 | #define hv_context(type,addr,expr) hv_store (hv, #expr, sizeof (#expr) - 1, newSVcfapi (type, addr context->expr), 0) |
|
|
302 | hv_context (CFAPI_POBJECT, ,who); |
|
|
303 | hv_context (CFAPI_POBJECT, ,activator); |
|
|
304 | hv_context (CFAPI_POBJECT, ,third); |
|
|
305 | hv_context (CFAPI_PMAP, ,map); |
|
|
306 | hv_context (CFAPI_STRING , ,message); |
|
|
307 | hv_context (CFAPI_INT ,&,fix); |
|
|
308 | hv_context (CFAPI_INT ,&,event_code); |
|
|
309 | hv_context (CFAPI_STRING , ,options); |
|
|
310 | hv_context (CFAPI_STRING , ,extension); |
|
|
311 | |
|
|
312 | XPUSHs (sv_2mortal (newRV_noinc ((SV *)hv))); |
|
|
313 | |
|
|
314 | PUTBACK; |
|
|
315 | int count = call_pv (func, G_SCALAR | G_EVAL); |
|
|
316 | SPAGAIN; |
|
|
317 | |
|
|
318 | if (SvTRUE (ERRSV)) |
|
|
319 | LOG (llevError, "event '%d' callback evaluation error: %s", context->event_code, SvPV_nolen (ERRSV)); |
|
|
320 | |
|
|
321 | context->returnvalue = count > 0 ? POPi : 0; |
|
|
322 | |
|
|
323 | PUTBACK; |
|
|
324 | FREETMPS; |
|
|
325 | LEAVE; |
|
|
326 | } |
|
|
327 | |
|
|
328 | ///////////////////////////////////////////////////////////////////////////// |
307 | ///////////////////////////////////////////////////////////////////////////// |
329 | |
308 | |
330 | int |
309 | int |
331 | initPlugin (const char *iversion, f_plug_api gethooksptr) |
310 | initPlugin (const char *iversion, f_plug_api gethooksptr) |
332 | { |
311 | { |
… | |
… | |
439 | |
418 | |
440 | /* Pick the global events you want to monitor from this plugin */ |
419 | /* Pick the global events you want to monitor from this plugin */ |
441 | registerGlobalEvent (NULL, EVENT_BORN, PLUGIN_NAME, globalEventListener); |
420 | registerGlobalEvent (NULL, EVENT_BORN, PLUGIN_NAME, globalEventListener); |
442 | registerGlobalEvent (NULL, EVENT_CLOCK, PLUGIN_NAME, globalEventListener); |
421 | registerGlobalEvent (NULL, EVENT_CLOCK, PLUGIN_NAME, globalEventListener); |
443 | //registerGlobalEvent (NULL, EVENT_CRASH, PLUGIN_NAME, globalEventListener); |
422 | //registerGlobalEvent (NULL, EVENT_CRASH, PLUGIN_NAME, globalEventListener); |
|
|
423 | registerGlobalEvent (NULL, EVENT_FIND_UNARMED_SKILL, PLUGIN_NAME, globalEventListener); |
|
|
424 | registerGlobalEvent (NULL, EVENT_PLAYER_USE_SKILL, PLUGIN_NAME, globalEventListener); |
|
|
425 | registerGlobalEvent (NULL, EVENT_MONSTER_USE_SKILL, PLUGIN_NAME, globalEventListener); |
444 | registerGlobalEvent (NULL, EVENT_PLAYER_DEATH, PLUGIN_NAME, globalEventListener); |
426 | registerGlobalEvent (NULL, EVENT_PLAYER_DEATH, PLUGIN_NAME, globalEventListener); |
445 | registerGlobalEvent (NULL, EVENT_GKILL, PLUGIN_NAME, globalEventListener); |
427 | registerGlobalEvent (NULL, EVENT_GKILL, PLUGIN_NAME, globalEventListener); |
446 | registerGlobalEvent (NULL, EVENT_LOGIN, PLUGIN_NAME, globalEventListener); |
428 | registerGlobalEvent (NULL, EVENT_LOGIN, PLUGIN_NAME, globalEventListener); |
447 | registerGlobalEvent (NULL, EVENT_LOGOUT, PLUGIN_NAME, globalEventListener); |
429 | registerGlobalEvent (NULL, EVENT_LOGOUT, PLUGIN_NAME, globalEventListener); |
448 | registerGlobalEvent (NULL, EVENT_MAPENTER, PLUGIN_NAME, globalEventListener); |
430 | registerGlobalEvent (NULL, EVENT_MAPENTER, PLUGIN_NAME, globalEventListener); |
… | |
… | |
456 | registerGlobalEvent (NULL, EVENT_SHOUT, PLUGIN_NAME, globalEventListener); |
438 | registerGlobalEvent (NULL, EVENT_SHOUT, PLUGIN_NAME, globalEventListener); |
457 | registerGlobalEvent (NULL, EVENT_TELL, PLUGIN_NAME, globalEventListener); |
439 | registerGlobalEvent (NULL, EVENT_TELL, PLUGIN_NAME, globalEventListener); |
458 | registerGlobalEvent (NULL, EVENT_MUZZLE, PLUGIN_NAME, globalEventListener); |
440 | registerGlobalEvent (NULL, EVENT_MUZZLE, PLUGIN_NAME, globalEventListener); |
459 | registerGlobalEvent (NULL, EVENT_KICK, PLUGIN_NAME, globalEventListener); |
441 | registerGlobalEvent (NULL, EVENT_KICK, PLUGIN_NAME, globalEventListener); |
460 | registerGlobalEvent (NULL, EVENT_FREE_OB, PLUGIN_NAME, globalEventListener); |
442 | registerGlobalEvent (NULL, EVENT_FREE_OB, PLUGIN_NAME, globalEventListener); |
|
|
443 | registerGlobalEvent (NULL, EVENT_PLAYER_LOAD, PLUGIN_NAME, globalEventListener); |
|
|
444 | registerGlobalEvent (NULL, EVENT_PLAYER_SAVE, PLUGIN_NAME, globalEventListener); |
|
|
445 | registerGlobalEvent (NULL, EVENT_EXTCMD, PLUGIN_NAME, globalEventListener); |
461 | |
446 | |
462 | char *argv[] = { |
447 | char *argv[] = { |
463 | "", |
448 | "", |
464 | "-e" |
449 | "-e" |
465 | "BEGIN {" |
450 | "BEGIN {" |
… | |
… | |
471 | }; |
456 | }; |
472 | |
457 | |
473 | perl = perl_alloc (); |
458 | perl = perl_alloc (); |
474 | perl_construct (perl); |
459 | perl_construct (perl); |
475 | |
460 | |
|
|
461 | PL_exit_flags |= PERL_EXIT_DESTRUCT_END; |
|
|
462 | |
476 | if (perl_parse (perl, xs_init, 2, argv, (char **)NULL) || perl_run (perl)) |
463 | if (perl_parse (perl, xs_init, 2, argv, (char **)NULL) || perl_run (perl)) |
477 | { |
464 | { |
478 | printf ("unable to initialize perl-interpreter, continuing without.\n"); |
465 | printf ("unable to initialize perl-interpreter, continuing without.\n"); |
479 | |
466 | |
480 | perl_destruct (perl); |
467 | perl_destruct (perl); |
… | |
… | |
491 | |
478 | |
492 | void * |
479 | void * |
493 | globalEventListener (int *type, ...) |
480 | globalEventListener (int *type, ...) |
494 | { |
481 | { |
495 | va_list args; |
482 | va_list args; |
496 | static int rv = 0; |
483 | static int rv; |
497 | CFPContext context; |
484 | int event_code; |
498 | char *buf; |
|
|
499 | player *pl; |
|
|
500 | object *op; |
|
|
501 | |
485 | |
502 | if (!perl) |
486 | if (!perl) |
503 | return; |
487 | return; |
504 | |
488 | |
505 | memset (&context, 0, sizeof (context)); |
|
|
506 | |
|
|
507 | va_start (args, type); |
489 | va_start (args, type); |
508 | context.event_code = va_arg (args, int); |
490 | event_code = va_arg (args, int); |
509 | |
491 | |
510 | switch (context.event_code) |
492 | if (event_code == EVENT_FREE_OB) |
511 | { |
493 | { |
512 | case EVENT_CRASH: |
494 | player *pl; |
513 | printf ("Unimplemented for now\n"); |
495 | object *op; |
514 | break; |
496 | SV *sv; |
515 | |
497 | |
516 | case EVENT_MAPLOAD: |
|
|
517 | case EVENT_MAPOUT: |
|
|
518 | case EVENT_MAPIN: |
|
|
519 | case EVENT_MAPCLEAN: |
|
|
520 | context.map = va_arg (args, mapstruct *); |
|
|
521 | break; |
|
|
522 | |
|
|
523 | case EVENT_MAPENTER: |
|
|
524 | case EVENT_MAPLEAVE: |
|
|
525 | case EVENT_FREE_OB: |
|
|
526 | case EVENT_BORN: |
|
|
527 | case EVENT_REMOVE: |
|
|
528 | context.activator = va_arg (args, object *); |
|
|
529 | break; |
|
|
530 | |
|
|
531 | case EVENT_PLAYER_DEATH: |
|
|
532 | context.who = va_arg (args, object *); |
498 | op = va_arg (args, object *); |
533 | break; |
|
|
534 | |
|
|
535 | case EVENT_GKILL: |
|
|
536 | context.who = va_arg (args, object *); |
|
|
537 | context.activator = va_arg (args, object *); |
|
|
538 | break; |
|
|
539 | |
|
|
540 | case EVENT_LOGIN: |
|
|
541 | case EVENT_LOGOUT: |
|
|
542 | pl = va_arg (args, player *); |
|
|
543 | context.activator = pl->ob; |
|
|
544 | buf = va_arg (args, char *); |
|
|
545 | if (buf != 0) |
|
|
546 | strncpy (context.message, buf, sizeof (context.message)); |
|
|
547 | break; |
|
|
548 | |
|
|
549 | case EVENT_SHOUT: |
|
|
550 | case EVENT_MUZZLE: |
|
|
551 | case EVENT_KICK: |
|
|
552 | context.activator = va_arg (args, object *); |
|
|
553 | buf = va_arg (args, char *); |
|
|
554 | if (buf != 0) |
|
|
555 | strncpy (context.message, buf, sizeof (context.message)); |
|
|
556 | break; |
|
|
557 | |
|
|
558 | case EVENT_CLOCK: |
|
|
559 | clean_obj_cache (); |
|
|
560 | break; |
|
|
561 | |
|
|
562 | case EVENT_TELL: |
|
|
563 | break; |
|
|
564 | |
|
|
565 | case EVENT_MAPRESET: |
|
|
566 | /* stupid, should be the map itself, not "message"??? */ |
|
|
567 | buf = va_arg (args, char *); |
|
|
568 | if (buf != 0) |
|
|
569 | strncpy (context.message, buf, sizeof (context.message)); |
|
|
570 | break; |
|
|
571 | } |
|
|
572 | |
|
|
573 | va_end (args); |
|
|
574 | |
|
|
575 | if (context.event_code == EVENT_FREE_OB) |
|
|
576 | { |
|
|
577 | SV *sv = hv_delete (obj_cache, (char *)&context.activator, sizeof (void *), 0); |
499 | sv = hv_delete (obj_cache, (char *)&op, sizeof (void *), 0); |
578 | |
500 | |
579 | if (sv) |
501 | if (sv) |
580 | clearSVptr (sv); |
502 | clearSVptr (sv); |
|
|
503 | |
|
|
504 | rv = 0; |
|
|
505 | } |
|
|
506 | else if (event_code == EVENT_CLOCK) |
|
|
507 | { |
|
|
508 | dSP; |
|
|
509 | int i, count; |
|
|
510 | |
|
|
511 | clean_obj_cache (); |
|
|
512 | |
|
|
513 | ENTER; |
|
|
514 | SAVETMPS; |
|
|
515 | |
|
|
516 | // service up to 8 events per tick better would be |
|
|
517 | // to check for elapsed time and stop processing after |
|
|
518 | // 0.25 * server_tick or so |
|
|
519 | for (i = 9; --i; ) |
|
|
520 | { |
|
|
521 | PUSHMARK (SP); |
|
|
522 | XPUSHs (sv_2mortal (newSViv (0))); |
|
|
523 | PUTBACK; |
|
|
524 | count = call_pv ("Event::one_event", G_SCALAR | G_EVAL); |
|
|
525 | SPAGAIN; |
|
|
526 | |
|
|
527 | if (!count || !POPi) |
|
|
528 | break; |
|
|
529 | } |
|
|
530 | |
|
|
531 | FREETMPS; |
|
|
532 | LEAVE; |
581 | } |
533 | } |
582 | else |
534 | else |
583 | inject_event ("cf::inject_global_event", &context); |
535 | { |
584 | |
536 | dSP; |
585 | rv = context.returnvalue; |
537 | |
|
|
538 | ENTER; |
|
|
539 | SAVETMPS; |
|
|
540 | |
|
|
541 | PUSHMARK (SP); |
|
|
542 | |
|
|
543 | EXTEND (SP, 10); |
|
|
544 | PUSHs (sv_2mortal (newSViv (event_code))); |
|
|
545 | |
|
|
546 | switch (event_code) |
|
|
547 | { |
|
|
548 | case EVENT_CRASH: |
|
|
549 | break; |
|
|
550 | |
|
|
551 | case EVENT_PLAYER_LOAD: |
|
|
552 | case EVENT_PLAYER_SAVE: |
|
|
553 | PUSH_OB; |
|
|
554 | PUSH_PV; |
|
|
555 | break; |
|
|
556 | |
|
|
557 | case EVENT_MAPLOAD: |
|
|
558 | case EVENT_MAPOUT: |
|
|
559 | case EVENT_MAPIN: |
|
|
560 | case EVENT_MAPCLEAN: |
|
|
561 | case EVENT_MAPRESET: |
|
|
562 | PUSH_MAP; |
|
|
563 | break; |
|
|
564 | |
|
|
565 | case EVENT_MAPENTER: |
|
|
566 | case EVENT_MAPLEAVE: |
|
|
567 | case EVENT_BORN: |
|
|
568 | case EVENT_REMOVE: |
|
|
569 | case EVENT_PLAYER_DEATH: |
|
|
570 | PUSH_OB; |
|
|
571 | break; |
|
|
572 | |
|
|
573 | case EVENT_GKILL: |
|
|
574 | PUSH_OB; |
|
|
575 | PUSH_OB; |
|
|
576 | break; |
|
|
577 | |
|
|
578 | case EVENT_LOGIN: |
|
|
579 | case EVENT_LOGOUT: |
|
|
580 | PUSH_PL; |
|
|
581 | PUSH_PV; |
|
|
582 | break; |
|
|
583 | |
|
|
584 | case EVENT_SHOUT: |
|
|
585 | case EVENT_MUZZLE: |
|
|
586 | case EVENT_KICK: |
|
|
587 | PUSH_OB; |
|
|
588 | PUSH_PV; |
|
|
589 | break; |
|
|
590 | |
|
|
591 | case EVENT_FIND_UNARMED_SKILL: |
|
|
592 | PUSH_OB; |
|
|
593 | break; |
|
|
594 | |
|
|
595 | case EVENT_PLAYER_USE_SKILL: |
|
|
596 | case EVENT_MONSTER_USE_SKILL: |
|
|
597 | PUSH_OB; |
|
|
598 | PUSH_OB; |
|
|
599 | PUSH_OB; |
|
|
600 | PUSH_IV; |
|
|
601 | PUSH_PV; |
|
|
602 | break; |
|
|
603 | |
|
|
604 | case EVENT_EXTCMD: |
|
|
605 | PUSH_PL; |
|
|
606 | { |
|
|
607 | char *buf = va_arg (args, char *); |
|
|
608 | int len = va_arg (args, int); |
|
|
609 | PUSHs (sv_2mortal (newSVpvn (buf, len))); |
|
|
610 | } |
|
|
611 | break; |
|
|
612 | |
|
|
613 | case EVENT_TELL: |
|
|
614 | break; |
|
|
615 | } |
|
|
616 | |
|
|
617 | va_end (args); |
|
|
618 | |
|
|
619 | PUTBACK; |
|
|
620 | int count = call_pv ("cf::inject_global_event", G_SCALAR | G_EVAL); |
|
|
621 | SPAGAIN; |
|
|
622 | |
|
|
623 | if (SvTRUE (ERRSV)) |
|
|
624 | LOG (llevError, "global event '%d' callback evaluation error: %s", event_code, SvPV_nolen (ERRSV)); |
|
|
625 | |
|
|
626 | rv = count > 0 ? POPi : 0; |
|
|
627 | |
|
|
628 | PUTBACK; |
|
|
629 | FREETMPS; |
|
|
630 | LEAVE; |
|
|
631 | } |
586 | |
632 | |
587 | return &rv; |
633 | return &rv; |
588 | } |
634 | } |
589 | |
635 | |
590 | void * |
636 | void * |
591 | eventListener (int *type, ...) |
637 | eventListener (int *type, ...) |
592 | { |
638 | { |
593 | static int rv = 0; |
639 | static int rv; |
594 | va_list args; |
640 | va_list args; |
595 | char *buf; |
641 | int event_code; |
596 | CFPContext context; |
642 | object *who, *activator, *third, *event; |
|
|
643 | char *message, *extension, *options; |
597 | |
644 | |
598 | if (!perl) |
645 | if (!perl) |
599 | return; |
646 | return; |
600 | |
647 | |
601 | memset (&context, 0, sizeof (context)); |
|
|
602 | |
|
|
603 | va_start (args, type); |
648 | va_start (args, type); |
604 | |
|
|
605 | context.who = va_arg (args, object *); |
649 | who = va_arg (args, object *); |
606 | context.event_code = va_arg (args, int); |
650 | event_code = va_arg (args, int); |
607 | context.activator = va_arg (args, object *); |
651 | activator = va_arg (args, object *); |
608 | context.third = va_arg (args, object *); |
652 | third = va_arg (args, object *); |
609 | |
|
|
610 | buf = va_arg (args, char *); |
653 | message = va_arg (args, char *); |
611 | if (buf != 0) |
654 | va_arg (args, int); // fix yourself |
612 | strncpy (context.message, buf, sizeof (context.message)); |
655 | extension = va_arg (args, char *); |
613 | |
656 | options = va_arg (args, char *); |
614 | context.fix = va_arg (args, int); |
657 | event = va_arg (args, object *); |
615 | strncpy (context.extension, va_arg (args, char *), sizeof (context.extension)); |
|
|
616 | strncpy (context.options, va_arg (args, char *), sizeof (context.options)); |
|
|
617 | context.returnvalue = 0; |
|
|
618 | va_end (args); |
658 | va_end (args); |
619 | |
659 | |
620 | inject_event ("cf::inject_event", &context); |
660 | { |
|
|
661 | dSP; |
|
|
662 | |
|
|
663 | ENTER; |
|
|
664 | SAVETMPS; |
|
|
665 | |
|
|
666 | PUSHMARK (SP); |
|
|
667 | EXTEND (SP, 10); |
|
|
668 | |
|
|
669 | PUSHcfapi (STRING, extension); |
|
|
670 | PUSHs (sv_2mortal (newSViv (event_code))); |
|
|
671 | |
|
|
672 | PUSHcfapi (POBJECT, event); |
|
|
673 | PUSHcfapi (POBJECT, who); |
|
|
674 | |
|
|
675 | switch (event_code) |
|
|
676 | { |
|
|
677 | case EVENT_STOP: // $ob (e.g. arrow) |
|
|
678 | case EVENT_TIME: // $ob |
|
|
679 | case EVENT_TIMER: // $ob |
|
|
680 | break; |
|
|
681 | |
|
|
682 | case EVENT_APPLY: // $ob, $who |
|
|
683 | case EVENT_DROP: // $ob, $who |
|
|
684 | case EVENT_CLOSE: // $ob, $who |
|
|
685 | case EVENT_DEATH: // $ob[, $killer] |
|
|
686 | case EVENT_MOVE: // $ob, $enemy |
|
|
687 | case EVENT_THROW: // $ob, $thrower |
|
|
688 | PUSHcfapi (POBJECT, activator); |
|
|
689 | break; |
|
|
690 | |
|
|
691 | case EVENT_ATTACK: // $ob, $who, $victim (?? please god enlighten me) |
|
|
692 | PUSHcfapi (POBJECT, activator); |
|
|
693 | PUSHcfapi (POBJECT, third); |
|
|
694 | break; |
|
|
695 | |
|
|
696 | case EVENT_TRIGGER: // $ob, $originator, [$victim], [$msg] |
|
|
697 | PUSHcfapi (POBJECT, activator); |
|
|
698 | PUSHcfapi (POBJECT, third); |
|
|
699 | PUSHcfapi (POBJECT, message); |
|
|
700 | break; |
|
|
701 | |
|
|
702 | case EVENT_SAY: // $ob, $who, $msg |
|
|
703 | PUSHcfapi (POBJECT, activator); |
|
|
704 | PUSHcfapi (STRING, message); |
|
|
705 | break; |
|
|
706 | |
|
|
707 | default: |
|
|
708 | LOG (llevError, "perl plugin called for unsupported event type %d", event_code); |
|
|
709 | break; |
|
|
710 | } |
|
|
711 | |
|
|
712 | PUTBACK; |
|
|
713 | int count = call_pv ("cf::inject_event", G_SCALAR | G_EVAL); |
|
|
714 | SPAGAIN; |
|
|
715 | |
|
|
716 | if (SvTRUE (ERRSV)) |
|
|
717 | LOG (llevError, "event '%d' callback evaluation error: %s", event_code, SvPV_nolen (ERRSV)); |
|
|
718 | |
|
|
719 | rv = count > 0 ? POPi : 0; |
|
|
720 | |
|
|
721 | PUTBACK; |
|
|
722 | FREETMPS; |
|
|
723 | LEAVE; |
|
|
724 | } |
621 | |
725 | |
622 | rv = context.returnvalue; |
|
|
623 | return &rv; |
726 | return &rv; |
624 | } |
727 | } |
625 | |
728 | |
626 | int |
729 | int |
627 | closePlugin () |
730 | closePlugin () |
… | |
… | |
653 | const_iv (llevInfo) |
756 | const_iv (llevInfo) |
654 | const_iv (llevDebug) |
757 | const_iv (llevDebug) |
655 | const_iv (llevMonster) |
758 | const_iv (llevMonster) |
656 | |
759 | |
657 | const_iv (PLAYER) |
760 | const_iv (PLAYER) |
|
|
761 | const_iv (TRANSPORT) |
658 | const_iv (ROD) |
762 | const_iv (ROD) |
659 | const_iv (TREASURE) |
763 | const_iv (TREASURE) |
660 | const_iv (POTION) |
764 | const_iv (POTION) |
661 | const_iv (FOOD) |
765 | const_iv (FOOD) |
662 | const_iv (POISON) |
766 | const_iv (POISON) |
… | |
… | |
667 | const_iv (BOW) |
771 | const_iv (BOW) |
668 | const_iv (WEAPON) |
772 | const_iv (WEAPON) |
669 | const_iv (ARMOUR) |
773 | const_iv (ARMOUR) |
670 | const_iv (PEDESTAL) |
774 | const_iv (PEDESTAL) |
671 | const_iv (ALTAR) |
775 | const_iv (ALTAR) |
672 | const_iv (CONFUSION) |
|
|
673 | const_iv (LOCKED_DOOR) |
776 | const_iv (LOCKED_DOOR) |
674 | const_iv (SPECIAL_KEY) |
777 | const_iv (SPECIAL_KEY) |
675 | const_iv (MAP) |
778 | const_iv (MAP) |
676 | const_iv (DOOR) |
779 | const_iv (DOOR) |
677 | const_iv (KEY) |
780 | const_iv (KEY) |
… | |
… | |
778 | const_iv (ITEM_TRANSFORMER) |
881 | const_iv (ITEM_TRANSFORMER) |
779 | const_iv (QUEST) |
882 | const_iv (QUEST) |
780 | |
883 | |
781 | const_iv (ST_BD_BUILD) |
884 | const_iv (ST_BD_BUILD) |
782 | const_iv (ST_BD_REMOVE) |
885 | const_iv (ST_BD_REMOVE) |
|
|
886 | |
783 | const_iv (ST_MAT_FLOOR) |
887 | const_iv (ST_MAT_FLOOR) |
784 | const_iv (ST_MAT_WALL) |
888 | const_iv (ST_MAT_WALL) |
785 | const_iv (ST_MAT_ITEM) |
889 | const_iv (ST_MAT_ITEM) |
786 | |
890 | |
787 | const_iv (AT_PHYSICAL) |
891 | const_iv (AT_PHYSICAL) |
… | |
… | |
988 | const_iv (WILL_APPLY_DOOR) |
1092 | const_iv (WILL_APPLY_DOOR) |
989 | const_iv (WILL_APPLY_FOOD) |
1093 | const_iv (WILL_APPLY_FOOD) |
990 | |
1094 | |
991 | const_iv (SAVE_MODE) |
1095 | const_iv (SAVE_MODE) |
992 | const_iv (SAVE_DIR_MODE) |
1096 | const_iv (SAVE_DIR_MODE) |
|
|
1097 | |
|
|
1098 | const_iv (M_PAPER) |
|
|
1099 | const_iv (M_IRON) |
|
|
1100 | const_iv (M_GLASS) |
|
|
1101 | const_iv (M_LEATHER) |
|
|
1102 | const_iv (M_WOOD) |
|
|
1103 | const_iv (M_ORGANIC) |
|
|
1104 | const_iv (M_STONE) |
|
|
1105 | const_iv (M_CLOTH) |
|
|
1106 | const_iv (M_ADAMANT) |
|
|
1107 | const_iv (M_LIQUID) |
|
|
1108 | const_iv (M_SOFT_METAL) |
|
|
1109 | const_iv (M_BONE) |
|
|
1110 | const_iv (M_ICE) |
|
|
1111 | const_iv (M_SPECIAL) |
|
|
1112 | |
|
|
1113 | const_iv (SK_EXP_ADD_SKILL) |
|
|
1114 | const_iv (SK_EXP_TOTAL) |
|
|
1115 | const_iv (SK_EXP_NONE) |
|
|
1116 | const_iv (SK_SUBTRACT_SKILL_EXP) |
|
|
1117 | |
|
|
1118 | const_iv (SK_LOCKPICKING) |
|
|
1119 | const_iv (SK_HIDING) |
|
|
1120 | const_iv (SK_SMITHERY) |
|
|
1121 | const_iv (SK_BOWYER) |
|
|
1122 | const_iv (SK_JEWELER) |
|
|
1123 | const_iv (SK_ALCHEMY) |
|
|
1124 | const_iv (SK_STEALING) |
|
|
1125 | const_iv (SK_LITERACY) |
|
|
1126 | const_iv (SK_BARGAINING) |
|
|
1127 | const_iv (SK_JUMPING) |
|
|
1128 | const_iv (SK_DET_MAGIC) |
|
|
1129 | const_iv (SK_ORATORY) |
|
|
1130 | const_iv (SK_SINGING) |
|
|
1131 | const_iv (SK_DET_CURSE) |
|
|
1132 | const_iv (SK_FIND_TRAPS) |
|
|
1133 | const_iv (SK_MEDITATION) |
|
|
1134 | const_iv (SK_PUNCHING) |
|
|
1135 | const_iv (SK_FLAME_TOUCH) |
|
|
1136 | const_iv (SK_KARATE) |
|
|
1137 | const_iv (SK_CLIMBING) |
|
|
1138 | const_iv (SK_WOODSMAN) |
|
|
1139 | const_iv (SK_INSCRIPTION) |
|
|
1140 | const_iv (SK_ONE_HANDED_WEAPON) |
|
|
1141 | const_iv (SK_MISSILE_WEAPON) |
|
|
1142 | const_iv (SK_THROWING) |
|
|
1143 | const_iv (SK_USE_MAGIC_ITEM) |
|
|
1144 | const_iv (SK_DISARM_TRAPS) |
|
|
1145 | const_iv (SK_SET_TRAP) |
|
|
1146 | const_iv (SK_THAUMATURGY) |
|
|
1147 | const_iv (SK_PRAYING) |
|
|
1148 | const_iv (SK_CLAWING) |
|
|
1149 | const_iv (SK_LEVITATION) |
|
|
1150 | const_iv (SK_SUMMONING) |
|
|
1151 | const_iv (SK_PYROMANCY) |
|
|
1152 | const_iv (SK_EVOCATION) |
|
|
1153 | const_iv (SK_SORCERY) |
|
|
1154 | const_iv (SK_TWO_HANDED_WEAPON) |
|
|
1155 | const_iv (SK_SPARK_TOUCH) |
|
|
1156 | const_iv (SK_SHIVER) |
|
|
1157 | const_iv (SK_ACID_SPLASH) |
|
|
1158 | const_iv (SK_POISON_NAIL) |
|
|
1159 | |
|
|
1160 | const_iv (SOUND_NEW_PLAYER) |
|
|
1161 | const_iv (SOUND_FIRE_ARROW) |
|
|
1162 | const_iv (SOUND_LEARN_SPELL) |
|
|
1163 | const_iv (SOUND_FUMBLE_SPELL) |
|
|
1164 | const_iv (SOUND_WAND_POOF) |
|
|
1165 | const_iv (SOUND_OPEN_DOOR) |
|
|
1166 | const_iv (SOUND_PUSH_PLAYER) |
|
|
1167 | const_iv (SOUND_PLAYER_HITS1) |
|
|
1168 | const_iv (SOUND_PLAYER_HITS2) |
|
|
1169 | const_iv (SOUND_PLAYER_HITS3) |
|
|
1170 | const_iv (SOUND_PLAYER_HITS4) |
|
|
1171 | const_iv (SOUND_PLAYER_IS_HIT1) |
|
|
1172 | const_iv (SOUND_PLAYER_IS_HIT2) |
|
|
1173 | const_iv (SOUND_PLAYER_IS_HIT3) |
|
|
1174 | const_iv (SOUND_PLAYER_KILLS) |
|
|
1175 | const_iv (SOUND_PET_IS_KILLED) |
|
|
1176 | const_iv (SOUND_PLAYER_DIES) |
|
|
1177 | const_iv (SOUND_OB_EVAPORATE) |
|
|
1178 | const_iv (SOUND_OB_EXPLODE) |
|
|
1179 | const_iv (SOUND_CLOCK) |
|
|
1180 | const_iv (SOUND_TURN_HANDLE) |
|
|
1181 | const_iv (SOUND_FALL_HOLE) |
|
|
1182 | const_iv (SOUND_DRINK_POISON) |
|
|
1183 | const_iv (SOUND_CAST_SPELL_0) |
|
|
1184 | |
|
|
1185 | const_iv (PREFER_LOW) |
|
|
1186 | const_iv (PREFER_HIGH) |
|
|
1187 | |
|
|
1188 | const_iv (ATNR_PHYSICAL) |
|
|
1189 | const_iv (ATNR_MAGIC) |
|
|
1190 | const_iv (ATNR_FIRE) |
|
|
1191 | const_iv (ATNR_ELECTRICITY) |
|
|
1192 | const_iv (ATNR_COLD) |
|
|
1193 | const_iv (ATNR_CONFUSION) |
|
|
1194 | const_iv (ATNR_ACID) |
|
|
1195 | const_iv (ATNR_DRAIN) |
|
|
1196 | const_iv (ATNR_WEAPONMAGIC) |
|
|
1197 | const_iv (ATNR_GHOSTHIT) |
|
|
1198 | const_iv (ATNR_POISON) |
|
|
1199 | const_iv (ATNR_SLOW) |
|
|
1200 | const_iv (ATNR_PARALYZE) |
|
|
1201 | const_iv (ATNR_TURN_UNDEAD) |
|
|
1202 | const_iv (ATNR_FEAR) |
|
|
1203 | const_iv (ATNR_CANCELLATION) |
|
|
1204 | const_iv (ATNR_DEPLETE) |
|
|
1205 | const_iv (ATNR_DEATH) |
|
|
1206 | const_iv (ATNR_CHAOS) |
|
|
1207 | const_iv (ATNR_COUNTERSPELL) |
|
|
1208 | const_iv (ATNR_GODPOWER) |
|
|
1209 | const_iv (ATNR_HOLYWORD) |
|
|
1210 | const_iv (ATNR_BLIND) |
|
|
1211 | const_iv (ATNR_INTERNAL) |
|
|
1212 | const_iv (ATNR_LIFE_STEALING) |
|
|
1213 | const_iv (ATNR_DISEASE) |
|
|
1214 | |
|
|
1215 | const_iv (MAP_FLUSH) |
|
|
1216 | const_iv (MAP_PLAYER_UNIQUE) |
|
|
1217 | const_iv (MAP_BLOCK) |
|
|
1218 | const_iv (MAP_STYLE) |
|
|
1219 | const_iv (MAP_OVERLAY) |
|
|
1220 | |
|
|
1221 | const_iv (MAP_IN_MEMORY) |
|
|
1222 | const_iv (MAP_SWAPPED) |
|
|
1223 | const_iv (MAP_LOADING) |
|
|
1224 | const_iv (MAP_SAVING) |
993 | }; |
1225 | }; |
994 | |
1226 | |
995 | for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; ) |
1227 | for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; ) |
996 | newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv)); |
1228 | newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv)); |
997 | |
1229 | |
… | |
… | |
1014 | const_event (CLOSE) |
1246 | const_event (CLOSE) |
1015 | const_event (TIMER) |
1247 | const_event (TIMER) |
1016 | const_event (MOVE) |
1248 | const_event (MOVE) |
1017 | |
1249 | |
1018 | const_event (BORN) |
1250 | const_event (BORN) |
1019 | const_event (CLOCK) |
1251 | //const_event (CLOCK) |
1020 | const_event (CRASH) |
1252 | const_event (CRASH) |
1021 | const_event (PLAYER_DEATH) |
1253 | const_event (PLAYER_DEATH) |
|
|
1254 | const_event (PLAYER_LOAD) |
|
|
1255 | const_event (PLAYER_SAVE) |
1022 | const_event (GKILL) |
1256 | const_event (GKILL) |
1023 | const_event (LOGIN) |
1257 | const_event (LOGIN) |
1024 | const_event (LOGOUT) |
1258 | const_event (LOGOUT) |
1025 | const_event (MAPENTER) |
1259 | const_event (MAPENTER) |
1026 | const_event (MAPLEAVE) |
1260 | const_event (MAPLEAVE) |
… | |
… | |
1032 | const_event (REMOVE) |
1266 | const_event (REMOVE) |
1033 | const_event (SHOUT) |
1267 | const_event (SHOUT) |
1034 | const_event (TELL) |
1268 | const_event (TELL) |
1035 | const_event (MUZZLE) |
1269 | const_event (MUZZLE) |
1036 | const_event (KICK) |
1270 | const_event (KICK) |
|
|
1271 | const_event (PLAYER_USE_SKILL) |
|
|
1272 | const_event (MONSTER_USE_SKILL) |
|
|
1273 | const_event (FIND_UNARMED_SKILL) |
|
|
1274 | const_event (EXTCMD) |
1037 | //const_event (FREE_OB) |
1275 | //const_event (FREE_OB) |
1038 | }; |
1276 | }; |
1039 | |
1277 | |
1040 | AV *av = get_av ("cf::EVENT", 1); |
1278 | AV *av = get_av ("cf::EVENT", 1); |
1041 | |
1279 | |
… | |
… | |
1139 | prop (CFAPI_INT, OBJECT_PROP_ANIM_SPEED) |
1377 | prop (CFAPI_INT, OBJECT_PROP_ANIM_SPEED) |
1140 | prop (CFAPI_INT, OBJECT_PROP_FRIENDLY) |
1378 | prop (CFAPI_INT, OBJECT_PROP_FRIENDLY) |
1141 | prop (CFAPI_STRING, OBJECT_PROP_SHORT_NAME) |
1379 | prop (CFAPI_STRING, OBJECT_PROP_SHORT_NAME) |
1142 | prop (CFAPI_INT, OBJECT_PROP_MAGICAL) |
1380 | prop (CFAPI_INT, OBJECT_PROP_MAGICAL) |
1143 | prop (CFAPI_INT, OBJECT_PROP_LUCK) |
1381 | prop (CFAPI_INT, OBJECT_PROP_LUCK) |
1144 | prop (CFAPI_LONG, OBJECT_PROP_EXP) |
|
|
1145 | prop (CFAPI_POBJECT, OBJECT_PROP_OWNER) |
1382 | prop (CFAPI_POBJECT, OBJECT_PROP_OWNER) |
1146 | prop (CFAPI_POBJECT, OBJECT_PROP_PRESENT) |
1383 | prop (CFAPI_POBJECT, OBJECT_PROP_PRESENT) |
1147 | prop (CFAPI_INT, OBJECT_PROP_CHEATER) |
1384 | prop (CFAPI_INT, OBJECT_PROP_CHEATER) |
1148 | prop (CFAPI_INT, OBJECT_PROP_MERGEABLE) |
1385 | prop (CFAPI_INT, OBJECT_PROP_MERGEABLE) |
1149 | prop (CFAPI_INT, OBJECT_PROP_PICKABLE) |
1386 | prop (CFAPI_INT, OBJECT_PROP_PICKABLE) |
… | |
… | |
1176 | for (cprop = prop_table + sizeof (prop_table) / sizeof (prop_table [0]); cprop-- > prop_table; ) |
1413 | for (cprop = prop_table + sizeof (prop_table) / sizeof (prop_table [0]); cprop-- > prop_table; ) |
1177 | { |
1414 | { |
1178 | hv_store (prop_type, cprop->name, strlen (cprop->name), newSViv (cprop->dtype), 0); |
1415 | hv_store (prop_type, cprop->name, strlen (cprop->name), newSViv (cprop->dtype), 0); |
1179 | hv_store (prop_idx, cprop->name, strlen (cprop->name), newSViv (cprop->idx ), 0); |
1416 | hv_store (prop_idx, cprop->name, strlen (cprop->name), newSViv (cprop->idx ), 0); |
1180 | } |
1417 | } |
|
|
1418 | |
|
|
1419 | //I_EVENT_API (PACKAGE); |
1181 | } |
1420 | } |
1182 | |
1421 | |
1183 | void |
1422 | void |
1184 | LOG (int level, char *msg) |
1423 | LOG (int level, char *msg) |
1185 | PROTOTYPE: $$ |
1424 | PROTOTYPE: $$ |
1186 | C_ARGS: level, "%s", msg |
1425 | C_ARGS: level, "%s", msg |
|
|
1426 | |
|
|
1427 | char *path_combine (char *base, char *path) |
|
|
1428 | PROTOTYPE: $$ |
|
|
1429 | |
|
|
1430 | char *path_combine_and_normalize (char *base, char *path) |
|
|
1431 | PROTOTYPE: $$ |
1187 | |
1432 | |
1188 | char * |
1433 | char * |
1189 | cf_get_maps_directory (char *path) |
1434 | cf_get_maps_directory (char *path) |
1190 | PROTOTYPE: $ |
1435 | PROTOTYPE: $ |
1191 | ALIAS: maps_directory = 0 |
1436 | ALIAS: maps_directory = 0 |
… | |
… | |
1210 | |
1455 | |
1211 | int |
1456 | int |
1212 | cf_find_animation (char *text) |
1457 | cf_find_animation (char *text) |
1213 | PROTOTYPE: $ |
1458 | PROTOTYPE: $ |
1214 | |
1459 | |
|
|
1460 | int random_roll(int min, int max, object *op, int goodbad); |
|
|
1461 | |
|
|
1462 | int |
|
|
1463 | exp_to_level (val64 exp) |
|
|
1464 | CODE: |
|
|
1465 | { |
|
|
1466 | int i = 0; |
|
|
1467 | |
|
|
1468 | RETVAL = settings.max_level; |
|
|
1469 | |
|
|
1470 | for (i = 1; i <= settings.max_level; i++) |
|
|
1471 | { |
|
|
1472 | if (levels[i] > exp) |
|
|
1473 | { |
|
|
1474 | RETVAL = i - 1; |
|
|
1475 | break; |
|
|
1476 | } |
|
|
1477 | } |
|
|
1478 | } |
|
|
1479 | OUTPUT: RETVAL |
|
|
1480 | |
|
|
1481 | val64 |
|
|
1482 | level_to_min_exp (int level) |
|
|
1483 | CODE: |
|
|
1484 | if (level > settings.max_level) |
|
|
1485 | RETVAL = levels[settings.max_level]; |
|
|
1486 | else if (level < 1) |
|
|
1487 | RETVAL = 0; |
|
|
1488 | else |
|
|
1489 | RETVAL = levels[level]; |
|
|
1490 | OUTPUT: RETVAL |
|
|
1491 | |
|
|
1492 | SV * |
|
|
1493 | resistance_to_string (int atnr) |
|
|
1494 | CODE: |
|
|
1495 | if (atnr >= 0 && atnr < NROFATTACKS) |
|
|
1496 | RETVAL = newSVpv (resist_plus[atnr], 0); |
|
|
1497 | else |
|
|
1498 | XSRETURN_UNDEF; |
|
|
1499 | OUTPUT: RETVAL |
|
|
1500 | |
1215 | MODULE = cf PACKAGE = cf::object PREFIX = cf_object_ |
1501 | MODULE = cf PACKAGE = cf::object PREFIX = cf_object_ |
1216 | |
1502 | |
1217 | SV * |
1503 | SV * |
1218 | get_property (object *obj, int type, int idx) |
1504 | get_property (object *obj, int type, int idx) |
1219 | CODE: |
1505 | CODE: |
… | |
… | |
1227 | { |
1513 | { |
1228 | case CFAPI_INT: |
1514 | case CFAPI_INT: |
1229 | cf_object_set_int_property (obj, idx, SvIV (newval)); |
1515 | cf_object_set_int_property (obj, idx, SvIV (newval)); |
1230 | break; |
1516 | break; |
1231 | case CFAPI_LONG: |
1517 | case CFAPI_LONG: |
1232 | cf_object_set_long_property (obj, idx, SvNV (newval)); |
1518 | cf_object_set_long_property (obj, idx, SvVAL64 (newval)); |
1233 | break; |
1519 | break; |
1234 | case CFAPI_DOUBLE: |
1520 | case CFAPI_DOUBLE: |
1235 | { |
1521 | { |
1236 | int unused_type; |
1522 | int unused_type; |
1237 | object_set_property (&unused_type, obj, idx, (double)SvNV (newval)); |
1523 | object_set_property (&unused_type, obj, idx, (double)SvNV (newval)); |
… | |
… | |
1263 | attacktype = 0 |
1549 | attacktype = 0 |
1264 | CODE: |
1550 | CODE: |
1265 | RETVAL = obj->attacktype; |
1551 | RETVAL = obj->attacktype; |
1266 | OUTPUT: RETVAL |
1552 | OUTPUT: RETVAL |
1267 | |
1553 | |
|
|
1554 | # missing in plug-in api, of course |
1268 | void |
1555 | void |
1269 | set_food (object *obj, int food) |
1556 | set_food (object *obj, int food) |
1270 | CODE: |
1557 | CODE: |
1271 | obj->stats.food = food; |
1558 | obj->stats.food = food; |
1272 | |
1559 | |
… | |
… | |
1306 | |
1593 | |
1307 | void cf_object_free (object *op) |
1594 | void cf_object_free (object *op) |
1308 | |
1595 | |
1309 | object *cf_object_present_archname_inside (object *op, char *whatstr) |
1596 | object *cf_object_present_archname_inside (object *op, char *whatstr) |
1310 | |
1597 | |
1311 | int cf_object_transfer (object *op, int x, int y, int r, object *orig) |
1598 | int cf_object_transfer (object *op, int x, int y, int r = 0, object_ornull *orig = 0) |
1312 | |
1599 | |
1313 | int cf_object_change_map (object *op, int x, int y, mapstruct *map) |
1600 | int cf_object_change_map (object *op, int x, int y, mapstruct *map) |
1314 | |
1601 | |
1315 | object *cf_object_clone (object *op, int clonetype = 0) |
1602 | object *cf_object_clone (object *op, int clonetype = 0) |
1316 | |
1603 | |
1317 | int cf_object_pay_item (object *op, object *buyer) |
1604 | int cf_object_pay_item (object *op, object *buyer) |
1318 | |
1605 | |
1319 | int cf_object_pay_amount (object *op, double amount) |
1606 | int cf_object_pay_amount (object *op, val64 amount) |
1320 | |
1607 | |
1321 | int cf_object_cast_spell (object *caster, object *ctoo, int dir, object *spell_ob, char *stringarg = 0) |
1608 | int cf_object_cast_spell (object *caster, object *ctoo, int dir, object *spell_ob, char *stringarg = 0) |
1322 | |
1609 | |
1323 | int cf_object_cast_ability (object *caster, object *ctoo, int dir, object *sp_, char *stringarg = 0) |
1610 | int cf_object_cast_ability (object *caster, object *ctoo, int dir, object *sp_, char *stringarg = 0) |
1324 | |
1611 | |
… | |
… | |
1359 | |
1646 | |
1360 | void cf_object_update (object *op, int flags) |
1647 | void cf_object_update (object *op, int flags) |
1361 | |
1648 | |
1362 | void cf_object_pickup (object *op, object *what) |
1649 | void cf_object_pickup (object *op, object *what) |
1363 | |
1650 | |
1364 | char *cf_object_get_key (object *op, char *keyname) |
|
|
1365 | ALIAS: key = 0 |
|
|
1366 | |
|
|
1367 | void cf_object_set_key (object *op, char *keyname, char *value) |
|
|
1368 | |
|
|
1369 | object *cf_create_object_by_name (const char *name) |
1651 | object *cf_create_object_by_name (const char *name) |
|
|
1652 | |
|
|
1653 | void change_exp (object *op, val64 exp, const char *skill_name = 0, int flag = 0) |
|
|
1654 | |
|
|
1655 | void pay_player (object *op, val64 amount) |
|
|
1656 | |
|
|
1657 | val64 pay_player_arch (object *op, const char *arch, val64 amount) |
|
|
1658 | |
|
|
1659 | void player_lvl_adj (object *who, object *skill = 0) |
|
|
1660 | |
|
|
1661 | int kill_object (object *op, int dam = 0, object *hitter = 0, int type = AT_PHYSICAL) |
|
|
1662 | |
|
|
1663 | int calc_skill_exp (object *who, object *op, object *skill); |
|
|
1664 | |
|
|
1665 | void |
|
|
1666 | cf_object_set_resistance (object *op, int rtype, int val) |
|
|
1667 | CODE: |
|
|
1668 | if (rtype >= 0 && rtype < NROFATTACKS) |
|
|
1669 | op->resist[rtype] = val; |
|
|
1670 | |
1370 | |
1671 | |
1371 | MODULE = cf PACKAGE = cf::object PREFIX = cf_ |
1672 | MODULE = cf PACKAGE = cf::object PREFIX = cf_ |
1372 | |
1673 | |
1373 | void cf_fix_object (object *pl) |
1674 | void cf_fix_object (object *pl) |
1374 | ALIAS: fix = 0 |
1675 | ALIAS: fix = 0 |
… | |
… | |
1390 | CODE: |
1691 | CODE: |
1391 | { |
1692 | { |
1392 | int unused_type; |
1693 | int unused_type; |
1393 | RETVAL = (object *)object_insert (&unused_type, ob, 0, where, orig, flag, x, y); |
1694 | RETVAL = (object *)object_insert (&unused_type, ob, 0, where, orig, flag, x, y); |
1394 | } |
1695 | } |
|
|
1696 | |
|
|
1697 | # syntatic sugar for easier use in event callbacks. |
|
|
1698 | const char *options (object *op) |
|
|
1699 | CODE: |
|
|
1700 | RETVAL = op->name; |
|
|
1701 | OUTPUT: |
|
|
1702 | RETVAL |
|
|
1703 | |
|
|
1704 | const char *get_ob_key_value (object *op, const char *key) |
|
|
1705 | |
|
|
1706 | bool set_ob_key_value (object *op, const char *key, const char *value = 0, int add_key = 1) |
1395 | |
1707 | |
1396 | object *get_nearest_player (object *ob) |
1708 | object *get_nearest_player (object *ob) |
1397 | ALIAS: nearest_player = 0 |
1709 | ALIAS: nearest_player = 0 |
1398 | PREINIT: |
1710 | PREINIT: |
1399 | extern object *get_nearest_player (object *); |
1711 | extern object *get_nearest_player (object *); |
… | |
… | |
1421 | base_name (object *ob, int plural) |
1733 | base_name (object *ob, int plural) |
1422 | CODE: |
1734 | CODE: |
1423 | RETVAL = cf_query_base_name (ob, plural); |
1735 | RETVAL = cf_query_base_name (ob, plural); |
1424 | OUTPUT: RETVAL |
1736 | OUTPUT: RETVAL |
1425 | |
1737 | |
|
|
1738 | living * |
|
|
1739 | stats (object *ob) |
|
|
1740 | CODE: |
|
|
1741 | RETVAL = &ob->stats; |
|
|
1742 | OUTPUT: RETVAL |
|
|
1743 | |
1426 | |
1744 | |
1427 | MODULE = cf PACKAGE = cf::object::player PREFIX = cf_player_ |
1745 | MODULE = cf PACKAGE = cf::object::player PREFIX = cf_player_ |
1428 | |
1746 | |
1429 | player *player (object *op) |
1747 | player *player (object *op) |
1430 | CODE: |
1748 | CODE: |
1431 | RETVAL = cf_player_find (cf_query_name (op)); |
1749 | RETVAL = op->contr; |
1432 | OUTPUT: RETVAL |
1750 | OUTPUT: RETVAL |
1433 | |
1751 | |
1434 | void cf_player_message (object *obj, char *txt, int flags = NDI_ORANGE | NDI_UNIQUE) |
1752 | void cf_player_message (object *obj, char *txt, int flags = NDI_ORANGE | NDI_UNIQUE) |
1435 | |
1753 | |
1436 | object *cf_player_send_inventory (object *op) |
1754 | object *cf_player_send_inventory (object *op) |
… | |
… | |
1451 | partylist *cf_player_get_party (object *op) |
1769 | partylist *cf_player_get_party (object *op) |
1452 | ALIAS: party = 0 |
1770 | ALIAS: party = 0 |
1453 | |
1771 | |
1454 | void cf_player_set_party (object *op, partylist *party) |
1772 | void cf_player_set_party (object *op, partylist *party) |
1455 | |
1773 | |
|
|
1774 | void change_skill (object *op, val64 exp, char *skill_name = 0, int flag = 0) |
|
|
1775 | |
|
|
1776 | void kill_player (object *op) |
1456 | |
1777 | |
1457 | MODULE = cf PACKAGE = cf::object::map PREFIX = cf_ |
1778 | MODULE = cf PACKAGE = cf::object::map PREFIX = cf_ |
1458 | |
1779 | |
1459 | MODULE = cf PACKAGE = cf::player PREFIX = cf_player_ |
1780 | MODULE = cf PACKAGE = cf::player PREFIX = cf_player_ |
1460 | |
1781 | |
… | |
… | |
1463 | |
1784 | |
1464 | void cf_player_move (player *pl, int dir) |
1785 | void cf_player_move (player *pl, int dir) |
1465 | |
1786 | |
1466 | void MapNewmapCmd (player *pl) |
1787 | void MapNewmapCmd (player *pl) |
1467 | |
1788 | |
|
|
1789 | void play_sound_player_only (player *pl, int soundnum, int x = 0, int y = 0); |
|
|
1790 | |
1468 | # nonstandard |
1791 | # nonstandard |
1469 | object *ob (player *pl) |
1792 | object *ob (player *pl) |
1470 | CODE: |
1793 | CODE: |
1471 | RETVAL = pl->ob; |
1794 | RETVAL = pl->ob; |
1472 | OUTPUT: RETVAL |
1795 | OUTPUT: RETVAL |
… | |
… | |
1478 | |
1801 | |
1479 | player *next (player *pl) |
1802 | player *next (player *pl) |
1480 | CODE: |
1803 | CODE: |
1481 | RETVAL = pl->next; |
1804 | RETVAL = pl->next; |
1482 | OUTPUT: RETVAL |
1805 | OUTPUT: RETVAL |
|
|
1806 | |
|
|
1807 | bool |
|
|
1808 | cell_visible (player *pl, int dx, int dy) |
|
|
1809 | CODE: |
|
|
1810 | RETVAL = FABS (dx) <= pl->socket.mapx / 2 && FABS (dy) <= pl->socket.mapy / 2 |
|
|
1811 | && !pl->blocked_los [dx + pl->socket.mapx / 2][dy + pl->socket.mapy / 2]; |
|
|
1812 | OUTPUT: |
|
|
1813 | RETVAL |
|
|
1814 | |
|
|
1815 | void |
|
|
1816 | send (player *pl, SV *packet) |
|
|
1817 | CODE: |
|
|
1818 | { |
|
|
1819 | STRLEN len; |
|
|
1820 | char *buf = SvPVbyte (packet, len); |
|
|
1821 | |
|
|
1822 | Write_String_To_Socket (&pl->socket, buf, len); |
|
|
1823 | } |
|
|
1824 | |
|
|
1825 | int |
|
|
1826 | listening (player *pl, int new_value = -1) |
|
|
1827 | CODE: |
|
|
1828 | RETVAL = pl->listening; |
|
|
1829 | if (new_value >= 0) |
|
|
1830 | pl->listening = new_value; |
|
|
1831 | OUTPUT: |
|
|
1832 | RETVAL |
|
|
1833 | |
|
|
1834 | void get_savebed (player *pl) |
|
|
1835 | ALIAS: |
|
|
1836 | savebed = 0 |
|
|
1837 | PPCODE: |
|
|
1838 | EXTEND (SP, 3); |
|
|
1839 | PUSHs (sv_2mortal (newSVpv (pl->savebed_map, 0))); |
|
|
1840 | PUSHs (sv_2mortal (newSViv (pl->bed_x))); |
|
|
1841 | PUSHs (sv_2mortal (newSViv (pl->bed_y))); |
|
|
1842 | |
|
|
1843 | void set_savebed (player *pl, char *map_path, int x, int y) |
|
|
1844 | CODE: |
|
|
1845 | strcpy (pl->savebed_map, map_path); |
|
|
1846 | pl->bed_x = x; |
|
|
1847 | pl->bed_y = y; |
1483 | |
1848 | |
1484 | void |
1849 | void |
1485 | list () |
1850 | list () |
1486 | PPCODE: |
1851 | PPCODE: |
1487 | { |
1852 | { |
1488 | player *pl; |
1853 | player *pl; |
1489 | for (pl = first_player; pl; pl = pl->next) |
1854 | for (pl = first_player; pl; pl = pl->next) |
1490 | XPUSHs (newSVcfapi (CFAPI_PPLAYER, pl)); |
1855 | XPUSHs (newSVcfapi (CFAPI_PPLAYER, pl)); |
1491 | } |
1856 | } |
|
|
1857 | |
|
|
1858 | bool |
|
|
1859 | peaceful (player *pl, bool new_setting = 0) |
|
|
1860 | PROTOTYPE: $;$ |
|
|
1861 | CODE: |
|
|
1862 | RETVAL = pl->peaceful; |
|
|
1863 | if (items > 1) |
|
|
1864 | pl->peaceful = new_setting; |
|
|
1865 | OUTPUT: |
|
|
1866 | RETVAL |
|
|
1867 | |
|
|
1868 | living * |
|
|
1869 | orig_stats (player *pl) |
|
|
1870 | CODE: |
|
|
1871 | RETVAL = &pl->orig_stats; |
|
|
1872 | OUTPUT: RETVAL |
|
|
1873 | |
|
|
1874 | living * |
|
|
1875 | last_stats (player *pl) |
|
|
1876 | CODE: |
|
|
1877 | RETVAL = &pl->last_stats; |
|
|
1878 | OUTPUT: RETVAL |
1492 | |
1879 | |
1493 | |
1880 | |
1494 | MODULE = cf PACKAGE = cf::map PREFIX = cf_map_ |
1881 | MODULE = cf PACKAGE = cf::map PREFIX = cf_map_ |
1495 | |
1882 | |
1496 | SV * |
1883 | SV * |
… | |
… | |
1519 | RETVAL = map_get_map (&unused_type, 0, width, height); |
1906 | RETVAL = map_get_map (&unused_type, 0, width, height); |
1520 | } |
1907 | } |
1521 | OUTPUT: |
1908 | OUTPUT: |
1522 | RETVAL |
1909 | RETVAL |
1523 | |
1910 | |
|
|
1911 | void delete_map (mapstruct *map) |
|
|
1912 | |
|
|
1913 | void clean_tmp_map (mapstruct *map) |
|
|
1914 | |
|
|
1915 | void play_sound_map (mapstruct *map, int x, int y, int sound_num) |
|
|
1916 | |
|
|
1917 | mapstruct *tile_map (mapstruct *map, unsigned int dir) |
|
|
1918 | CODE: |
|
|
1919 | RETVAL = dir < 4 ? map->tile_map [dir] : 0; |
|
|
1920 | OUTPUT: |
|
|
1921 | RETVAL |
|
|
1922 | |
|
|
1923 | char *tile_path (mapstruct *map, unsigned int dir) |
|
|
1924 | CODE: |
|
|
1925 | if (dir >= 4) |
|
|
1926 | XSRETURN_UNDEF; |
|
|
1927 | RETVAL = map->tile_path [dir]; |
|
|
1928 | OUTPUT: |
|
|
1929 | RETVAL |
|
|
1930 | |
1524 | mapstruct *cf_map_get_map (char *name) |
1931 | mapstruct *cf_map_get_map (char *name) |
1525 | PROTOTYPE: $ |
1932 | PROTOTYPE: $ |
1526 | ALIAS: map = 0 |
1933 | ALIAS: map = 0 |
1527 | |
1934 | |
|
|
1935 | mapstruct *has_been_loaded (char *name) |
|
|
1936 | PROTOTYPE: $ |
|
|
1937 | |
1528 | mapstruct *cf_map_get_first () |
1938 | mapstruct *cf_map_get_first () |
1529 | PROTOTYPE: |
1939 | PROTOTYPE: |
1530 | ALIAS: first = 0 |
1940 | ALIAS: first = 0 |
1531 | |
1941 | |
|
|
1942 | # whoever "designed" the plug-in api should have wasted |
|
|
1943 | # his/her time with staying away from the project - would have |
|
|
1944 | # saved others a lot of time, without doubt. |
|
|
1945 | void set_path (mapstruct *where, char *path) |
|
|
1946 | CODE: |
|
|
1947 | strcpy (where->path, path); |
|
|
1948 | |
|
|
1949 | int in_memory (mapstruct *map) |
|
|
1950 | CODE: |
|
|
1951 | RETVAL = map->in_memory; |
|
|
1952 | OUTPUT: |
|
|
1953 | RETVAL |
|
|
1954 | |
|
|
1955 | bool unique (mapstruct *map) |
|
|
1956 | CODE: |
|
|
1957 | RETVAL = map->unique; |
|
|
1958 | OUTPUT: |
|
|
1959 | RETVAL |
|
|
1960 | |
|
|
1961 | void set_unique (mapstruct *map, bool unique) |
|
|
1962 | CODE: |
|
|
1963 | map->unique = unique; |
|
|
1964 | |
1532 | object *cf_map_insert_object_there (mapstruct *where, object *op, object *originator, int flags) |
1965 | object *cf_map_insert_object_there (mapstruct *where, object *op, object *originator, int flags) |
1533 | |
1966 | |
1534 | object *cf_map_insert_object (mapstruct *where, object* op, int x, int y) |
1967 | object *cf_map_insert_object (mapstruct *where, object* op, int x, int y) |
1535 | |
1968 | |
1536 | object* cf_map_present_arch_by_name (mapstruct *map, const char* str, int nx, int ny) |
1969 | object* cf_map_present_arch_by_name (mapstruct *map, const char* str, int nx, int ny) |
1537 | C_ARGS: str, map, nx, ny |
1970 | C_ARGS: str, map, nx, ny |
1538 | |
1971 | |
1539 | #int cf_map_get_flags (mapstruct* map, mapstruct** nmap, I16 x, I16 y, I16 *nx, I16 *ny) |
|
|
1540 | |
|
|
1541 | void |
1972 | void |
|
|
1973 | cf_map_normalise (mapstruct *map, int x, int y) |
|
|
1974 | PPCODE: |
|
|
1975 | { |
|
|
1976 | mapstruct *nmap = 0; |
|
|
1977 | I16 nx = 0, ny = 0; |
|
|
1978 | int flags = cf_map_get_flags (map, &nmap, x, y, &nx, &ny); |
|
|
1979 | |
|
|
1980 | EXTEND (SP, 4); |
|
|
1981 | PUSHs (sv_2mortal (newSViv (flags))); |
|
|
1982 | |
|
|
1983 | if (GIMME_V == G_ARRAY) |
|
|
1984 | { |
|
|
1985 | PUSHs (sv_2mortal (newSVcfapi (CFAPI_PMAP, nmap))); |
|
|
1986 | PUSHs (sv_2mortal (newSViv (nx))); |
|
|
1987 | PUSHs (sv_2mortal (newSViv (ny))); |
|
|
1988 | } |
|
|
1989 | } |
|
|
1990 | |
|
|
1991 | void |
1542 | at (mapstruct *obj, unsigned int x, unsigned int y) |
1992 | at (mapstruct *map, unsigned int x, unsigned int y) |
1543 | PROTOTYPE: $$$ |
1993 | PROTOTYPE: $$$ |
1544 | INIT: |
|
|
1545 | if (x >= MAP_WIDTH (obj) || y >= MAP_HEIGHT (obj)) XSRETURN_EMPTY; |
|
|
1546 | PPCODE: |
1994 | PPCODE: |
1547 | { |
1995 | { |
1548 | object *o; |
1996 | object *o; |
1549 | |
1997 | mapstruct *nmap = 0; |
|
|
1998 | I16 nx, ny; |
|
|
1999 | |
|
|
2000 | cf_map_get_flags (map, &nmap, x, y, &nx, &ny); |
|
|
2001 | |
|
|
2002 | if (nmap) |
1550 | for (o = GET_MAP_OB (obj, x, y); o; o = o->above) |
2003 | for (o = GET_MAP_OB (nmap, nx, ny); o; o = o->above) |
1551 | XPUSHs (sv_2mortal (newSVcfapi (CFAPI_POBJECT, o))); |
2004 | XPUSHs (sv_2mortal (newSVcfapi (CFAPI_POBJECT, o))); |
1552 | } |
2005 | } |
1553 | |
2006 | |
1554 | SV * |
2007 | SV * |
1555 | bot_at (mapstruct *obj, unsigned int x, unsigned int y) |
2008 | bot_at (mapstruct *obj, unsigned int x, unsigned int y) |
1556 | PROTOTYPE: $$$ |
2009 | PROTOTYPE: $$$ |
… | |
… | |
1649 | } |
2102 | } |
1650 | } |
2103 | } |
1651 | } |
2104 | } |
1652 | } |
2105 | } |
1653 | |
2106 | |
|
|
2107 | |
1654 | MODULE = cf PACKAGE = cf::arch PREFIX = cf_archetype_ |
2108 | MODULE = cf PACKAGE = cf::arch PREFIX = cf_archetype_ |
1655 | |
2109 | |
1656 | archetype *cf_archetype_get_first() |
2110 | archetype *cf_archetype_get_first() |
1657 | PROTOTYPE: |
2111 | PROTOTYPE: |
1658 | ALIAS: first = 0 |
2112 | ALIAS: first = 0 |
… | |
… | |
1710 | |
2164 | |
1711 | const char *cf_region_get_message (region *reg) |
2165 | const char *cf_region_get_message (region *reg) |
1712 | ALIAS: message = 0 |
2166 | ALIAS: message = 0 |
1713 | |
2167 | |
1714 | |
2168 | |
|
|
2169 | MODULE = cf PACKAGE = cf::living PREFIX = cf_living_ |
|
|
2170 | |
|
|
2171 | val64 |
|
|
2172 | exp (living *liv, val64 new_val = 0) |
|
|
2173 | PROTOTYPE: $;$ |
|
|
2174 | ALIAS: |
|
|
2175 | Str = 1 |
|
|
2176 | Dex = 2 |
|
|
2177 | Con = 3 |
|
|
2178 | Wis = 4 |
|
|
2179 | Cha = 5 |
|
|
2180 | Int = 6 |
|
|
2181 | Pow = 7 |
|
|
2182 | wc = 8 |
|
|
2183 | ac = 9 |
|
|
2184 | hp = 10 |
|
|
2185 | maxhp = 11 |
|
|
2186 | sp = 12 |
|
|
2187 | maxsp = 13 |
|
|
2188 | grace = 14 |
|
|
2189 | maxgrace = 15 |
|
|
2190 | food = 16 |
|
|
2191 | dam = 17 |
|
|
2192 | luck = 18 |
|
|
2193 | CODE: |
|
|
2194 | # define LIVING_ACC(acc,idx) case idx: RETVAL = liv->acc; if (items > 1) liv->acc = new_val; break |
|
|
2195 | switch (ix) |
|
|
2196 | { |
|
|
2197 | LIVING_ACC (exp , 0); |
|
|
2198 | LIVING_ACC (Str , 1); |
|
|
2199 | LIVING_ACC (Dex , 2); |
|
|
2200 | LIVING_ACC (Con , 3); |
|
|
2201 | LIVING_ACC (Wis , 4); |
|
|
2202 | LIVING_ACC (Cha , 5); |
|
|
2203 | LIVING_ACC (Int , 6); |
|
|
2204 | LIVING_ACC (Pow , 7); |
|
|
2205 | LIVING_ACC (wc , 8); |
|
|
2206 | LIVING_ACC (ac , 9); |
|
|
2207 | LIVING_ACC (hp , 10); |
|
|
2208 | LIVING_ACC (maxhp , 11); |
|
|
2209 | LIVING_ACC (sp , 12); |
|
|
2210 | LIVING_ACC (maxsp , 13); |
|
|
2211 | LIVING_ACC (grace , 14); |
|
|
2212 | LIVING_ACC (maxgrace, 15); |
|
|
2213 | LIVING_ACC (food , 16); |
|
|
2214 | LIVING_ACC (dam , 17); |
|
|
2215 | LIVING_ACC (luck , 18); |
|
|
2216 | } |
|
|
2217 | # undef LIVING_ACC |
|
|
2218 | OUTPUT: |
|
|
2219 | RETVAL |
|
|
2220 | |