1 | /* |
1 | /* |
2 | * This file is part of Deliantra, the Roguelike Realtime MMORPG. |
2 | * This file is part of Deliantra, the Roguelike Realtime MMORPG. |
3 | * |
3 | * |
4 | * Copyright (©) 2005,2006,2007,2008,2009 Marc Alexander Lehmann / Robin Redeker / the Deliantra team |
4 | * Copyright (©) 2005,2006,2007,2008,2009,2010 Marc Alexander Lehmann / Robin Redeker / the Deliantra team |
5 | * |
5 | * |
6 | * Deliantra is free software: you can redistribute it and/or modify it under |
6 | * Deliantra is free software: you can redistribute it and/or modify it under |
7 | * the terms of the Affero GNU General Public License as published by the |
7 | * the terms of the Affero GNU General Public License as published by the |
8 | * Free Software Foundation, either version 3 of the License, or (at your |
8 | * Free Software Foundation, either version 3 of the License, or (at your |
9 | * option) any later version. |
9 | * option) any later version. |
… | |
… | |
24 | // cfperl.h perl interface |
24 | // cfperl.h perl interface |
25 | // |
25 | // |
26 | #ifndef CFPERL_H__ |
26 | #ifndef CFPERL_H__ |
27 | #define CFPERL_H__ |
27 | #define CFPERL_H__ |
28 | |
28 | |
29 | #include <cstdarg> |
|
|
30 | #include <cstdio> |
|
|
31 | #include <bitset> |
|
|
32 | |
|
|
33 | using namespace std; |
29 | using namespace std; |
34 | |
30 | |
35 | #include <EXTERN.h> |
31 | #include <EXTERN.h> |
36 | #include <perl.h> |
32 | #include <perl.h> |
37 | #include <XSUB.h> |
33 | #include <XSUB.h> |
… | |
… | |
85 | LOG (llevError, "runtime error in %s: %s", __func__, SvPVutf8_nolen (ERRSV)); |
81 | LOG (llevError, "runtime error in %s: %s", __func__, SvPVutf8_nolen (ERRSV)); |
86 | |
82 | |
87 | inline int call_pvsv (const char *ob, I32 flags) { return call_pv (ob, flags); } |
83 | inline int call_pvsv (const char *ob, I32 flags) { return call_pv (ob, flags); } |
88 | inline int call_pvsv (SV *ob, I32 flags) { return call_sv (ob, flags); } |
84 | inline int call_pvsv (SV *ob, I32 flags) { return call_sv (ob, flags); } |
89 | |
85 | |
|
|
86 | // TODO: temporarily enabled pushstack/popstack for all calls from the core, to maybe fix object memleak? |
|
|
87 | #define CALL_PUSH PUSHSTACKi (PERLSI_UNKNOWN) |
|
|
88 | #define CALL_POP PUTBACK; POPSTACK; SPAGAIN |
90 | #define CALL_BEGIN(args) dSP; ENTER; SAVETMPS; PUSHMARK (SP); EXTEND (SP, args) |
89 | #define CALL_BEGIN(args) dSP; CALL_PUSH; ENTER; SAVETMPS; PUSHMARK (SP); EXTEND (SP, args) |
91 | #define CALL_ARG_SV(sv) PUSHs (sv_2mortal (sv)) // separate because no refcount inc |
90 | #define CALL_ARG_SV(sv) PUSHs (sv_2mortal (sv)) // separate because no refcount inc |
92 | #define CALL_ARG(expr) PUSHs (sv_2mortal (to_sv (expr))) |
91 | #define CALL_ARG(expr) PUSHs (sv_2mortal (to_sv (expr))) |
93 | #define CALL_CALL(name, flags) PUTBACK; int count = call_pvsv (name, (flags) | G_EVAL); SPAGAIN; |
92 | #define CALL_CALL(name, flags) PUTBACK; int count = call_pvsv (name, (flags) | G_EVAL); SPAGAIN; |
94 | #define CALL_END PUTBACK; CHECK_ERROR; FREETMPS; LEAVE |
93 | #define CALL_END PUTBACK; CHECK_ERROR; FREETMPS; LEAVE; CALL_POP |
95 | |
94 | |
96 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
95 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
97 | |
96 | |
98 | void cfperl_init (); |
97 | void cfperl_init (); |
99 | void cfperl_main (); |
98 | void cfperl_main (); |
100 | void cfperl_tick (); |
99 | void cfperl_tick (); |
101 | void cfperl_emergency_save (); |
100 | void cfperl_emergency_save (); |
102 | void cfperl_cleanup (int make_core); |
101 | void cfperl_cleanup (int make_core); |
103 | void cfperl_make_book (object *book, int level); |
102 | void cfperl_make_book (object *book, int level); |
104 | void cfperl_send_msg (client *ns, int color, const char *type, const char *msg); |
103 | void cfperl_send_msg (client *ns, int color, const_utf8_string type, const_utf8_string msg); |
105 | int cfperl_can_merge (object *ob1, object *ob2); |
104 | int cfperl_can_merge (object *ob1, object *ob2); |
106 | void cfperl_mapscript_activate (object *ob, int state, object *activator, object *originator = 0); |
105 | void cfperl_mapscript_activate (object *ob, int state, object *activator, object *originator = 0); |
107 | |
106 | |
108 | bool is_match_expr (const char *expr); |
107 | bool is_match_expr (const_utf8_string expr); |
109 | // applies the match expression and returns true if it matches |
108 | // applies the match expression and returns true if it matches |
110 | bool match (const char *expr, object *ob, object *self = 0, object *source = 0, object *originator = 0); |
109 | bool match (const_utf8_string expr, object *ob, object *self = 0, object *source = 0, object *originator = 0); |
111 | // same as above, but returns the first object found, or 0 |
110 | // same as above, but returns the first object found, or 0 |
112 | object *match_one (const char *expr, object *ob, object *self = 0, object *source = 0, object *originator = 0); |
111 | object *match_one (const_utf8_string expr, object *ob, object *self = 0, object *source = 0, object *originator = 0); |
113 | |
112 | |
114 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
113 | ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// |
115 | |
114 | |
116 | #if IVSIZE >= 8 |
115 | #if IVSIZE >= 8 |
117 | typedef IV val64; |
116 | typedef IV val64; |
… | |
… | |
240 | : attachable_flags (0), self (0), cb (0), attach (src.attach) |
239 | : attachable_flags (0), self (0), cb (0), attach (src.attach) |
241 | { |
240 | { |
242 | } |
241 | } |
243 | |
242 | |
244 | // set a custom key to the given value, or delete it if value = 0 |
243 | // set a custom key to the given value, or delete it if value = 0 |
245 | void set_key (const char *key, const char *value = 0, bool is_utf8 = 0); |
244 | void set_key (const_utf8_string key, const_utf8_string value = 0, bool is_utf8 = 0); |
246 | |
245 | |
247 | void set_key_text (const char *key, const char *value = 0) |
246 | void set_key_text (const_utf8_string key, const_utf8_string value = 0) |
248 | { |
247 | { |
249 | set_key (key, value, 1); |
248 | set_key (key, value, 1); |
250 | } |
249 | } |
251 | |
250 | |
252 | void set_key_data (const char *key, const char *value = 0) |
251 | void set_key_data (const_utf8_string key, const_utf8_string value = 0) |
253 | { |
252 | { |
254 | set_key (key, value, 0); |
253 | set_key (key, value, 0); |
255 | } |
254 | } |
256 | |
255 | |
257 | attachable &operator =(const attachable &src); |
256 | attachable &operator =(const attachable &src); |
258 | |
257 | |
259 | // used to _quickly_ device wether to shortcut the evaluation |
258 | // used to _quickly_ decide wether to shortcut the evaluation |
260 | bool should_invoke (event_type event) |
259 | bool should_invoke (event_type event) |
261 | { |
260 | { |
262 | return ev_want_event [event] || cb; |
261 | return ev_want_event [event] || cb; |
263 | } |
262 | } |
264 | |
263 | |
… | |
… | |
342 | memcpy (p, k.s, k.l); p += k.l; *p++ = ' '; |
341 | memcpy (p, k.s, k.l); p += k.l; *p++ = ' '; |
343 | memcpy (p, v.s, v.l); p += v.l; *p++ = '\n'; |
342 | memcpy (p, v.s, v.l); p += v.l; *p++ = '\n'; |
344 | alloc (p); |
343 | alloc (p); |
345 | } |
344 | } |
346 | |
345 | |
347 | void put (const keyword_string k, const char *v) |
346 | void put (const keyword_string k, const_utf8_string v) |
348 | { |
347 | { |
349 | if (expect_true (v)) |
348 | if (expect_true (v)) |
350 | put (k, keyword_string (v, strlen (v))); |
349 | put (k, keyword_string (v, strlen (v))); |
351 | else |
350 | else |
352 | put (k); |
351 | put (k); |
… | |
… | |
419 | { |
418 | { |
420 | put (k, (T *)v); |
419 | put (k, (T *)v); |
421 | } |
420 | } |
422 | |
421 | |
423 | MTH bool save (const_octet_string path); |
422 | MTH bool save (const_octet_string path); |
424 | char *as_string (); // like strdup |
423 | utf8_string as_string (); // like strdup |
425 | |
424 | |
426 | operator bool () { return !!av; } |
425 | operator bool () { return !!av; } |
427 | }; |
426 | }; |
428 | |
427 | |
429 | INTERFACE_CLASS(object_thawer) |
428 | INTERFACE_CLASS(object_thawer) |
… | |
… | |
439 | const char *value_nn; // the value, or the empty string if no value |
438 | const char *value_nn; // the value, or the empty string if no value |
440 | const char *name; |
439 | const char *name; |
441 | |
440 | |
442 | operator bool () const { return !!text; } |
441 | operator bool () const { return !!text; } |
443 | |
442 | |
444 | object_thawer (const char *path = 0); |
443 | object_thawer (const_utf8_string path = 0); |
445 | object_thawer (const char *data, AV *perlav); |
444 | object_thawer (const_utf8_string data, AV *perlav); |
446 | ~object_thawer (); |
445 | ~object_thawer (); |
447 | |
446 | |
448 | void get (attachable *obj, int oid); |
447 | void get (attachable *obj, int oid); |
449 | |
448 | |
450 | // parse next line |
449 | // parse next line |
… | |
… | |
452 | // skip the current key-value (usually fetch next line, for |
451 | // skip the current key-value (usually fetch next line, for |
453 | // multiline-fields, skips till the corresponding end-kw |
452 | // multiline-fields, skips till the corresponding end-kw |
454 | MTH void skip (); |
453 | MTH void skip (); |
455 | MTH void skip_block (); // skips till and over KW_end |
454 | MTH void skip_block (); // skips till and over KW_end |
456 | |
455 | |
457 | char *get_str () { return value; } // may be 0 |
456 | bool has_value () { return value; } |
|
|
457 | const_utf8_string get_str () { return value_nn; } // empty string when missing |
458 | void get_ml (keyword kend, shstr &sh); |
458 | void get_ml (keyword kend, shstr &sh); |
459 | |
459 | |
460 | void get_ornull (shstr &sh) const { sh = value; } |
460 | void get_ornull (shstr &sh) const { sh = value; } |
461 | void get (shstr &sh) const { sh = value; } // might want to check for non-null here |
461 | void get (shstr &sh) const { sh = value; } // might want to check for non-null here |
462 | |
462 | |
… | |
… | |
476 | void get (sint32 &i) { i = get_sint32 (); } |
476 | void get (sint32 &i) { i = get_sint32 (); } |
477 | |
477 | |
478 | void get (uint32 &i) { i = get_sint64 (); } |
478 | void get (uint32 &i) { i = get_sint64 (); } |
479 | void get (sint64 &i) { i = get_sint64 (); } |
479 | void get (sint64 &i) { i = get_sint64 (); } |
480 | |
480 | |
481 | MTH void parse_warn (const char *msg); |
481 | MTH void parse_warn (const_utf8_string msg); |
482 | MTH bool parse_error (const char *type = 0, const char *name = 0, bool skip = true); |
482 | MTH bool parse_error (const_utf8_string type = 0, const_utf8_string name = 0, bool skip = true); |
483 | |
483 | |
484 | struct delayed_ref { |
484 | struct delayed_ref { |
485 | attachable *op; |
485 | attachable *op; |
486 | object_ptr *ptr; |
486 | object_ptr *ptr; |
487 | const char *ref; |
487 | const_utf8_string ref; |
488 | }; |
488 | }; |
489 | std::vector<delayed_ref> delrefs; |
489 | std::vector<delayed_ref> delrefs; |
490 | |
490 | |
491 | void delayed_deref (attachable *op, object_ptr &ptr, const char *ref); |
491 | void delayed_deref (attachable *op, object_ptr &ptr, const_utf8_string ref); |
492 | MTH void resolve_delayed_derefs (bool deref = true); |
492 | MTH void resolve_delayed_derefs (bool deref = true); |
493 | }; |
493 | }; |
494 | |
494 | |
495 | //TODO: remove |
495 | //TODO: remove |
496 | char *fgets (char *s, int n, object_thawer &thawer); |
496 | char *fgets (char *s, int n, object_thawer &thawer); |