hack.main.c revision 1.2 1 1.2 mycroft /*
2 1.2 mycroft * Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985.
3 1.2 mycroft */
4 1.2 mycroft
5 1.2 mycroft #ifndef lint
6 1.2 mycroft static char rcsid[] = "$Id: hack.main.c,v 1.2 1993/08/02 17:17:18 mycroft Exp $";
7 1.2 mycroft #endif /* not lint */
8 1.1 cgd
9 1.1 cgd #include <stdio.h>
10 1.1 cgd #include <signal.h>
11 1.1 cgd #include "hack.h"
12 1.1 cgd
13 1.1 cgd #ifdef QUEST
14 1.1 cgd #define gamename "quest"
15 1.1 cgd #else
16 1.1 cgd #define gamename "hack"
17 1.1 cgd #endif
18 1.1 cgd
19 1.1 cgd extern char *getlogin(), *getenv();
20 1.1 cgd extern char plname[PL_NSIZ], pl_character[PL_CSIZ];
21 1.1 cgd extern struct permonst mons[CMNUM+2];
22 1.1 cgd extern char genocided[], fut_geno[];
23 1.1 cgd
24 1.1 cgd int (*afternmv)();
25 1.1 cgd int (*occupation)();
26 1.1 cgd char *occtxt; /* defined when occupation != NULL */
27 1.1 cgd
28 1.1 cgd void done1();
29 1.1 cgd void hangup();
30 1.1 cgd
31 1.1 cgd int hackpid; /* current pid */
32 1.1 cgd int locknum; /* max num of players */
33 1.1 cgd #ifdef DEF_PAGER
34 1.1 cgd char *catmore; /* default pager */
35 1.1 cgd #endif
36 1.1 cgd char SAVEF[PL_NSIZ + 11] = "save/"; /* save/99999player */
37 1.1 cgd char *hname; /* name of the game (argv[0] of call) */
38 1.1 cgd char obuf[BUFSIZ]; /* BUFSIZ is defined in stdio.h */
39 1.1 cgd
40 1.1 cgd extern char *nomovemsg;
41 1.1 cgd extern long wailmsg;
42 1.1 cgd
43 1.1 cgd #ifdef CHDIR
44 1.1 cgd static void chdirx();
45 1.1 cgd #endif
46 1.1 cgd
47 1.1 cgd main(argc,argv)
48 1.1 cgd int argc;
49 1.1 cgd char *argv[];
50 1.1 cgd {
51 1.1 cgd register int fd;
52 1.1 cgd #ifdef CHDIR
53 1.1 cgd register char *dir;
54 1.1 cgd #endif
55 1.1 cgd
56 1.1 cgd hname = argv[0];
57 1.1 cgd hackpid = getpid();
58 1.1 cgd
59 1.1 cgd #ifdef CHDIR /* otherwise no chdir() */
60 1.1 cgd /*
61 1.1 cgd * See if we must change directory to the playground.
62 1.1 cgd * (Perhaps hack runs suid and playground is inaccessible
63 1.1 cgd * for the player.)
64 1.1 cgd * The environment variable HACKDIR is overridden by a
65 1.1 cgd * -d command line option (must be the first option given)
66 1.1 cgd */
67 1.1 cgd
68 1.1 cgd dir = getenv("HACKDIR");
69 1.1 cgd if(argc > 1 && !strncmp(argv[1], "-d", 2)) {
70 1.1 cgd argc--;
71 1.1 cgd argv++;
72 1.1 cgd dir = argv[0]+2;
73 1.1 cgd if(*dir == '=' || *dir == ':') dir++;
74 1.1 cgd if(!*dir && argc > 1) {
75 1.1 cgd argc--;
76 1.1 cgd argv++;
77 1.1 cgd dir = argv[0];
78 1.1 cgd }
79 1.1 cgd if(!*dir)
80 1.1 cgd error("Flag -d must be followed by a directory name.");
81 1.1 cgd }
82 1.1 cgd #endif
83 1.1 cgd
84 1.1 cgd /*
85 1.1 cgd * Who am i? Algorithm: 1. Use name as specified in HACKOPTIONS
86 1.1 cgd * 2. Use $USER or $LOGNAME (if 1. fails)
87 1.1 cgd * 3. Use getlogin() (if 2. fails)
88 1.1 cgd * The resulting name is overridden by command line options.
89 1.1 cgd * If everything fails, or if the resulting name is some generic
90 1.1 cgd * account like "games", "play", "player", "hack" then eventually
91 1.1 cgd * we'll ask him.
92 1.1 cgd * Note that we trust him here; it is possible to play under
93 1.1 cgd * somebody else's name.
94 1.1 cgd */
95 1.1 cgd { register char *s;
96 1.1 cgd
97 1.1 cgd initoptions();
98 1.1 cgd if(!*plname && (s = getenv("USER")))
99 1.1 cgd (void) strncpy(plname, s, sizeof(plname)-1);
100 1.1 cgd if(!*plname && (s = getenv("LOGNAME")))
101 1.1 cgd (void) strncpy(plname, s, sizeof(plname)-1);
102 1.1 cgd if(!*plname && (s = getlogin()))
103 1.1 cgd (void) strncpy(plname, s, sizeof(plname)-1);
104 1.1 cgd }
105 1.1 cgd
106 1.1 cgd /*
107 1.1 cgd * Now we know the directory containing 'record' and
108 1.1 cgd * may do a prscore().
109 1.1 cgd */
110 1.1 cgd if(argc > 1 && !strncmp(argv[1], "-s", 2)) {
111 1.1 cgd #ifdef CHDIR
112 1.1 cgd chdirx(dir,0);
113 1.1 cgd #endif
114 1.1 cgd prscore(argc, argv);
115 1.1 cgd exit(0);
116 1.1 cgd }
117 1.1 cgd
118 1.1 cgd /*
119 1.1 cgd * It seems he really wants to play.
120 1.1 cgd * Remember tty modes, to be restored on exit.
121 1.1 cgd */
122 1.1 cgd gettty();
123 1.1 cgd setbuf(stdout,obuf);
124 1.1 cgd setrandom();
125 1.1 cgd startup();
126 1.1 cgd cls();
127 1.1 cgd u.uhp = 1; /* prevent RIP on early quits */
128 1.1 cgd u.ux = FAR; /* prevent nscr() */
129 1.1 cgd (void) signal(SIGHUP, hangup);
130 1.1 cgd
131 1.1 cgd /*
132 1.1 cgd * Find the creation date of this game,
133 1.1 cgd * so as to avoid restoring outdated savefiles.
134 1.1 cgd */
135 1.1 cgd gethdate(hname);
136 1.1 cgd
137 1.1 cgd /*
138 1.1 cgd * We cannot do chdir earlier, otherwise gethdate will fail.
139 1.1 cgd */
140 1.1 cgd #ifdef CHDIR
141 1.1 cgd chdirx(dir,1);
142 1.1 cgd #endif
143 1.1 cgd
144 1.1 cgd /*
145 1.1 cgd * Process options.
146 1.1 cgd */
147 1.1 cgd while(argc > 1 && argv[1][0] == '-'){
148 1.1 cgd argv++;
149 1.1 cgd argc--;
150 1.1 cgd switch(argv[0][1]){
151 1.1 cgd #ifdef WIZARD
152 1.1 cgd case 'D':
153 1.1 cgd /* if(!strcmp(getlogin(), WIZARD)) */
154 1.1 cgd wizard = TRUE;
155 1.1 cgd /* else
156 1.1 cgd printf("Sorry.\n"); */
157 1.1 cgd break;
158 1.1 cgd #endif
159 1.1 cgd #ifdef NEWS
160 1.1 cgd case 'n':
161 1.1 cgd flags.nonews = TRUE;
162 1.1 cgd break;
163 1.1 cgd #endif
164 1.1 cgd case 'u':
165 1.1 cgd if(argv[0][2])
166 1.1 cgd (void) strncpy(plname, argv[0]+2, sizeof(plname)-1);
167 1.1 cgd else if(argc > 1) {
168 1.1 cgd argc--;
169 1.1 cgd argv++;
170 1.1 cgd (void) strncpy(plname, argv[0], sizeof(plname)-1);
171 1.1 cgd } else
172 1.1 cgd printf("Player name expected after -u\n");
173 1.1 cgd break;
174 1.1 cgd default:
175 1.1 cgd /* allow -T for Tourist, etc. */
176 1.1 cgd (void) strncpy(pl_character, argv[0]+1,
177 1.1 cgd sizeof(pl_character)-1);
178 1.1 cgd
179 1.1 cgd /* printf("Unknown option: %s\n", *argv); */
180 1.1 cgd }
181 1.1 cgd }
182 1.1 cgd
183 1.1 cgd if(argc > 1)
184 1.1 cgd locknum = atoi(argv[1]);
185 1.1 cgd #ifdef MAX_NR_OF_PLAYERS
186 1.1 cgd if(!locknum || locknum > MAX_NR_OF_PLAYERS)
187 1.1 cgd locknum = MAX_NR_OF_PLAYERS;
188 1.1 cgd #endif
189 1.1 cgd #ifdef DEF_PAGER
190 1.1 cgd if(!(catmore = getenv("HACKPAGER")) && !(catmore = getenv("PAGER")))
191 1.1 cgd catmore = DEF_PAGER;
192 1.1 cgd #endif
193 1.1 cgd #ifdef MAIL
194 1.1 cgd getmailstatus();
195 1.1 cgd #endif
196 1.1 cgd #ifdef WIZARD
197 1.1 cgd if(wizard) (void) strcpy(plname, "wizard"); else
198 1.1 cgd #endif
199 1.1 cgd if(!*plname || !strncmp(plname, "player", 4)
200 1.1 cgd || !strncmp(plname, "games", 4))
201 1.1 cgd askname();
202 1.1 cgd plnamesuffix(); /* strip suffix from name; calls askname() */
203 1.1 cgd /* again if suffix was whole name */
204 1.1 cgd /* accepts any suffix */
205 1.1 cgd #ifdef WIZARD
206 1.1 cgd if(!wizard) {
207 1.1 cgd #endif
208 1.1 cgd /*
209 1.1 cgd * check for multiple games under the same name
210 1.1 cgd * (if !locknum) or check max nr of players (otherwise)
211 1.1 cgd */
212 1.1 cgd (void) signal(SIGQUIT,SIG_IGN);
213 1.1 cgd (void) signal(SIGINT,SIG_IGN);
214 1.1 cgd if(!locknum)
215 1.1 cgd (void) strcpy(lock,plname);
216 1.1 cgd getlock(); /* sets lock if locknum != 0 */
217 1.1 cgd #ifdef WIZARD
218 1.1 cgd } else {
219 1.1 cgd register char *sfoo;
220 1.1 cgd (void) strcpy(lock,plname);
221 1.1 cgd if(sfoo = getenv("MAGIC"))
222 1.1 cgd while(*sfoo) {
223 1.1 cgd switch(*sfoo++) {
224 1.1 cgd case 'n': (void) srandom(*sfoo++);
225 1.1 cgd break;
226 1.1 cgd }
227 1.1 cgd }
228 1.1 cgd if(sfoo = getenv("GENOCIDED")){
229 1.1 cgd if(*sfoo == '!'){
230 1.1 cgd register struct permonst *pm = mons;
231 1.1 cgd register char *gp = genocided;
232 1.1 cgd
233 1.1 cgd while(pm < mons+CMNUM+2){
234 1.1 cgd if(!index(sfoo, pm->mlet))
235 1.1 cgd *gp++ = pm->mlet;
236 1.1 cgd pm++;
237 1.1 cgd }
238 1.1 cgd *gp = 0;
239 1.1 cgd } else
240 1.1 cgd (void) strcpy(genocided, sfoo);
241 1.1 cgd (void) strcpy(fut_geno, genocided);
242 1.1 cgd }
243 1.1 cgd }
244 1.1 cgd #endif
245 1.1 cgd setftty();
246 1.1 cgd (void) sprintf(SAVEF, "save/%d%s", getuid(), plname);
247 1.1 cgd regularize(SAVEF+5); /* avoid . or / in name */
248 1.1 cgd if((fd = open(SAVEF,0)) >= 0 &&
249 1.1 cgd (uptodate(fd) || unlink(SAVEF) == 666)) {
250 1.1 cgd (void) signal(SIGINT,done1);
251 1.1 cgd pline("Restoring old save file...");
252 1.1 cgd (void) fflush(stdout);
253 1.1 cgd if(!dorecover(fd))
254 1.1 cgd goto not_recovered;
255 1.1 cgd pline("Hello %s, welcome to %s!", plname, gamename);
256 1.1 cgd flags.move = 0;
257 1.1 cgd } else {
258 1.1 cgd not_recovered:
259 1.1 cgd fobj = fcobj = invent = 0;
260 1.1 cgd fmon = fallen_down = 0;
261 1.1 cgd ftrap = 0;
262 1.1 cgd fgold = 0;
263 1.1 cgd flags.ident = 1;
264 1.1 cgd init_objects();
265 1.1 cgd u_init();
266 1.1 cgd
267 1.1 cgd (void) signal(SIGINT,done1);
268 1.1 cgd mklev();
269 1.1 cgd u.ux = xupstair;
270 1.1 cgd u.uy = yupstair;
271 1.1 cgd (void) inshop();
272 1.1 cgd setsee();
273 1.1 cgd flags.botlx = 1;
274 1.1 cgd makedog();
275 1.1 cgd { register struct monst *mtmp;
276 1.1 cgd if(mtmp = m_at(u.ux, u.uy)) mnexto(mtmp); /* riv05!a3 */
277 1.1 cgd }
278 1.1 cgd seemons();
279 1.1 cgd #ifdef NEWS
280 1.1 cgd if(flags.nonews || !readnews())
281 1.1 cgd /* after reading news we did docrt() already */
282 1.1 cgd #endif
283 1.1 cgd docrt();
284 1.1 cgd
285 1.1 cgd /* give welcome message before pickup messages */
286 1.1 cgd pline("Hello %s, welcome to %s!", plname, gamename);
287 1.1 cgd
288 1.1 cgd pickup(1);
289 1.1 cgd read_engr_at(u.ux,u.uy);
290 1.1 cgd flags.move = 1;
291 1.1 cgd }
292 1.1 cgd
293 1.1 cgd flags.moonphase = phase_of_the_moon();
294 1.1 cgd if(flags.moonphase == FULL_MOON) {
295 1.1 cgd pline("You are lucky! Full moon tonight.");
296 1.1 cgd u.uluck++;
297 1.1 cgd } else if(flags.moonphase == NEW_MOON) {
298 1.1 cgd pline("Be careful! New moon tonight.");
299 1.1 cgd }
300 1.1 cgd
301 1.1 cgd initrack();
302 1.1 cgd
303 1.1 cgd for(;;) {
304 1.1 cgd if(flags.move) { /* actual time passed */
305 1.1 cgd
306 1.1 cgd settrack();
307 1.1 cgd
308 1.1 cgd if(moves%2 == 0 ||
309 1.1 cgd (!(Fast & ~INTRINSIC) && (!Fast || rn2(3)))) {
310 1.1 cgd extern struct monst *makemon();
311 1.1 cgd movemon();
312 1.1 cgd if(!rn2(70))
313 1.1 cgd (void) makemon((struct permonst *)0, 0, 0);
314 1.1 cgd }
315 1.1 cgd if(Glib) glibr();
316 1.1 cgd timeout();
317 1.1 cgd ++moves;
318 1.1 cgd if(flags.time) flags.botl = 1;
319 1.1 cgd if(u.uhp < 1) {
320 1.1 cgd pline("You die...");
321 1.1 cgd done("died");
322 1.1 cgd }
323 1.1 cgd if(u.uhp*10 < u.uhpmax && moves-wailmsg > 50){
324 1.1 cgd wailmsg = moves;
325 1.1 cgd if(u.uhp == 1)
326 1.1 cgd pline("You hear the wailing of the Banshee...");
327 1.1 cgd else
328 1.1 cgd pline("You hear the howling of the CwnAnnwn...");
329 1.1 cgd }
330 1.1 cgd if(u.uhp < u.uhpmax) {
331 1.1 cgd if(u.ulevel > 9) {
332 1.1 cgd if(Regeneration || !(moves%3)) {
333 1.1 cgd flags.botl = 1;
334 1.1 cgd u.uhp += rnd((int) u.ulevel-9);
335 1.1 cgd if(u.uhp > u.uhpmax)
336 1.1 cgd u.uhp = u.uhpmax;
337 1.1 cgd }
338 1.1 cgd } else if(Regeneration ||
339 1.1 cgd (!(moves%(22-u.ulevel*2)))) {
340 1.1 cgd flags.botl = 1;
341 1.1 cgd u.uhp++;
342 1.1 cgd }
343 1.1 cgd }
344 1.1 cgd if(Teleportation && !rn2(85)) tele();
345 1.1 cgd if(Searching && multi >= 0) (void) dosearch();
346 1.1 cgd gethungry();
347 1.1 cgd invault();
348 1.1 cgd amulet();
349 1.1 cgd }
350 1.1 cgd if(multi < 0) {
351 1.1 cgd if(!++multi){
352 1.1 cgd pline(nomovemsg ? nomovemsg :
353 1.1 cgd "You can move again.");
354 1.1 cgd nomovemsg = 0;
355 1.1 cgd if(afternmv) (*afternmv)();
356 1.1 cgd afternmv = 0;
357 1.1 cgd }
358 1.1 cgd }
359 1.1 cgd
360 1.1 cgd find_ac();
361 1.1 cgd #ifndef QUEST
362 1.1 cgd if(!flags.mv || Blind)
363 1.1 cgd #endif
364 1.1 cgd {
365 1.1 cgd seeobjs();
366 1.1 cgd seemons();
367 1.1 cgd nscr();
368 1.1 cgd }
369 1.1 cgd if(flags.botl || flags.botlx) bot();
370 1.1 cgd
371 1.1 cgd flags.move = 1;
372 1.1 cgd
373 1.1 cgd if(multi >= 0 && occupation) {
374 1.1 cgd if(monster_nearby())
375 1.1 cgd stop_occupation();
376 1.1 cgd else if ((*occupation)() == 0)
377 1.1 cgd occupation = 0;
378 1.1 cgd continue;
379 1.1 cgd }
380 1.1 cgd
381 1.1 cgd if(multi > 0) {
382 1.1 cgd #ifdef QUEST
383 1.1 cgd if(flags.run >= 4) finddir();
384 1.1 cgd #endif
385 1.1 cgd lookaround();
386 1.1 cgd if(!multi) { /* lookaround may clear multi */
387 1.1 cgd flags.move = 0;
388 1.1 cgd continue;
389 1.1 cgd }
390 1.1 cgd if(flags.mv) {
391 1.1 cgd if(multi < COLNO && !--multi)
392 1.1 cgd flags.mv = flags.run = 0;
393 1.1 cgd domove();
394 1.1 cgd } else {
395 1.1 cgd --multi;
396 1.1 cgd rhack(save_cm);
397 1.1 cgd }
398 1.1 cgd } else if(multi == 0) {
399 1.1 cgd #ifdef MAIL
400 1.1 cgd ckmailstatus();
401 1.1 cgd #endif
402 1.1 cgd rhack((char *) 0);
403 1.1 cgd }
404 1.1 cgd if(multi && multi%7 == 0)
405 1.1 cgd (void) fflush(stdout);
406 1.1 cgd }
407 1.1 cgd }
408 1.1 cgd
409 1.1 cgd glo(foo)
410 1.1 cgd register foo;
411 1.1 cgd {
412 1.1 cgd /* construct the string xlock.n */
413 1.1 cgd register char *tf;
414 1.1 cgd
415 1.1 cgd tf = lock;
416 1.1 cgd while(*tf && *tf != '.') tf++;
417 1.1 cgd (void) sprintf(tf, ".%d", foo);
418 1.1 cgd }
419 1.1 cgd
420 1.1 cgd /*
421 1.1 cgd * plname is filled either by an option (-u Player or -uPlayer) or
422 1.1 cgd * explicitly (-w implies wizard) or by askname.
423 1.1 cgd * It may still contain a suffix denoting pl_character.
424 1.1 cgd */
425 1.1 cgd askname(){
426 1.1 cgd register int c,ct;
427 1.1 cgd printf("\nWho are you? ");
428 1.1 cgd (void) fflush(stdout);
429 1.1 cgd ct = 0;
430 1.1 cgd while((c = getchar()) != '\n'){
431 1.1 cgd if(c == EOF) error("End of input\n");
432 1.1 cgd /* some people get confused when their erase char is not ^H */
433 1.1 cgd if(c == '\010') {
434 1.1 cgd if(ct) ct--;
435 1.1 cgd continue;
436 1.1 cgd }
437 1.1 cgd if(c != '-')
438 1.1 cgd if(c < 'A' || (c > 'Z' && c < 'a') || c > 'z') c = '_';
439 1.1 cgd if(ct < sizeof(plname)-1) plname[ct++] = c;
440 1.1 cgd }
441 1.1 cgd plname[ct] = 0;
442 1.1 cgd if(ct == 0) askname();
443 1.1 cgd }
444 1.1 cgd
445 1.1 cgd /*VARARGS1*/
446 1.1 cgd impossible(s,x1,x2)
447 1.1 cgd register char *s;
448 1.1 cgd {
449 1.1 cgd pline(s,x1,x2);
450 1.1 cgd pline("Program in disorder - perhaps you'd better Quit.");
451 1.1 cgd }
452 1.1 cgd
453 1.1 cgd #ifdef CHDIR
454 1.1 cgd static void
455 1.1 cgd chdirx(dir, wr)
456 1.1 cgd char *dir;
457 1.1 cgd boolean wr;
458 1.1 cgd {
459 1.1 cgd
460 1.1 cgd #ifdef SECURE
461 1.1 cgd if(dir /* User specified directory? */
462 1.1 cgd #ifdef HACKDIR
463 1.1 cgd && strcmp(dir, HACKDIR) /* and not the default? */
464 1.1 cgd #endif
465 1.1 cgd ) {
466 1.1 cgd (void) setuid(getuid()); /* Ron Wessels */
467 1.1 cgd (void) setgid(getgid());
468 1.1 cgd }
469 1.1 cgd #endif
470 1.1 cgd
471 1.1 cgd #ifdef HACKDIR
472 1.1 cgd if(dir == NULL)
473 1.1 cgd dir = HACKDIR;
474 1.1 cgd #endif
475 1.1 cgd
476 1.1 cgd if(dir && chdir(dir) < 0) {
477 1.1 cgd perror(dir);
478 1.1 cgd error("Cannot chdir to %s.", dir);
479 1.1 cgd }
480 1.1 cgd
481 1.1 cgd /* warn the player if he cannot write the record file */
482 1.1 cgd /* perhaps we should also test whether . is writable */
483 1.1 cgd /* unfortunately the access systemcall is worthless */
484 1.1 cgd if(wr) {
485 1.1 cgd register fd;
486 1.1 cgd
487 1.1 cgd if(dir == NULL)
488 1.1 cgd dir = ".";
489 1.1 cgd if((fd = open(RECORD, 2)) < 0) {
490 1.1 cgd printf("Warning: cannot write %s/%s", dir, RECORD);
491 1.1 cgd getret();
492 1.1 cgd } else
493 1.1 cgd (void) close(fd);
494 1.1 cgd }
495 1.1 cgd }
496 1.1 cgd #endif
497 1.1 cgd
498 1.1 cgd stop_occupation()
499 1.1 cgd {
500 1.1 cgd if(occupation) {
501 1.1 cgd pline("You stop %s.", occtxt);
502 1.1 cgd occupation = 0;
503 1.1 cgd }
504 1.1 cgd }
505