ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/server/server/cfperl.xs
(Generate patch)

Comparing deliantra/server/server/cfperl.xs (file contents):
Revision 1.39 by elmex, Fri Sep 1 17:16:47 2006 UTC vs.
Revision 1.41 by root, Sun Sep 3 22:45:57 2006 UTC

51 51
52static f_plug_api gethook = cfapi_get_hooks; 52static f_plug_api gethook = cfapi_get_hooks;
53static f_plug_api object_set_property = cfapi_object_set_property; 53static f_plug_api object_set_property = cfapi_object_set_property;
54static f_plug_api object_insert = cfapi_object_insert; 54static f_plug_api object_insert = cfapi_object_insert;
55 55
56static bool perl_booted;
57
58/* this is a stupid way to do things, and awkward to use for plug-in authors */ 56/* this is a stupid way to do things, and awkward to use for plug-in authors */
59typedef struct 57typedef struct
60{ 58{
61 object* who; 59 object* who;
62 object* activator; 60 object* activator;
326 { 324 {
327 SvREFCNT_dec (cb); 325 SvREFCNT_dec (cb);
328 cb = 0; 326 cb = 0;
329 } 327 }
330 328
331 if (attach)
332 {
333 free_string (attach);
334 attach = 0; 329 attach = 0;
335 }
336} 330}
337 331
338void attachable_base::optimise () 332void attachable_base::optimise ()
339{ 333{
340 if (!self) 334 if (!self)
359 PUSHMARK (SP); 353 PUSHMARK (SP);
360 EXTEND (SP, 2); 354 EXTEND (SP, 2);
361 PUSHs (sv_2mortal (newSVdt (type, obj))); 355 PUSHs (sv_2mortal (newSVdt (type, obj)));
362 PUSHs (sv_2mortal (newSVpv (attach, 0))); 356 PUSHs (sv_2mortal (newSVpv (attach, 0)));
363 357
364 free_string (attach);
365 attach = 0; 358 attach = 0;
366 359
367 PUTBACK; 360 PUTBACK;
368 call_pv ("cf::instantiate", G_DISCARD | G_VOID | G_EVAL); 361 call_pv ("cf::instantiate", G_DISCARD | G_VOID | G_EVAL);
369 FREETMPS; 362 FREETMPS;
472 freezer.add (s); 465 freezer.add (s);
473} 466}
474 467
475object_thawer::object_thawer (const char *filename) 468object_thawer::object_thawer (const char *filename)
476{ 469{
470 static const char eof[] = "\n\n\n\0\0\0";
471
477 av = 0; 472 av = 0;
478 fp = 0; 473 text = 0;
479 474
480 if (!filename) 475 if (filename)
481 return;
482
483 fp = fopen (filename, "r");
484 if (!fp)
485 {
486 LOG (llevError, "object_thawer: unable to open '%s': %s.\n", filename, strerror (errno));
487 return;
488 }
489
490 if (perl_booted)
491 { 476 {
492 dSP; 477 dSP;
493 ENTER; 478 ENTER;
494 SAVETMPS; 479 SAVETMPS;
495 PUSHMARK (SP); 480 PUSHMARK (SP);
496 XPUSHs (sv_2mortal (newSVpv (filename, 0))); 481 XPUSHs (sv_2mortal (newSVpv (filename, 0)));
497 PUTBACK; 482 PUTBACK;
498 483
499 if (0 < call_pv ("cf::object_thawer_load", G_SCALAR | G_EVAL)) 484 if (2 == call_pv ("cf::object_thawer_load", G_ARRAY | G_EVAL))
500 { 485 {
501 SPAGAIN; 486 SPAGAIN;
487
488 // second value - perl objects
489 {
502 SV *sv = POPs; 490 SV *sv = POPs;
503 if (SvROK (sv)) 491 if (SvROK (sv))
504 av = (AV *)SvREFCNT_inc (SvRV (sv)); 492 av = (AV *)SvREFCNT_inc (SvRV (sv));
493 }
494
495 // first value - text part, pad with 3 zeroes
496 {
497 SV *sv = POPs;
498 STRLEN len;
499 char *sv_ = SvPVbyte (sv, len);
500 text = newSV (len + sizeof (eof));
501 SvCUR_set (text, len);
502 memcpy (SvPVX (text), sv_, len);
503 memcpy (SvEND (text), eof, sizeof (eof)); // just to be sure
504 }
505 } 505 }
506 506
507 PUTBACK;
507 FREETMPS; 508 FREETMPS;
508 LEAVE; 509 LEAVE;
509 } 510 }
511
512 if (!text)
513 text = newSVpvn (eof, sizeof (eof));
514
515 line = SvPVX (text);
510} 516}
511 517
512void object_thawer::get (data_type type, void *obj, attachable_base *ext, int oid) 518void object_thawer::get (data_type type, void *obj, attachable_base *ext, int oid)
513{ 519{
514 if (!av || oid < 0) // this is actually an error of sorts 520 if (!av || oid < 0) // this is actually an error of sorts
531 reattach (type, obj); 537 reattach (type, obj);
532} 538}
533 539
534object_thawer::~object_thawer () 540object_thawer::~object_thawer ()
535{ 541{
536 if (fp) fclose (fp); 542 if (text) SvREFCNT_dec (text);
537 if (av) SvREFCNT_dec ((AV *)av); 543 if (av) SvREFCNT_dec (av);
538} 544}
539 545
546char *fgets (char *s, int n, object_thawer &thawer)
547{
548 char *p = thawer.line;
549 char *q = s;
550
551 while (--n)
552 {
553 if (!*p)
554 break;
555
556 *q++ = *p;
557
558 if (*p++ == '\n')
559 break;
560 }
561
562 *q = 0;
563 thawer.line = p;
564
565 return s == q ? 0 : s;
566}
567
540token object_thawer::get_token () 568keyword object_thawer::get_kv ()
541{ 569{
542#if 0
543 for (;;) 570 for (;;)
544 { 571 {
545 if (!fgets (line, sizeof (line), fp)) 572 char *p = line;
546 return token (KW_eof);
547 573
548 unsigned char *p = (unsigned char *)line; 574 if (!*p)
575 return KW_EOF;
549 576
577 // parse keyword
550 while (*p > ' ') 578 while (*p > ' ')
551 p++; 579 p++;
552 580
553 int len = p - (unsigned char *)line; 581 int klen = p - line;
554 582
555 while ((*p - 1) < ' ') 583 if (*p++ != '\n')
556 p++;
557
558 if (*p)
559 { 584 {
560 char *v = p; 585 // parse value
586 while (*p <= ' ' && *p != '\n') // skip 0x01 .. 0x20
587 ++p;
561 588
589 last_value = p;
590
562 while (*p && *p != '\n') 591 while (*p != '\n')
563 p++; 592 p++;
564 593
565 *p = 0; 594 *p++ = 0;
566
567 return token (k, v);
568 } 595 }
569 else 596 else
570 return token (k); 597 last_value = 0;
598
599 line [klen] = 0;
600 keyword_idx *kw = kw_lex::match (line, klen);
601
602 //printf ("KV %d<%s,%s>\n", kw ? kw->index : 0, line, last_value);//D
603
604 if (kw)
605 {
606 line = p;
607 return kw->index;
608 }
609 else if (!*line || *line == '#')
610 {
611 // empty/comment line
612 line = p;
613 }
614 else
615 return KW_ERROR;
571 } 616 }
572#endif
573} 617}
574 618
619void object_thawer::get (shstr &sh) const
620{
621 if (last_value)
622 sh = last_value;
623 else
624 {
625 sh = "<value missing>";
626 LOG (llevError, "keyword requires value: <%s>\n", line);//TODO: add filename
627 }
628}
629
630void object_thawer::get_ml (keyword kend, shstr &sh)
631{
632 char kw[128];
633
634 // multi-line strings are delimited by "\nendXXX\n"
635 kw [0] = '\n';
636 strcpy (kw + 1, keyword_str [kend]);
637
638 char *end = strstr (line, kw);
639
640 if (!end)
641 {
642 sh = 0;
643 return;
644 }
645
646 *end = 0;
647 sh = line;
648
649 line = end + keyword_len [kend] + 1;
650
651 while (*line++ != '\n')
652 ;
653}
654
655sint32 object_thawer::get_sint32 () const
656{
657 char *p = last_value;
658
659 if (!p)
660 return 0;
661
662 sint32 val = 0;
663 bool negate;
664
665 if (*p == '-')
666 {
667 negate = true;
668 ++p;
669 }
670 else
671 negate = false;
672
673 do
674 {
675 val *= 10;
676 val += *p++ - '0';
677 }
678 while (*p);
679
680 return negate ? -val : val;
681}
682
683sint64 object_thawer::get_sint64 () const
684{
685 return last_value ? atoll (last_value) : 0;
686}
687
688double object_thawer::get_double () const
689{
690 return last_value ? atof (last_value) : 0;
691}
692
575///////////////////////////////////////////////////////////////////////////// 693/////////////////////////////////////////////////////////////////////////////
576 694
577extern "C" int cfperl_initPlugin (const char *iversion, f_plug_api gethooksptr) 695extern "C" int cfperl_initPlugin (const char *iversion, f_plug_api gethooksptr)
578{ 696{
579 return 0; 697 return 0;
687 char *argv[] = { 805 char *argv[] = {
688 "", 806 "",
689 "-e" 807 "-e"
690 "cf->bootstrap;" 808 "cf->bootstrap;"
691 "unshift @INC, cf::datadir ();" 809 "unshift @INC, cf::datadir ();"
810 "require cf;"
692 }; 811 };
693 812
694 if (perl_parse (perl, xs_init, 2, argv, (char **)NULL) || perl_run (perl)) 813 if (perl_parse (perl, xs_init, 2, argv, (char **)NULL) || perl_run (perl))
695 { 814 {
696 printf ("unable to initialize perl-interpreter, aborting.\n"); 815 printf ("unable to initialize perl-interpreter, aborting.\n");
697 exit (EXIT_FAILURE); 816 exit (EXIT_FAILURE);
698 } 817 }
699 818
700 obj_cache = newHV (); 819 obj_cache = newHV ();
701}
702
703void cfperl_boot ()
704{
705 perl_booted = true;
706
707 eval_pv ("require cf", 1);
708} 820}
709 821
710void cfperl_main () 822void cfperl_main ()
711{ 823{
712 dSP; 824 dSP;

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines