ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/server/common/time.C
(Generate patch)

Comparing deliantra/server/common/time.C (file contents):
Revision 1.1 by elmex, Sun Aug 13 17:16:01 2006 UTC vs.
Revision 1.23 by root, Fri Mar 26 00:59:20 2010 UTC

1/* 1/*
2 * static char *rcsid_time_c = 2 * This file is part of Deliantra, the Roguelike Realtime MMORPG.
3 * "$Id: time.C,v 1.1 2006/08/13 17:16:01 elmex Exp $"; 3 *
4 * Copyright (©) 2005,2006,2007,2008 Marc Alexander Lehmann / Robin Redeker / the Deliantra team
5 * Copyright (©) 2002 Mark Wedel & Crossfire Development Team
6 * Copyright (©) 1992 Frank Tore Johansen
7 *
8 * Deliantra is free software: you can redistribute it and/or modify it under
9 * the terms of the Affero GNU General Public License as published by the
10 * Free Software Foundation, either version 3 of the License, or (at your
11 * option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the Affero GNU General Public License
19 * and the GNU General Public License along with this program. If not, see
20 * <http://www.gnu.org/licenses/>.
21 *
22 * The authors can be reached via e-mail to <support@deliantra.net>
4 */ 23 */
5 24
6/*
7 CrossFire, A Multiplayer game for X-windows
8
9 Copyright (C) 2002 Mark Wedel & Crossfire Development Team
10 Copyright (C) 1992 Frank Tore Johansen
11
12 This program is free software; you can redistribute it and/or modify
13 it under the terms of the GNU General Public License as published by
14 the Free Software Foundation; either version 2 of the License, or
15 (at your option) any later version.
16
17 This program is distributed in the hope that it will be useful,
18 but WITHOUT ANY WARRANTY; without even the implied warranty of
19 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 GNU General Public License for more details.
21
22 You should have received a copy of the GNU General Public License
23 along with this program; if not, write to the Free Software
24 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
25
26 The authors can be reached via e-mail at crossfire-devel@real-time.com
27*/
28
29#include <global.h> 25#include <global.h>
30#include <funcpoint.h>
31#include <tod.h> 26#include <tod.h>
32 27
33#ifndef WIN32 /* ---win32 exclude header */
34#include <stdio.h> 28#include <cstdio>
29
35#include <sys/types.h> 30#include <sys/types.h>
36#include <sys/time.h> 31#include <sys/time.h>
37#endif /* win32 */
38 32
39/* 33/*
40 * Gloabal variables: 34 * Gloabal variables:
41 */ 35 */
42long max_time = MAX_TIME; 36tick_t pticks;
43struct timeval last_time;
44 37
45#define PBUFLEN 100
46long process_utime_save[PBUFLEN];
47long psaveind;
48long process_max_utime = 0;
49long process_min_utime = 999999999;
50long process_tot_mtime;
51long pticks;
52long process_utime_long_count;
53
54const char *season_name[] = 38static const char *season_name[5] = {
55{ 39 "New Year",
56 "The Season of New Year", 40 "Growth",
57 "The Season of Growth", 41 "Harvest",
58 "The Season of Harvest", 42 "Decay",
59 "The Season of Decay", 43 "the Blizzard",
60 "The Season of the Blizzard",
61 "\n"
62}; 44};
63 45
64const char *weekdays[DAYS_PER_WEEK] = { 46static const char *weekdays[DAYS_PER_WEEK] = {
65 "the Day of the Moon", 47 "the Moon",
66 "the Day of the Bull", 48 "the Bull",
67 "the Day of the Deception", 49 "the Deception",
68 "the Day of Thunder", 50 "Thunder",
69 "the Day of Freedom", 51 "Freedom",
70 "the Day of the Great Gods", 52 "the Great Gods",
71 "the Day of the Sun" 53 "the Sun"
72}; 54};
73 55
74const char *month_name[MONTHS_PER_YEAR] = { 56static const char *month_name[MONTHS_PER_YEAR] = {
75 "Month of Winter", /* 0 */ 57 "Winter", /* 0 */
76 "Month of the Ice Dragon", 58 "the Ice Dragon",
77 "Month of the Frost Giant", 59 "the Frost Giant",
78 "Month of Valriel", 60 "Valriel",
79 "Month of Lythander", 61 "Lythander",
80 "Month of the Harvest", 62 "the Harvest",
81 "Month of Gaea", 63 "Gaea",
82 "Month of Futility", 64 "Futility",
83 "Month of the Dragon", 65 "the Dragon",
84 "Month of the Sun", 66 "the Sun",
85 "Month of the Great Infernus", 67 "the Great Infernus",
86 "Month of Ruggilli", 68 "Ruggilli",
87 "Month of the Dark Shades", 69 "the Dark Shades",
88 "Month of the Devourers", 70 "the Devourers",
89 "Month of Sorig", 71 "Sorig",
90 "Month of the Ancient Darkness", 72 "the Ancient Darkness",
91 "Month of Gorokh" 73 "Gorokh"
92}; 74};
93 75
94/* 76void
95 * Initialise all variables used in the timing routines. 77get_tod (timeofday_t *tod)
96 */ 78{
79 unsigned int todtick = (unsigned int)(runtime * (1. / RUNTIME_PER_HOUR));
97 80
98void 81 tod->year = todtick / HOURS_PER_YEAR + EPOCH;
99reset_sleep(void) 82 tod->month = todtick / HOURS_PER_MONTH % MONTHS_PER_YEAR;
100{ 83 tod->day = todtick % HOURS_PER_MONTH / DAYS_PER_MONTH;
101 int i; 84 tod->hour = todtick % HOURS_PER_DAY;
102 for(i = 0; i < PBUFLEN; i++) 85 tod->minute = (int)((runtime - todtick * RUNTIME_PER_HOUR) * (60. / RUNTIME_PER_HOUR));
103 process_utime_save[i] = 0;
104 psaveind = 0;
105 process_max_utime = 0;
106 process_min_utime = 999999999;
107 process_tot_mtime = 0;
108 pticks = 0;
109 86
110 (void) GETTIMEOFDAY(&last_time); 87 tod->dayofweek = tod->day % DAYS_PER_WEEK;
88 tod->weekofmonth = tod->day / WEEKS_PER_MONTH;
89 tod->season = min (4, tod->month / 4);
111} 90}
112 91
113void 92char *
114log_time(long process_utime) 93format_tod (char *buf, int len, timeofday_t *tod)
115{ 94{
116 pticks++; 95 int day = tod->day + 1;
117 if (++psaveind >= PBUFLEN)
118 psaveind = 0;
119 process_utime_save[psaveind] = process_utime;
120 if (process_utime > process_max_utime)
121 process_max_utime = process_utime;
122 if (process_utime < process_min_utime)
123 process_min_utime = process_utime;
124 process_tot_mtime += process_utime/1000;
125}
126 96
127/* 97 const char *suf;
128 * enough_elapsed_time will return true if the time passed since
129 * last tick is more than max-time.
130 */
131 98
132int
133enough_elapsed_time(void)
134{
135 static struct timeval new_time;
136 long elapsed_utime;
137
138 (void) GETTIMEOFDAY(&new_time);
139
140 elapsed_utime = (new_time.tv_sec - last_time.tv_sec) * 1000000 +
141 new_time.tv_usec - last_time.tv_usec;
142 if (elapsed_utime > max_time) {
143 log_time(elapsed_utime);
144 last_time.tv_sec = new_time.tv_sec;
145 last_time.tv_usec = new_time.tv_usec;
146 return 1;
147 }
148 return 0;
149}
150
151/*
152 * sleep_delta checks how much time has elapsed since last tick.
153 * If it is less than max_time, the remaining time is slept with select().
154 */
155
156void
157sleep_delta(void)
158{
159 static struct timeval new_time;
160 long sleep_sec, sleep_usec;
161
162 (void) GETTIMEOFDAY(&new_time);
163
164 sleep_sec = last_time.tv_sec - new_time.tv_sec;
165 sleep_usec = max_time - (new_time.tv_usec - last_time.tv_usec);
166
167 /* This is very ugly, but probably the fastest for our use: */
168 while (sleep_usec < 0) {
169 sleep_usec += 1000000;
170 sleep_sec -= 1;
171 }
172 while (sleep_usec > 1000000) {
173 sleep_usec -= 1000000;
174 sleep_sec +=1;
175 }
176
177 log_time((new_time.tv_sec - last_time.tv_sec)*1000000
178 + new_time.tv_usec - last_time.tv_usec);
179
180 if (sleep_sec >= 0 && sleep_usec > 0) {
181 static struct timeval sleep_time;
182 sleep_time.tv_sec = sleep_sec;
183 sleep_time.tv_usec = sleep_usec;
184
185#ifndef WIN32 /* 'select' doesn't work on Windows, 'Sleep' is used instead */
186 select(0, NULL, NULL, NULL, &sleep_time);
187#else
188 if (sleep_time.tv_sec) Sleep(sleep_time.tv_sec*1000);
189 Sleep((int)(sleep_time.tv_usec/1000.));
190#endif
191 }
192 else
193 process_utime_long_count++;
194 /*
195 * Set last_time to when we're expected to wake up:
196 */
197 last_time.tv_usec += max_time;
198 while (last_time.tv_usec > 1000000) {
199 last_time.tv_usec -= 1000000;
200 last_time.tv_sec++;
201 }
202 /*
203 * Don't do too much catching up:
204 * (Things can still get jerky on a slow/loaded computer)
205 */
206 if (last_time.tv_sec * 1000000 + last_time.tv_usec <
207 new_time.tv_sec * 1000000 + new_time.tv_usec)
208 {
209 last_time.tv_sec = new_time.tv_sec;
210 last_time.tv_usec = new_time.tv_usec;
211 }
212}
213
214void
215set_max_time(long t) {
216 max_time = t;
217}
218
219extern unsigned long todtick;
220
221void
222get_tod(timeofday_t *tod)
223{
224 tod->year = todtick/HOURS_PER_YEAR;
225 tod->month = (todtick/HOURS_PER_MONTH)%MONTHS_PER_YEAR;
226 tod->day = (todtick%HOURS_PER_MONTH)/DAYS_PER_MONTH;
227 tod->dayofweek = tod->day%DAYS_PER_WEEK;
228 tod->hour = todtick%HOURS_PER_DAY;
229 tod->minute = (pticks%PTICKS_PER_CLOCK)/(PTICKS_PER_CLOCK/58);
230 if (tod->minute > 58)
231 tod->minute = 58; /* it's imprecise at best anyhow */
232 tod->weekofmonth = tod->day/WEEKS_PER_MONTH;
233 if (tod->month < 3)
234 tod->season = 0;
235 else if (tod->month < 6)
236 tod->season = 1;
237 else if (tod->month < 9)
238 tod->season = 2;
239 else if (tod->month < 12)
240 tod->season = 3;
241 else
242 tod->season = 4;
243}
244
245void
246print_tod(object *op)
247{
248 timeofday_t tod;
249 const char *suf;
250 int day;
251
252 get_tod(&tod);
253 sprintf(errmsg, "It is %d minute%s past %d o'clock %s, on %s",
254 tod.minute+1, ((tod.minute+1 < 2) ? "" : "s"),
255 ((tod.hour % 14 == 0) ? 14 : ((tod.hour)%14)),
256 ((tod.hour >= 14) ? "pm" : "am"),
257 weekdays[tod.dayofweek]);
258 new_draw_info(NDI_UNIQUE, 0,op,errmsg);
259
260 day = tod.day + 1;
261 if (day == 1 || ((day % 10) == 1 && day > 20)) 99 if (day == 1 || ((day % 10) == 1 && day > 20))
262 suf = "st"; 100 suf = "st";
263 else if (day == 2 || ((day % 10) == 2 && day > 20)) 101 else if (day == 2 || ((day % 10) == 2 && day > 20))
264 suf = "nd"; 102 suf = "nd";
265 else if (day == 3 || ((day % 10) == 3 && day > 20)) 103 else if (day == 3 || ((day % 10) == 3 && day > 20))
266 suf = "rd"; 104 suf = "rd";
267 else 105 else
268 suf = "th"; 106 suf = "th";
269 sprintf(errmsg, "The %d%s Day of the %s, Year %d", day, suf,
270 month_name[tod.month], tod.year+1);
271 new_draw_info(NDI_UNIQUE, 0,op,errmsg);
272 107
273 sprintf(errmsg, "Time of Year: %s", season_name[tod.season]); 108 snprintf (buf, len,
274 new_draw_info(NDI_UNIQUE, 0,op,errmsg); 109 "It is %d minute%s past %d o'clock %s, on the Day of %s,\n"
110 "the %d%s Day of the Month of %s, Year %d.\n"
111 "It is the Season of %s.",
112
113 tod->minute ? tod->minute : 60,
114 tod->minute ? "s" : "",
115 (tod->hour + 13) % 14 + 1,
116 (tod->hour >= 14 ? "pm" : "am"),
117 weekdays[tod->dayofweek],
118
119 day, suf, month_name[tod->month], tod->year + 1,
120
121 season_name[tod->season]);
122
123 return buf;
124}
125
126static const char *
127get_current_tod_str ()
128{
129 timeofday_t tod;
130 get_tod (&tod);
131
132 static char todbuf[512];
133 format_tod (todbuf, sizeof (todbuf), &tod);
134
135 return todbuf;
275} 136}
276 137
277void 138void
278time_info(object *op) 139print_tod (object *op)
279{ 140{
280 int tot = 0, maxt = 0, mint = 99999999, long_count = 0, i;
281
282 print_tod(op);
283 if (!QUERY_FLAG(op,FLAG_WIZ))
284 return;
285
286 new_draw_info (NDI_UNIQUE, 0,op,"Total time:");
287 sprintf(errmsg,"ticks=%ld time=%ld.%2ld",
288 pticks, process_tot_mtime/1000, process_tot_mtime%1000);
289 new_draw_info (NDI_UNIQUE, 0,op,errmsg); 141 new_draw_info (NDI_UNIQUE, 0, op, get_current_tod_str ());
290 sprintf(errmsg,"avg time=%ldms max time=%ldms min time=%ldms",
291 process_tot_mtime/pticks, process_max_utime/1000,
292 process_min_utime/1000);
293 new_draw_info (NDI_UNIQUE, 0,op,errmsg);
294 sprintf(errmsg,"ticks longer than max time (%ldms) = %ld (%ld%%)",
295 max_time/1000,
296 process_utime_long_count, 100*process_utime_long_count/pticks);
297 new_draw_info (NDI_UNIQUE, 0,op,errmsg);
298
299 sprintf(errmsg,"Time last %ld ticks:", pticks > PBUFLEN ? PBUFLEN : pticks);
300 new_draw_info (NDI_UNIQUE, 0,op,errmsg);
301
302 for (i = 0; i < (pticks > PBUFLEN ? PBUFLEN : pticks); i++) {
303 tot += process_utime_save[i];
304 if (process_utime_save[i] > maxt) maxt = process_utime_save[i];
305 if (process_utime_save[i] < mint) mint = process_utime_save[i];
306 if (process_utime_save[i] > max_time) long_count++;
307 }
308
309 sprintf(errmsg,"avg time=%ldms max time=%dms min time=%dms",
310 tot/(pticks > PBUFLEN ? PBUFLEN : pticks)/1000, maxt/1000,
311 mint/1000);
312 new_draw_info (NDI_UNIQUE, 0,op,errmsg);
313 sprintf(errmsg,"ticks longer than max time (%ldms) = %d (%ld%%)",
314 max_time/1000, long_count,
315 100*long_count/(pticks > PBUFLEN ? PBUFLEN : pticks));
316 new_draw_info (NDI_UNIQUE, 0,op,errmsg);
317} 142}
318 143
319long
320seconds(void)
321{
322 struct timeval now;
323
324 (void) GETTIMEOFDAY(&now);
325 return now.tv_sec;
326}

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines