ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/server/lib/cf.pm
(Generate patch)

Comparing deliantra/server/lib/cf.pm (file contents):
Revision 1.97 by root, Fri Dec 22 06:03:20 2006 UTC vs.
Revision 1.113 by root, Mon Jan 1 15:32:40 2007 UTC

8use Storable; 8use Storable;
9use Opcode; 9use Opcode;
10use Safe; 10use Safe;
11use Safe::Hole; 11use Safe::Hole;
12 12
13use Coro; 13use Coro 3.3;
14use Coro::Event; 14use Coro::Event;
15use Coro::Timer; 15use Coro::Timer;
16use Coro::Signal; 16use Coro::Signal;
17use Coro::Semaphore; 17use Coro::Semaphore;
18
19use IO::AIO; 18use Coro::AIO;
19
20use Digest::MD5;
21use Fcntl;
22use IO::AIO 2.31 ();
20use YAML::Syck (); 23use YAML::Syck ();
21use Time::HiRes; 24use Time::HiRes;
22 25
23use Event; $Event::Eval = 1; # no idea why this is required, but it is 26use Event; $Event::Eval = 1; # no idea why this is required, but it is
24 27
25# work around bug in YAML::Syck - bad news for perl6, will it be as broken wrt. unicode? 28# work around bug in YAML::Syck - bad news for perl6, will it be as broken wrt. unicode?
26$YAML::Syck::ImplicitUnicode = 1; 29$YAML::Syck::ImplicitUnicode = 1;
27 30
28$Coro::main->prio (Coro::PRIO_MIN); 31$Coro::main->prio (2); # run main coroutine ("the server") with very high priority
29 32
30sub WF_AUTOCANCEL () { 1 } # automatically cancel this watcher on reload 33sub WF_AUTOCANCEL () { 1 } # automatically cancel this watcher on reload
31 34
32our %COMMAND = (); 35our %COMMAND = ();
33our %COMMAND_TIME = (); 36our %COMMAND_TIME = ();
34our %EXTCMD = (); 37our %EXTCMD = ();
35 38
36_init_vars;
37
38our @EVENT; 39our @EVENT;
39our $LIBDIR = datadir . "/ext"; 40our $LIBDIR = datadir . "/ext";
40 41
41our $TICK = MAX_TIME * 1e-6; 42our $TICK = MAX_TIME * 1e-6;
42our $TICK_WATCHER; 43our $TICK_WATCHER;
43our $NEXT_TICK; 44our $NEXT_TICK;
45our $NOW;
44 46
45our %CFG; 47our %CFG;
46 48
47our $UPTIME; $UPTIME ||= time; 49our $UPTIME; $UPTIME ||= time;
50our $RUNTIME;
51
52our %MAP; # all maps
53our $LINK_MAP; # the special {link} map
54our $FREEZE;
55our $RANDOM_MAPS = cf::localdir . "/random";
56our %EXT_CORO;
57
58binmode STDOUT;
59binmode STDERR;
60
61# read virtual server time, if available
62unless ($RUNTIME || !-e cf::localdir . "/runtime") {
63 open my $fh, "<", cf::localdir . "/runtime"
64 or die "unable to read runtime file: $!";
65 $RUNTIME = <$fh> + 0.;
66}
67
68mkdir cf::localdir;
69mkdir cf::localdir . "/" . cf::playerdir;
70mkdir cf::localdir . "/" . cf::tmpdir;
71mkdir cf::localdir . "/" . cf::uniquedir;
72mkdir $RANDOM_MAPS;
73
74# a special map that is always available
75our $LINK_MAP;
76
77our $EMERGENCY_POSITION = $cf::CFG{emergency_position} || ["/world/world_105_115", 5, 37];
48 78
49############################################################################# 79#############################################################################
50 80
51=head2 GLOBAL VARIABLES 81=head2 GLOBAL VARIABLES
52 82
53=over 4 83=over 4
54 84
55=item $cf::UPTIME 85=item $cf::UPTIME
56 86
57The timestamp of the server start (so not actually an uptime). 87The timestamp of the server start (so not actually an uptime).
88
89=item $cf::RUNTIME
90
91The time this server has run, starts at 0 and is increased by $cf::TICK on
92every server tick.
58 93
59=item $cf::LIBDIR 94=item $cf::LIBDIR
60 95
61The perl library directory, where extensions and cf-specific modules can 96The perl library directory, where extensions and cf-specific modules can
62be found. It will be added to C<@INC> automatically. 97be found. It will be added to C<@INC> automatically.
98
99=item $cf::NOW
100
101The time of the last (current) server tick.
63 102
64=item $cf::TICK 103=item $cf::TICK
65 104
66The interval between server ticks, in seconds. 105The interval between server ticks, in seconds.
67 106
75=cut 114=cut
76 115
77BEGIN { 116BEGIN {
78 *CORE::GLOBAL::warn = sub { 117 *CORE::GLOBAL::warn = sub {
79 my $msg = join "", @_; 118 my $msg = join "", @_;
119 utf8::encode $msg;
120
80 $msg .= "\n" 121 $msg .= "\n"
81 unless $msg =~ /\n$/; 122 unless $msg =~ /\n$/;
82 123
83 print STDERR "cfperl: $msg";
84 LOG llevError, "cfperl: $msg"; 124 LOG llevError, "cfperl: $msg";
85 }; 125 };
86} 126}
87 127
88@safe::cf::global::ISA = @cf::global::ISA = 'cf::attachable'; 128@safe::cf::global::ISA = @cf::global::ISA = 'cf::attachable';
93@safe::cf::object::player::ISA = @cf::object::player::ISA = 'cf::object'; 133@safe::cf::object::player::ISA = @cf::object::player::ISA = 'cf::object';
94 134
95# we bless all objects into (empty) derived classes to force a method lookup 135# we bless all objects into (empty) derived classes to force a method lookup
96# within the Safe compartment. 136# within the Safe compartment.
97for my $pkg (qw( 137for my $pkg (qw(
98 cf::global 138 cf::global cf::attachable
99 cf::object cf::object::player 139 cf::object cf::object::player
100 cf::client cf::player 140 cf::client cf::player
101 cf::arch cf::living 141 cf::arch cf::living
102 cf::map cf::party cf::region 142 cf::map cf::party cf::region
103)) { 143)) {
141sub to_json($) { 181sub to_json($) {
142 $JSON::Syck::ImplicitUnicode = 0; # work around JSON::Syck bugs 182 $JSON::Syck::ImplicitUnicode = 0; # work around JSON::Syck bugs
143 JSON::Syck::Dump $_[0] 183 JSON::Syck::Dump $_[0]
144} 184}
145 185
186=item cf::sync_job { BLOCK }
187
188The design of crossfire+ requires that the main coro ($Coro::main) is
189always able to handle events or runnable, as crossfire+ is only partly
190reentrant. Thus "blocking" it by e.g. waiting for I/O is not acceptable.
191
192If it must be done, put the blocking parts into C<sync_job>. This will run
193the given BLOCK in another coroutine while waiting for the result. The
194server will be frozen during this time, so the block should either finish
195fast or be very important.
196
197=cut
198
199sub sync_job(&) {
200 my ($job) = @_;
201
202 if ($Coro::current == $Coro::main) {
203 # this is the main coro, too bad, we have to block
204 # till the operation succeeds, freezing the server :/
205
206 # TODO: use suspend/resume instead
207 # (but this is cancel-safe)
208 local $FREEZE = 1;
209
210 my $busy = 1;
211 my @res;
212
213 (Coro::async {
214 @res = eval { $job->() };
215 warn $@ if $@;
216 undef $busy;
217 })->prio (Coro::PRIO_MAX);
218
219 while ($busy) {
220 Coro::cede_notself;
221 Event::one_event unless Coro::nready;
222 }
223
224 wantarray ? @res : $res[0]
225 } else {
226 # we are in another coroutine, how wonderful, everything just works
227
228 $job->()
229 }
230}
231
232=item $coro = cf::coro { BLOCK }
233
234Creates and returns a new coro. This coro is automcatially being canceled
235when the extension calling this is being unloaded.
236
237=cut
238
239sub coro(&) {
240 my $cb = shift;
241
242 my $coro; $coro = async {
243 eval {
244 $cb->();
245 };
246 warn $@ if $@;
247 };
248
249 $coro->on_destroy (sub {
250 delete $EXT_CORO{$coro+0};
251 });
252 $EXT_CORO{$coro+0} = $coro;
253
254 $coro
255}
256
257sub write_runtime {
258 my $runtime = cf::localdir . "/runtime";
259
260 my $fh = aio_open "$runtime~", O_WRONLY | O_CREAT, 0644
261 or return;
262
263 my $value = $cf::RUNTIME + 1 + 10; # 10 is the runtime save interval, for a monotonic clock
264 (aio_write $fh, 0, (length $value), $value, 0) <= 0
265 and return;
266
267 aio_fsync $fh
268 and return;
269
270 close $fh
271 or return;
272
273 aio_rename "$runtime~", $runtime
274 and return;
275
276 1
277}
278
146=back 279=back
147 280
148=cut 281=cut
282
283#############################################################################
284
285package cf::path;
286
287sub new {
288 my ($class, $path, $base) = @_;
289
290 $path = $path->as_string if ref $path;
291
292 my $self = bless { }, $class;
293
294 if ($path =~ s{^\?random/}{}) {
295 Coro::AIO::aio_load "$cf::RANDOM_MAPS/$path.meta", my $data;
296 $self->{random} = cf::from_json $data;
297 } else {
298 if ($path =~ s{^~([^/]+)?}{}) {
299 $self->{user_rel} = 1;
300
301 if (defined $1) {
302 $self->{user} = $1;
303 } elsif ($base =~ m{^~([^/]+)/}) {
304 $self->{user} = $1;
305 } else {
306 warn "cannot resolve user-relative path without user <$path,$base>\n";
307 }
308 } elsif ($path =~ /^\//) {
309 # already absolute
310 } else {
311 $base =~ s{[^/]+/?$}{};
312 return $class->new ("$base/$path");
313 }
314
315 for ($path) {
316 redo if s{/\.?/}{/};
317 redo if s{/[^/]+/\.\./}{/};
318 }
319 }
320
321 $self->{path} = $path;
322
323 $self
324}
325
326# the name / primary key / in-game path
327sub as_string {
328 my ($self) = @_;
329
330 $self->{user_rel} ? "~$self->{user}$self->{path}"
331 : $self->{random} ? "?random/$self->{path}"
332 : $self->{path}
333}
334
335# the displayed name, this is a one way mapping
336sub visible_name {
337 my ($self) = @_;
338
339# if (my $rmp = $self->{random}) {
340# # todo: be more intelligent about this
341# "?random/$rmp->{origin_map}+$rmp->{origin_x}+$rmp->{origin_y}/$rmp->{dungeon_level}"
342# } else {
343 $self->as_string
344# }
345}
346
347# escape the /'s in the path
348sub _escaped_path {
349 # ∕ is U+2215
350 (my $path = $_[0]{path}) =~ s/\//∕/g;
351 $path
352}
353
354# the original (read-only) location
355sub load_path {
356 my ($self) = @_;
357
358 sprintf "%s/%s/%s", cf::datadir, cf::mapdir, $self->{path}
359}
360
361# the temporary/swap location
362sub save_path {
363 my ($self) = @_;
364
365 $self->{user_rel} ? sprintf "%s/%s/%s/%s", cf::localdir, cf::playerdir, $self->{user}, $self->_escaped_path
366 : $self->{random} ? sprintf "%s/%s", $RANDOM_MAPS, $self->{path}
367 : sprintf "%s/%s/%s", cf::localdir, cf::tmpdir, $self->_escaped_path
368}
369
370# the unique path, might be eq to save_path
371sub uniq_path {
372 my ($self) = @_;
373
374 $self->{user_rel} || $self->{random}
375 ? undef
376 : sprintf "%s/%s/%s", cf::localdir, cf::uniquedir, $self->_escaped_path
377}
378
379# return random map parameters, or undef
380sub random_map_params {
381 my ($self) = @_;
382
383 $self->{random}
384}
385
386# this is somewhat ugly, but style maps do need special treatment
387sub is_style_map {
388 $_[0]{path} =~ m{^/styles/}
389}
390
391package cf;
149 392
150############################################################################# 393#############################################################################
151 394
152=head2 ATTACHABLE OBJECTS 395=head2 ATTACHABLE OBJECTS
153 396
269exception. 512exception.
270 513
271=cut 514=cut
272 515
273# the following variables are defined in .xs and must not be re-created 516# the following variables are defined in .xs and must not be re-created
274our @CB_GLOBAL = (); # registry for all global events 517our @CB_GLOBAL = (); # registry for all global events
518our @CB_ATTACHABLE = (); # registry for all attachables
275our @CB_OBJECT = (); # all objects (should not be used except in emergency) 519our @CB_OBJECT = (); # all objects (should not be used except in emergency)
276our @CB_PLAYER = (); 520our @CB_PLAYER = ();
277our @CB_CLIENT = (); 521our @CB_CLIENT = ();
278our @CB_TYPE = (); # registry for type (cf-object class) based events 522our @CB_TYPE = (); # registry for type (cf-object class) based events
279our @CB_MAP = (); 523our @CB_MAP = ();
280 524
281my %attachment; 525my %attachment;
282 526
283sub _attach_cb($$$$) { 527sub _attach_cb($$$$) {
284 my ($registry, $event, $prio, $cb) = @_; 528 my ($registry, $event, $prio, $cb) = @_;
289 533
290 @{$registry->[$event]} = sort 534 @{$registry->[$event]} = sort
291 { $a->[0] cmp $b->[0] } 535 { $a->[0] cmp $b->[0] }
292 @{$registry->[$event] || []}, $cb; 536 @{$registry->[$event] || []}, $cb;
293} 537}
538
539# hack
540my %attachable_klass = map +($_ => 1), KLASS_OBJECT, KLASS_CLIENT, KLASS_PLAYER, KLASS_MAP;
294 541
295# attach handles attaching event callbacks 542# attach handles attaching event callbacks
296# the only thing the caller has to do is pass the correct 543# the only thing the caller has to do is pass the correct
297# registry (== where the callback attaches to). 544# registry (== where the callback attaches to).
298sub _attach { 545sub _attach {
300 547
301 my $object_type; 548 my $object_type;
302 my $prio = 0; 549 my $prio = 0;
303 my %cb_id = map +("on_" . lc $EVENT[$_][0], $_) , grep $EVENT[$_][1] == $klass, 0 .. $#EVENT; 550 my %cb_id = map +("on_" . lc $EVENT[$_][0], $_) , grep $EVENT[$_][1] == $klass, 0 .. $#EVENT;
304 551
552 #TODO: get rid of this hack
553 if ($attachable_klass{$klass}) {
554 %cb_id = (%cb_id, map +("on_" . lc $EVENT[$_][0], $_) , grep $EVENT[$_][1] == KLASS_ATTACHABLE, 0 .. $#EVENT);
555 }
556
305 while (@arg) { 557 while (@arg) {
306 my $type = shift @arg; 558 my $type = shift @arg;
307 559
308 if ($type eq "prio") { 560 if ($type eq "prio") {
309 $prio = shift @arg; 561 $prio = shift @arg;
384 my ($obj, $name) = @_; 636 my ($obj, $name) = @_;
385 637
386 exists $obj->{_attachment}{$name} 638 exists $obj->{_attachment}{$name}
387} 639}
388 640
389for my $klass (qw(GLOBAL OBJECT PLAYER CLIENT MAP)) { 641for my $klass (qw(ATTACHABLE GLOBAL OBJECT PLAYER CLIENT MAP)) {
390 eval "#line " . __LINE__ . " 'cf.pm' 642 eval "#line " . __LINE__ . " 'cf.pm'
391 sub cf::\L$klass\E::_attach_registry { 643 sub cf::\L$klass\E::_attach_registry {
392 (\\\@CB_$klass, KLASS_$klass) 644 (\\\@CB_$klass, KLASS_$klass)
393 } 645 }
394 646
447=cut 699=cut
448 700
449############################################################################# 701#############################################################################
450# object support 702# object support
451 703
452sub instantiate {
453 my ($obj, $data) = @_;
454
455 $data = from_json $data;
456
457 for (@$data) {
458 my ($name, $args) = @$_;
459
460 $obj->attach ($name, %{$args || {} });
461 }
462}
463
464# basically do the same as instantiate, without calling instantiate
465sub reattach { 704sub reattach {
705 # basically do the same as instantiate, without calling instantiate
466 my ($obj) = @_; 706 my ($obj) = @_;
707
467 my $registry = $obj->registry; 708 my $registry = $obj->registry;
468 709
469 @$registry = (); 710 @$registry = ();
470 711
471 delete $obj->{_attachment} unless scalar keys %{ $obj->{_attachment} || {} }; 712 delete $obj->{_attachment} unless scalar keys %{ $obj->{_attachment} || {} };
480 warn "object uses attachment '$name' that is not available, postponing.\n"; 721 warn "object uses attachment '$name' that is not available, postponing.\n";
481 } 722 }
482 } 723 }
483} 724}
484 725
485sub object_freezer_save { 726cf::attachable->attach (
486 my ($filename, $rdata, $objs) = @_;
487
488 if (length $$rdata) {
489 warn sprintf "saving %s (%d,%d)\n",
490 $filename, length $$rdata, scalar @$objs;
491
492 if (open my $fh, ">:raw", "$filename~") {
493 chmod SAVE_MODE, $fh;
494 syswrite $fh, $$rdata;
495 close $fh;
496
497 if (@$objs && open my $fh, ">:raw", "$filename.pst~") {
498 chmod SAVE_MODE, $fh;
499 syswrite $fh, Storable::nfreeze { version => 1, objs => $objs };
500 close $fh;
501 rename "$filename.pst~", "$filename.pst";
502 } else {
503 unlink "$filename.pst";
504 }
505
506 rename "$filename~", $filename;
507 } else {
508 warn "FATAL: $filename~: $!\n";
509 }
510 } else {
511 unlink $filename;
512 unlink "$filename.pst";
513 }
514}
515
516sub object_freezer_as_string {
517 my ($rdata, $objs) = @_;
518
519 use Data::Dumper;
520
521 $$rdata . Dumper $objs
522}
523
524sub object_thawer_load {
525 my ($filename) = @_;
526
527 local $/;
528
529 my $av;
530
531 #TODO: use sysread etc.
532 if (open my $data, "<:raw:perlio", $filename) {
533 $data = <$data>;
534 if (open my $pst, "<:raw:perlio", "$filename.pst") {
535 $av = eval { (Storable::thaw <$pst>)->{objs} };
536 }
537 return ($data, $av);
538 }
539
540 ()
541}
542
543cf::object->attach (
544 prio => -1000000, 727 prio => -1000000,
728 on_instantiate => sub {
729 my ($obj, $data) = @_;
730
731 $data = from_json $data;
732
733 for (@$data) {
734 my ($name, $args) = @$_;
735
736 $obj->attach ($name, %{$args || {} });
737 }
738 },
739 on_reattach => \&reattach,
545 on_clone => sub { 740 on_clone => sub {
546 my ($src, $dst) = @_; 741 my ($src, $dst) = @_;
547 742
548 @{$dst->registry} = @{$src->registry}; 743 @{$dst->registry} = @{$src->registry};
549 744
551 746
552 %{$dst->{_attachment}} = %{$src->{_attachment}} 747 %{$dst->{_attachment}} = %{$src->{_attachment}}
553 if exists $src->{_attachment}; 748 if exists $src->{_attachment};
554 }, 749 },
555); 750);
751
752sub object_freezer_save {
753 my ($filename, $rdata, $objs) = @_;
754
755 sync_job {
756 if (length $$rdata) {
757 warn sprintf "saving %s (%d,%d)\n",
758 $filename, length $$rdata, scalar @$objs;
759
760 if (my $fh = aio_open "$filename~", O_WRONLY | O_CREAT, 0600) {
761 chmod SAVE_MODE, $fh;
762 aio_write $fh, 0, (length $$rdata), $$rdata, 0;
763 aio_fsync $fh;
764 close $fh;
765
766 if (@$objs) {
767 if (my $fh = aio_open "$filename.pst~", O_WRONLY | O_CREAT, 0600) {
768 chmod SAVE_MODE, $fh;
769 my $data = Storable::nfreeze { version => 1, objs => $objs };
770 aio_write $fh, 0, (length $data), $data, 0;
771 aio_fsync $fh;
772 close $fh;
773 aio_rename "$filename.pst~", "$filename.pst";
774 }
775 } else {
776 aio_unlink "$filename.pst";
777 }
778
779 aio_rename "$filename~", $filename;
780 } else {
781 warn "FATAL: $filename~: $!\n";
782 }
783 } else {
784 aio_unlink $filename;
785 aio_unlink "$filename.pst";
786 }
787 }
788}
789
790sub object_freezer_as_string {
791 my ($rdata, $objs) = @_;
792
793 use Data::Dumper;
794
795 $$rdata . Dumper $objs
796}
797
798sub object_thawer_load {
799 my ($filename) = @_;
800
801 my ($data, $av);
802
803 (aio_load $filename, $data) >= 0
804 or return;
805
806 unless (aio_stat "$filename.pst") {
807 (aio_load "$filename.pst", $av) >= 0
808 or return;
809 $av = eval { (Storable::thaw $av)->{objs} };
810 }
811
812 return ($data, $av);
813}
556 814
557############################################################################# 815#############################################################################
558# command handling &c 816# command handling &c
559 817
560=item cf::register_command $name => \&callback($ob,$args); 818=item cf::register_command $name => \&callback($ob,$args);
780 $self->send ("ext " . to_json \%msg); 1038 $self->send ("ext " . to_json \%msg);
781} 1039}
782 1040
783=back 1041=back
784 1042
1043
1044=head3 cf::map
1045
1046=over 4
1047
1048=cut
1049
1050package cf::map;
1051
1052use Fcntl;
1053use Coro::AIO;
1054
1055our $MAX_RESET = 7200;
1056our $DEFAULT_RESET = 3600;
1057
1058sub generate_random_map {
1059 my ($path, $rmp) = @_;
1060
1061 # mit "rum" bekleckern, nicht
1062 cf::map::_create_random_map
1063 $path,
1064 $rmp->{wallstyle}, $rmp->{wall_name}, $rmp->{floorstyle}, $rmp->{monsterstyle},
1065 $rmp->{treasurestyle}, $rmp->{layoutstyle}, $rmp->{doorstyle}, $rmp->{decorstyle},
1066 $rmp->{origin_map}, $rmp->{final_map}, $rmp->{exitstyle}, $rmp->{this_map},
1067 $rmp->{exit_on_final_map},
1068 $rmp->{xsize}, $rmp->{ysize},
1069 $rmp->{expand2x}, $rmp->{layoutoptions1}, $rmp->{layoutoptions2}, $rmp->{layoutoptions3},
1070 $rmp->{symmetry}, $rmp->{difficulty}, $rmp->{difficulty_given}, $rmp->{difficulty_increase},
1071 $rmp->{dungeon_level}, $rmp->{dungeon_depth}, $rmp->{decoroptions}, $rmp->{orientation},
1072 $rmp->{origin_y}, $rmp->{origin_x}, $rmp->{random_seed}, $rmp->{total_map_hp},
1073 $rmp->{map_layout_style}, $rmp->{treasureoptions}, $rmp->{symmetry_used},
1074 (cf::region::find $rmp->{region})
1075}
1076
1077# and all this just because we cannot iterate over
1078# all maps in C++...
1079sub change_all_map_light {
1080 my ($change) = @_;
1081
1082 $_->change_map_light ($change) for values %cf::MAP;
1083}
1084
1085sub try_load_header($) {
1086 my ($path) = @_;
1087
1088 utf8::encode $path;
1089 aio_open $path, O_RDONLY, 0
1090 or return;
1091
1092 my $map = cf::map::new
1093 or return;
1094
1095 $map->load_header ($path)
1096 or return;
1097
1098 $map->{load_path} = $path;
1099
1100 $map
1101}
1102
1103sub find_map {
1104 my ($path, $origin) = @_;
1105
1106 #warn "find_map<$path,$origin>\n";#d#
1107
1108 $path = new cf::path $path, $origin && $origin->path;
1109 my $key = $path->as_string;
1110
1111 $cf::MAP{$key} || do {
1112 # do it the slow way
1113 my $map = try_load_header $path->save_path;
1114
1115 if ($map) {
1116 # safety
1117 $map->{instantiate_time} = $cf::RUNTIME
1118 if $map->{instantiate_time} > $cf::RUNTIME;
1119 } else {
1120 if (my $rmp = $path->random_map_params) {
1121 $map = generate_random_map $key, $rmp;
1122 } else {
1123 $map = try_load_header $path->load_path;
1124 }
1125
1126 $map or return;
1127
1128 $map->{load_original} = 1;
1129 $map->{instantiate_time} = $cf::RUNTIME;
1130 $map->instantiate;
1131
1132 # per-player maps become, after loading, normal maps
1133 $map->per_player (0) if $path->{user_rel};
1134 }
1135
1136 $map->path ($key);
1137 $map->{path} = $path;
1138 $map->last_access ($cf::RUNTIME);
1139
1140 if ($map->should_reset) {
1141 $map->reset;
1142 $map = find_map $path;
1143 }
1144
1145 $cf::MAP{$key} = $map
1146 }
1147}
1148
1149sub load {
1150 my ($self) = @_;
1151
1152 return if $self->in_memory != cf::MAP_SWAPPED;
1153
1154 $self->in_memory (cf::MAP_LOADING);
1155
1156 my $path = $self->{path};
1157
1158 $self->alloc;
1159 $self->load_objects ($self->{load_path}, 1)
1160 or return;
1161
1162 $self->set_object_flag (cf::FLAG_OBJ_ORIGINAL, 1)
1163 if delete $self->{load_original};
1164
1165 if (my $uniq = $path->uniq_path) {
1166 utf8::encode $uniq;
1167 if (aio_open $uniq, O_RDONLY, 0) {
1168 $self->clear_unique_items;
1169 $self->load_objects ($uniq, 0);
1170 }
1171 }
1172
1173 # now do the right thing for maps
1174 $self->link_multipart_objects;
1175
1176 if ($self->{path}->is_style_map) {
1177 $self->{deny_save} = 1;
1178 $self->{deny_reset} = 1;
1179 } else {
1180 $self->fix_auto_apply;
1181 $self->decay_objects;
1182 $self->update_buttons;
1183 $self->set_darkness_map;
1184 $self->difficulty ($self->estimate_difficulty)
1185 unless $self->difficulty;
1186 $self->activate;
1187 }
1188
1189 $self->in_memory (cf::MAP_IN_MEMORY);
1190}
1191
1192sub load_map_sync {
1193 my ($path, $origin) = @_;
1194
1195 #warn "load_map_sync<$path, $origin>\n";#d#
1196
1197 cf::sync_job {
1198 my $map = cf::map::find_map $path, $origin
1199 or return;
1200 $map->load;
1201 $map
1202 }
1203}
1204
1205sub save {
1206 my ($self) = @_;
1207
1208 my $save = $self->{path}->save_path; utf8::encode $save;
1209 my $uniq = $self->{path}->uniq_path; utf8::encode $uniq;
1210
1211 $self->{last_save} = $cf::RUNTIME;
1212
1213 return unless $self->dirty;
1214
1215 $self->{load_path} = $save;
1216
1217 return if $self->{deny_save};
1218
1219 if ($uniq) {
1220 $self->save_objects ($save, cf::IO_HEADER | cf::IO_OBJECTS);
1221 $self->save_objects ($uniq, cf::IO_UNIQUES);
1222 } else {
1223 $self->save_objects ($save, cf::IO_HEADER | cf::IO_OBJECTS | cf::IO_UNIQUES);
1224 }
1225}
1226
1227sub swap_out {
1228 my ($self) = @_;
1229
1230 return if $self->players;
1231 return if $self->in_memory != cf::MAP_IN_MEMORY;
1232 return if $self->{deny_save};
1233
1234 $self->save;
1235 $self->clear;
1236 $self->in_memory (cf::MAP_SWAPPED);
1237}
1238
1239sub reset_at {
1240 my ($self) = @_;
1241
1242 # TODO: safety, remove and allow resettable per-player maps
1243 return 1e100 if $self->{path}{user_rel};
1244 return 1e100 if $self->{deny_reset};
1245
1246 my $time = $self->fixed_resettime ? $self->{instantiate_time} : $self->last_access;
1247 my $to = $self->reset_timeout || $DEFAULT_RESET;
1248 $to = $MAX_RESET if $to > $MAX_RESET;
1249
1250 $time + $to
1251}
1252
1253sub should_reset {
1254 my ($self) = @_;
1255
1256 $self->reset_at <= $cf::RUNTIME
1257}
1258
1259sub unlink_save {
1260 my ($self) = @_;
1261
1262 utf8::encode (my $save = $self->{path}->save_path);
1263 aioreq_pri 3; IO::AIO::aio_unlink $save;
1264 aioreq_pri 3; IO::AIO::aio_unlink "$save.pst";
1265}
1266
1267sub rename {
1268 my ($self, $new_path) = @_;
1269
1270 $self->unlink_save;
1271
1272 delete $cf::MAP{$self->path};
1273 $self->{path} = new cf::path $new_path;
1274 $self->path ($self->{path}->path);
1275 $cf::MAP{$self->path} = $self;
1276
1277 $self->save;
1278}
1279
1280sub reset {
1281 my ($self) = @_;
1282
1283 return if $self->players;
1284 return if $self->{path}{user_rel};#d#
1285
1286 warn "resetting map ", $self->path;#d#
1287
1288 delete $cf::MAP{$self->path};
1289
1290 $_->clear_links_to ($self) for values %cf::MAP;
1291
1292 $self->unlink_save;
1293 $self->destroy;
1294}
1295
1296sub customise_for {
1297 my ($map, $ob) = @_;
1298
1299 if ($map->per_player) {
1300 return cf::map::find_map "~" . $ob->name . "/" . $map->{path}{path};
1301 }
1302
1303 $map
1304}
1305
1306sub emergency_save {
1307 local $cf::FREEZE = 1;
1308
1309 warn "enter emergency map save\n";
1310
1311 cf::sync_job {
1312 warn "begin emergency map save\n";
1313 $_->save for values %cf::MAP;
1314 };
1315
1316 warn "end emergency map save\n";
1317}
1318
1319package cf;
1320
1321=back
1322
1323
785=head3 cf::object::player 1324=head3 cf::object::player
786 1325
787=over 4 1326=over 4
788 1327
789=item $player_object->reply ($npc, $msg[, $flags]) 1328=item $player_object->reply ($npc, $msg[, $flags])
824 (ref $cf::CFG{"may_$access"} 1363 (ref $cf::CFG{"may_$access"}
825 ? scalar grep $self->name eq $_, @{$cf::CFG{"may_$access"}} 1364 ? scalar grep $self->name eq $_, @{$cf::CFG{"may_$access"}}
826 : $cf::CFG{"may_$access"}) 1365 : $cf::CFG{"may_$access"})
827} 1366}
828 1367
1368sub cf::object::player::enter_link {
1369 my ($self) = @_;
1370
1371 return if $self->map == $LINK_MAP;
1372
1373 $self->{_link_pos} = [$self->map->{path}, $self->x, $self->y]
1374 if $self->map;
1375
1376 $self->enter_map ($LINK_MAP, 20, 20);
1377 $self->deactivate_recursive;
1378}
1379
1380sub cf::object::player::leave_link {
1381 my ($self, $map, $x, $y) = @_;
1382
1383 my $link_pos = delete $self->{_link_pos};
1384
1385 unless ($map) {
1386 $self->message ("The exit is closed", cf::NDI_UNIQUE | cf::NDI_RED);
1387
1388 # restore original map position
1389 ($map, $x, $y) = @{ $link_pos || [] };
1390 $map = cf::map::find_map $map;
1391
1392 unless ($map) {
1393 ($map, $x, $y) = @$EMERGENCY_POSITION;
1394 $map = cf::map::find_map $map
1395 or die "FATAL: cannot load emergency map\n";
1396 }
1397 }
1398
1399 ($x, $y) = (-1, -1)
1400 unless (defined $x) && (defined $y);
1401
1402 # use -1 or undef as default coordinates, not 0, 0
1403 ($x, $y) = ($map->enter_x, $map->enter_y)
1404 if $x <=0 && $y <= 0;
1405
1406 $map->load;
1407
1408 $self->activate_recursive;
1409 $self->enter_map ($map, $x, $y);
1410}
1411
1412=item $player_object->goto_map ($map, $x, $y)
1413
1414=cut
1415
1416sub cf::object::player::goto_map {
1417 my ($self, $path, $x, $y) = @_;
1418
1419 $self->enter_link;
1420
1421 (Coro::async {
1422 $path = new cf::path $path;
1423
1424 my $map = cf::map::find_map $path->as_string;
1425 $map = $map->customise_for ($self) if $map;
1426
1427 warn "entering ", $map->path, " at ($x, $y)\n"
1428 if $map;
1429
1430 $self->leave_link ($map, $x, $y);
1431 })->prio (1);
1432}
1433
1434=item $player_object->enter_exit ($exit_object)
1435
1436=cut
1437
1438sub parse_random_map_params {
1439 my ($spec) = @_;
1440
1441 my $rmp = { # defaults
1442 xsize => 10,
1443 ysize => 10,
1444 };
1445
1446 for (split /\n/, $spec) {
1447 my ($k, $v) = split /\s+/, $_, 2;
1448
1449 $rmp->{lc $k} = $v if (length $k) && (length $v);
1450 }
1451
1452 $rmp
1453}
1454
1455sub prepare_random_map {
1456 my ($exit) = @_;
1457
1458 # all this does is basically replace the /! path by
1459 # a new random map path (?random/...) with a seed
1460 # that depends on the exit object
1461
1462 my $rmp = parse_random_map_params $exit->msg;
1463
1464 if ($exit->map) {
1465 $rmp->{region} = $exit->map->region_name;
1466 $rmp->{origin_map} = $exit->map->path;
1467 $rmp->{origin_x} = $exit->x;
1468 $rmp->{origin_y} = $exit->y;
1469 }
1470
1471 $rmp->{random_seed} ||= $exit->random_seed;
1472
1473 my $data = cf::to_json $rmp;
1474 my $md5 = Digest::MD5::md5_hex $data;
1475
1476 if (my $fh = aio_open "$cf::RANDOM_MAPS/$md5.meta", O_WRONLY | O_CREAT, 0666) {
1477 aio_write $fh, 0, (length $data), $data, 0;
1478
1479 $exit->slaying ("?random/$md5");
1480 $exit->msg (undef);
1481 }
1482}
1483
1484sub cf::object::player::enter_exit {
1485 my ($self, $exit) = @_;
1486
1487 return unless $self->type == cf::PLAYER;
1488
1489 $self->enter_link;
1490
1491 (Coro::async {
1492 unless (eval {
1493
1494 prepare_random_map $exit
1495 if $exit->slaying eq "/!";
1496
1497 my $path = new cf::path $exit->slaying, $exit->map && $exit->map->path;
1498 $self->goto_map ($path, $exit->stats->hp, $exit->stats->sp);
1499
1500 1;
1501 }) {
1502 $self->message ("Something went wrong deep within the crossfire server. "
1503 . "I'll try to bring you back to the map you were before. "
1504 . "Please report this to the dungeon master",
1505 cf::NDI_UNIQUE | cf::NDI_RED);
1506
1507 warn "ERROR in enter_exit: $@";
1508 $self->leave_link;
1509 }
1510 })->prio (1);
1511}
1512
829=head3 cf::client 1513=head3 cf::client
830 1514
831=over 4 1515=over 4
832 1516
833=item $client->send_drawinfo ($text, $flags) 1517=item $client->send_drawinfo ($text, $flags)
886 1570
887 if (@{ $ns->{query_queue} } == @$queue) { 1571 if (@{ $ns->{query_queue} } == @$queue) {
888 if (@$queue) { 1572 if (@$queue) {
889 $ns->send_packet ($ns->{query_queue}[0][0]); 1573 $ns->send_packet ($ns->{query_queue}[0][0]);
890 } else { 1574 } else {
891 $ns->state (ST_PLAYING); 1575 $ns->state (ST_PLAYING) if $ns->state == ST_CUSTOM;
892 } 1576 }
893 } 1577 }
894 }, 1578 },
895); 1579);
896 1580
908 my $coro; $coro = async { 1592 my $coro; $coro = async {
909 eval { 1593 eval {
910 $cb->(); 1594 $cb->();
911 }; 1595 };
912 warn $@ if $@; 1596 warn $@ if $@;
1597 };
1598
1599 $coro->on_destroy (sub {
913 delete $self->{_coro}{$coro+0}; 1600 delete $self->{_coro}{$coro+0};
914 }; 1601 });
915 1602
916 $self->{_coro}{$coro+0} = $coro; 1603 $self->{_coro}{$coro+0} = $coro;
1604
1605 $coro
917} 1606}
918 1607
919cf::client->attach ( 1608cf::client->attach (
920 on_destroy => sub { 1609 on_destroy => sub {
921 my ($ns) = @_; 1610 my ($ns) = @_;
1153 local $/; 1842 local $/;
1154 *CFG = YAML::Syck::Load <$fh>; 1843 *CFG = YAML::Syck::Load <$fh>;
1155} 1844}
1156 1845
1157sub main { 1846sub main {
1847 # we must not ever block the main coroutine
1848 local $Coro::idle = sub {
1849 Carp::cluck "FATAL: Coro::idle was called, major BUG\n";#d#
1850 (Coro::unblock_sub {
1851 Event::one_event;
1852 })->();
1853 };
1854
1158 cfg_load; 1855 cfg_load;
1159 db_load; 1856 db_load;
1160 load_extensions; 1857 load_extensions;
1161 Event::loop; 1858 Event::loop;
1162} 1859}
1163 1860
1164############################################################################# 1861#############################################################################
1165# initialisation 1862# initialisation
1166 1863
1167sub _perl_reload(&) { 1864sub reload() {
1168 my ($msg) = @_; 1865 # can/must only be called in main
1866 if ($Coro::current != $Coro::main) {
1867 warn "can only reload from main coroutine\n";
1868 return;
1869 }
1169 1870
1170 $msg->("reloading..."); 1871 warn "reloading...";
1872
1873 local $FREEZE = 1;
1874 cf::emergency_save;
1171 1875
1172 eval { 1876 eval {
1877 # if anything goes wrong in here, we should simply crash as we already saved
1878
1173 # cancel all watchers 1879 # cancel all watchers
1174 for (Event::all_watchers) { 1880 for (Event::all_watchers) {
1175 $_->cancel if $_->data & WF_AUTOCANCEL; 1881 $_->cancel if $_->data & WF_AUTOCANCEL;
1176 } 1882 }
1177 1883
1884 # cancel all extension coros
1885 $_->cancel for values %EXT_CORO;
1886 %EXT_CORO = ();
1887
1178 # unload all extensions 1888 # unload all extensions
1179 for (@exts) { 1889 for (@exts) {
1180 $msg->("unloading <$_>"); 1890 warn "unloading <$_>";
1181 unload_extension $_; 1891 unload_extension $_;
1182 } 1892 }
1183 1893
1184 # unload all modules loaded from $LIBDIR 1894 # unload all modules loaded from $LIBDIR
1185 while (my ($k, $v) = each %INC) { 1895 while (my ($k, $v) = each %INC) {
1186 next unless $v =~ /^\Q$LIBDIR\E\/.*\.pm$/; 1896 next unless $v =~ /^\Q$LIBDIR\E\/.*\.pm$/;
1187 1897
1188 $msg->("removing <$k>"); 1898 warn "removing <$k>";
1189 delete $INC{$k}; 1899 delete $INC{$k};
1190 1900
1191 $k =~ s/\.pm$//; 1901 $k =~ s/\.pm$//;
1192 $k =~ s/\//::/g; 1902 $k =~ s/\//::/g;
1193 1903
1198 Symbol::delete_package $k; 1908 Symbol::delete_package $k;
1199 } 1909 }
1200 1910
1201 # sync database to disk 1911 # sync database to disk
1202 cf::db_sync; 1912 cf::db_sync;
1913 IO::AIO::flush;
1203 1914
1204 # get rid of safe::, as good as possible 1915 # get rid of safe::, as good as possible
1205 Symbol::delete_package "safe::$_" 1916 Symbol::delete_package "safe::$_"
1206 for qw(cf::object cf::object::player cf::player cf::map cf::party cf::region); 1917 for qw(cf::attachable cf::object cf::object::player cf::client cf::player cf::map cf::party cf::region);
1207 1918
1208 # remove register_script_function callbacks 1919 # remove register_script_function callbacks
1209 # TODO 1920 # TODO
1210 1921
1211 # unload cf.pm "a bit" 1922 # unload cf.pm "a bit"
1214 # don't, removes xs symbols, too, 1925 # don't, removes xs symbols, too,
1215 # and global variables created in xs 1926 # and global variables created in xs
1216 #Symbol::delete_package __PACKAGE__; 1927 #Symbol::delete_package __PACKAGE__;
1217 1928
1218 # reload cf.pm 1929 # reload cf.pm
1219 $msg->("reloading cf.pm"); 1930 warn "reloading cf.pm";
1220 require cf; 1931 require cf;
1932 cf::_connect_to_perl; # nominally unnecessary, but cannot hurt
1221 1933
1222 # load config and database again 1934 # load config and database again
1223 cf::cfg_load; 1935 cf::cfg_load;
1224 cf::db_load; 1936 cf::db_load;
1225 1937
1226 # load extensions 1938 # load extensions
1227 $msg->("load extensions"); 1939 warn "load extensions";
1228 cf::load_extensions; 1940 cf::load_extensions;
1229 1941
1230 # reattach attachments to objects 1942 # reattach attachments to objects
1231 $msg->("reattach"); 1943 warn "reattach";
1232 _global_reattach; 1944 _global_reattach;
1233 }; 1945 };
1234 $msg->($@) if $@;
1235 1946
1236 $msg->("reloaded"); 1947 if ($@) {
1948 warn $@;
1949 warn "error while reloading, exiting.";
1950 exit 1;
1951 }
1952
1953 warn "reloaded successfully";
1237}; 1954};
1238 1955
1239sub perl_reload() { 1956#############################################################################
1240 _perl_reload { 1957
1241 warn $_[0]; 1958unless ($LINK_MAP) {
1242 print "$_[0]\n"; 1959 $LINK_MAP = cf::map::new;
1243 }; 1960
1961 $LINK_MAP->width (41);
1962 $LINK_MAP->height (41);
1963 $LINK_MAP->alloc;
1964 $LINK_MAP->path ("{link}");
1965 $LINK_MAP->{path} = bless { path => "{link}" }, "cf::path";
1966 $LINK_MAP->in_memory (MAP_IN_MEMORY);
1967
1968 # dirty hack because... archetypes are not yet loaded
1969 Event->timer (
1970 after => 2,
1971 cb => sub {
1972 $_[0]->w->cancel;
1973
1974 # provide some exits "home"
1975 my $exit = cf::object::new "exit";
1976
1977 $exit->slaying ($EMERGENCY_POSITION->[0]);
1978 $exit->stats->hp ($EMERGENCY_POSITION->[1]);
1979 $exit->stats->sp ($EMERGENCY_POSITION->[2]);
1980
1981 $LINK_MAP->insert ($exit->clone, 19, 19);
1982 $LINK_MAP->insert ($exit->clone, 19, 20);
1983 $LINK_MAP->insert ($exit->clone, 19, 21);
1984 $LINK_MAP->insert ($exit->clone, 20, 19);
1985 $LINK_MAP->insert ($exit->clone, 20, 21);
1986 $LINK_MAP->insert ($exit->clone, 21, 19);
1987 $LINK_MAP->insert ($exit->clone, 21, 20);
1988 $LINK_MAP->insert ($exit->clone, 21, 21);
1989
1990 $exit->destroy;
1991 });
1992
1993 $LINK_MAP->{deny_save} = 1;
1994 $LINK_MAP->{deny_reset} = 1;
1995
1996 $cf::MAP{$LINK_MAP->path} = $LINK_MAP;
1244} 1997}
1245 1998
1246register "<global>", __PACKAGE__; 1999register "<global>", __PACKAGE__;
1247 2000
1248register_command "perl-reload" => sub { 2001register_command "reload" => sub {
1249 my ($who, $arg) = @_; 2002 my ($who, $arg) = @_;
1250 2003
1251 if ($who->flag (FLAG_WIZ)) { 2004 if ($who->flag (FLAG_WIZ)) {
1252 _perl_reload { 2005 $who->message ("start of reload.");
1253 warn $_[0]; 2006 reload;
1254 $who->message ($_[0]); 2007 $who->message ("end of reload.");
1255 };
1256 } 2008 }
1257}; 2009};
1258 2010
1259unshift @INC, $LIBDIR; 2011unshift @INC, $LIBDIR;
1260 2012
1261$TICK_WATCHER = Event->timer ( 2013$TICK_WATCHER = Event->timer (
2014 reentrant => 0,
1262 prio => 0, 2015 prio => 0,
1263 at => $NEXT_TICK || 1, 2016 at => $NEXT_TICK || $TICK,
1264 data => WF_AUTOCANCEL, 2017 data => WF_AUTOCANCEL,
1265 cb => sub { 2018 cb => sub {
2019 unless ($FREEZE) {
1266 cf::server_tick; # one server iteration 2020 cf::server_tick; # one server iteration
2021 $RUNTIME += $TICK;
2022 }
1267 2023
1268 my $NOW = Event::time;
1269 $NEXT_TICK += $TICK; 2024 $NEXT_TICK += $TICK;
1270 2025
1271 # if we are delayed by four ticks or more, skip them all 2026 # if we are delayed by four ticks or more, skip them all
1272 $NEXT_TICK = $NOW if $NOW >= $NEXT_TICK + $TICK * 4; 2027 $NEXT_TICK = Event::time if Event::time >= $NEXT_TICK + $TICK * 4;
1273 2028
1274 $TICK_WATCHER->at ($NEXT_TICK); 2029 $TICK_WATCHER->at ($NEXT_TICK);
1275 $TICK_WATCHER->start; 2030 $TICK_WATCHER->start;
1276 }, 2031 },
1277); 2032);
1278 2033
1279IO::AIO::max_poll_time $TICK * 0.2; 2034IO::AIO::max_poll_time $TICK * 0.2;
1280 2035
2036Event->io (
1281Event->io (fd => IO::AIO::poll_fileno, 2037 fd => IO::AIO::poll_fileno,
1282 poll => 'r', 2038 poll => 'r',
1283 prio => 5, 2039 prio => 5,
1284 data => WF_AUTOCANCEL, 2040 data => WF_AUTOCANCEL,
1285 cb => \&IO::AIO::poll_cb); 2041 cb => \&IO::AIO::poll_cb,
2042);
2043
2044Event->timer (
2045 data => WF_AUTOCANCEL,
2046 after => 0,
2047 interval => 10,
2048 cb => sub {
2049 (Coro::unblock_sub {
2050 write_runtime
2051 or warn "ERROR: unable to write runtime file: $!";
2052 })->();
2053 },
2054);
1286 2055
12871 20561
1288 2057

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines