]> git.pond.sub.org Git - empserver/blob - src/server/update.c
(kw_read, kw_find, kwtab, kw_list): Were used to evaluate the hours
[empserver] / src / server / update.c
1 /*
2  *  Empire - A multi-player, client/server Internet based war game.
3  *  Copyright (C) 1986-2000, Dave Pare, Jeff Bailey, Thomas Ruschak,
4  *                           Ken Stevens, Steve McClure
5  *
6  *  This program is free software; you can redistribute it and/or modify
7  *  it under the terms of the GNU General Public License as published by
8  *  the Free Software Foundation; either version 2 of the License, or
9  *  (at your option) any later version.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program; if not, write to the Free Software
18  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  *
20  *  ---
21  *
22  *  See the "LEGAL", "LICENSE", "CREDITS" and "README" files for all the
23  *  related information and legal notices. It is expected that any future
24  *  projects/authors will amend these files as needed.
25  *
26  *  ---
27  *
28  *  update.c: Update scheduler
29  * 
30  *  Known contributors to this file:
31  *     Dave Pare, 1994
32  *     Steve McClure, 1996
33  */
34
35 #include "misc.h"
36 #include "player.h"
37 #include "keyword.h"
38 #include "empthread.h"
39 #include "prototypes.h"
40 #include "optlist.h"
41 #include "server.h"
42
43 empth_sem_t *update_sem;
44
45 static void update_wait(void *unused);
46 time_t update_time;
47
48 /*ARGSUSED*/
49 void
50 update_sched(void *unused)
51 {
52     s_char *kw;
53     int hour[2];
54     int wind;
55     time_t now, delta;
56
57     update_sem = empth_sem_create("Update", 0);
58     empth_create(PP_SCHED, update_wait, (50 * 1024), 0, "UpdateWait",
59                  "Waits until players idle", 0);
60     time(&now);
61     (void)gamehours(now, hour);
62     if (s_p_etu <= 0) {
63         logerror("bad value for s_p_etu (%d)", s_p_etu);
64         s_p_etu = 2 * 60;
65         logerror("setting s_p_etu to %d", s_p_etu);
66     }
67     while (1) {
68         time(&now);
69         next_update_time(&now, &update_time, &delta);
70         if (update_window > 0) {
71             wind = (random() % update_window);
72             update_time += wind;
73             delta += wind;
74         }
75         logerror("Next update at %s", ctime(&update_time));
76         logerror("Next update in %ld seconds", (long)delta);
77         /* sleep until update is scheduled to go off */
78         empth_sleep(update_time);
79         time(&now);
80         now += adj_update;
81         if (!gamehours(now, hour)) {
82             logerror("No update permitted (hours restriction)");
83             continue;
84         }
85         if (!updatetime(&now)) {
86             logerror("No update wanted");
87             continue;
88         }
89         if (updates_disabled()) {
90             logerror("Updates disabled...skipping update");
91             continue;
92         }
93         empth_sem_signal(update_sem);
94     }
95     /*NOTREACHED*/
96 }
97
98 /*ARGSUSED*/
99 static void
100 update_wait(void *unused)
101 {
102     struct player *p;
103     int running;
104     time_t now;
105     int stacksize;
106     struct player *dp;
107
108     while (1) {
109         empth_sem_wait(update_sem);
110         running = 0;
111         for (p = player_next(0); p != 0; p = player_next(p)) {
112             if (p->state != PS_PLAYING)
113                 continue;
114             if (p->command) {
115                 pr_flash(p, "Update aborting command\n");
116                 p->aborted = 1;
117                 empth_wakeup(p->proc);
118                 running++;
119             }
120         }
121         time(&now);
122         if (running) {
123             /* sleep a few, wait for aborts to take effect */
124             empth_sleep(now + 2);
125         }
126         /* 
127          * we rely on the fact that update's priority is the highest
128          * in the land so it can finish before it yields.
129          */
130         dp = player_new(0, 0);
131         if (!dp) {
132             logerror("can't create dummy player for update");
133             continue;
134         }
135         stacksize = 100000 +
136 /* finish_sects */ WORLD_X * WORLD_Y * (2 * sizeof(double) +
137                                         sizeof(s_char *));
138
139         empth_create(PP_UPDATE, update_main, stacksize, 0,
140                      "UpdateRun", "Updates the world", dp);
141     }
142     /*NOTREACHED*/
143 }