xref: /openbsd-src/games/hack/hack.save.c (revision df69c215c7c66baf660f3f65414fd34796c96152)
1*df69c215Sderaadt /*	$OpenBSD: hack.save.c,v 1.14 2019/06/28 13:32:52 deraadt Exp $	*/
2d0b779f3Sniklas 
3df930be7Sderaadt /*
4d25013f2Scamield  * Copyright (c) 1985, Stichting Centrum voor Wiskunde en Informatica,
5d25013f2Scamield  * Amsterdam
6d25013f2Scamield  * All rights reserved.
7d25013f2Scamield  *
8d25013f2Scamield  * Redistribution and use in source and binary forms, with or without
9d25013f2Scamield  * modification, are permitted provided that the following conditions are
10d25013f2Scamield  * met:
11d25013f2Scamield  *
12d25013f2Scamield  * - Redistributions of source code must retain the above copyright notice,
13d25013f2Scamield  * this list of conditions and the following disclaimer.
14d25013f2Scamield  *
15d25013f2Scamield  * - Redistributions in binary form must reproduce the above copyright
16d25013f2Scamield  * notice, this list of conditions and the following disclaimer in the
17d25013f2Scamield  * documentation and/or other materials provided with the distribution.
18d25013f2Scamield  *
19d25013f2Scamield  * - Neither the name of the Stichting Centrum voor Wiskunde en
20d25013f2Scamield  * Informatica, nor the names of its contributors may be used to endorse or
21d25013f2Scamield  * promote products derived from this software without specific prior
22d25013f2Scamield  * written permission.
23d25013f2Scamield  *
24d25013f2Scamield  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
25d25013f2Scamield  * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
26d25013f2Scamield  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
27d25013f2Scamield  * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
28d25013f2Scamield  * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
29d25013f2Scamield  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
30d25013f2Scamield  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
31d25013f2Scamield  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
32d25013f2Scamield  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
33d25013f2Scamield  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
34d25013f2Scamield  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
35d25013f2Scamield  */
36d25013f2Scamield 
37d25013f2Scamield /*
38d25013f2Scamield  * Copyright (c) 1982 Jay Fenlason <hack@gnu.org>
39d25013f2Scamield  * All rights reserved.
40d25013f2Scamield  *
41d25013f2Scamield  * Redistribution and use in source and binary forms, with or without
42d25013f2Scamield  * modification, are permitted provided that the following conditions
43d25013f2Scamield  * are met:
44d25013f2Scamield  * 1. Redistributions of source code must retain the above copyright
45d25013f2Scamield  *    notice, this list of conditions and the following disclaimer.
46d25013f2Scamield  * 2. Redistributions in binary form must reproduce the above copyright
47d25013f2Scamield  *    notice, this list of conditions and the following disclaimer in the
48d25013f2Scamield  *    documentation and/or other materials provided with the distribution.
49d25013f2Scamield  * 3. The name of the author may not be used to endorse or promote products
50d25013f2Scamield  *    derived from this software without specific prior written permission.
51d25013f2Scamield  *
52d25013f2Scamield  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
53d25013f2Scamield  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
54d25013f2Scamield  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
55d25013f2Scamield  * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
56d25013f2Scamield  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
57d25013f2Scamield  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
58d25013f2Scamield  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
59d25013f2Scamield  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
60d25013f2Scamield  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
61d25013f2Scamield  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
62df930be7Sderaadt  */
63df930be7Sderaadt 
64aed906e4Smestre #include <signal.h>
654a5fbbc4Spjanzen #include <stdio.h>
66d07db98cSpjanzen #include <stdlib.h>
67df930be7Sderaadt #include <unistd.h>
68aed906e4Smestre 
694a5fbbc4Spjanzen #include "hack.h"
70df930be7Sderaadt 
714a5fbbc4Spjanzen extern char genocided[60];	/* defined in Decl.c */
724a5fbbc4Spjanzen extern char fut_geno[60];	/* idem */
73df930be7Sderaadt extern char SAVEF[], nul[];
74df930be7Sderaadt extern char pl_character[PL_CSIZ];
75df930be7Sderaadt 
764a5fbbc4Spjanzen static int dosave0(int);
774a5fbbc4Spjanzen 
784a5fbbc4Spjanzen int
dosave(void)79aed906e4Smestre dosave(void)
804a5fbbc4Spjanzen {
81df930be7Sderaadt 	if(dosave0(0)) {
82df930be7Sderaadt 		settty("Be seeing you ...\n");
83df930be7Sderaadt 		exit(0);
84df930be7Sderaadt 	}
85df930be7Sderaadt 	return(0);
86df930be7Sderaadt }
87df930be7Sderaadt 
884a5fbbc4Spjanzen void
hackhangup(int notused)894a5fbbc4Spjanzen hackhangup(int notused)
904a5fbbc4Spjanzen {
91df930be7Sderaadt 	(void) dosave0(1);
92df930be7Sderaadt 	exit(1);
93df930be7Sderaadt }
94df930be7Sderaadt 
95df930be7Sderaadt /* returns 1 if save successful */
964a5fbbc4Spjanzen static int
dosave0(int hu)974a5fbbc4Spjanzen dosave0(int hu)
984a5fbbc4Spjanzen {
994a5fbbc4Spjanzen 	int fd, ofd;
100df930be7Sderaadt 	int tmp;		/* not register ! */
101df930be7Sderaadt 
102df930be7Sderaadt 	(void) signal(SIGHUP, SIG_IGN);
103df930be7Sderaadt 	(void) signal(SIGINT, SIG_IGN);
104*df69c215Sderaadt 	if((fd = open(SAVEF, O_CREAT | O_TRUNC | O_WRONLY, FMASK)) == -1) {
105df930be7Sderaadt 		if(!hu) pline("Cannot open save file. (Continue or Quit)");
106df930be7Sderaadt 		(void) unlink(SAVEF);		/* ab@unido */
107df930be7Sderaadt 		return(0);
108df930be7Sderaadt 	}
109df930be7Sderaadt 	if(flags.moonphase == FULL_MOON)	/* ut-sally!fletcher */
110df930be7Sderaadt 		u.uluck--;			/* and unido!ab */
111df930be7Sderaadt 	savelev(fd,dlevel);
112df930be7Sderaadt 	saveobjchn(fd, invent);
113df930be7Sderaadt 	saveobjchn(fd, fcobj);
114df930be7Sderaadt 	savemonchn(fd, fallen_down);
115df930be7Sderaadt 	tmp = getuid();
116ebf3d589Sguenther 	bwrite(fd, &tmp, sizeof tmp);
117ebf3d589Sguenther 	bwrite(fd, &flags, sizeof(struct flag));
118ebf3d589Sguenther 	bwrite(fd, &dlevel, sizeof dlevel);
119ebf3d589Sguenther 	bwrite(fd, &maxdlevel, sizeof maxdlevel);
120ebf3d589Sguenther 	bwrite(fd, &moves, sizeof moves);
121ebf3d589Sguenther 	bwrite(fd, &u, sizeof(struct you));
122df930be7Sderaadt 	if(u.ustuck)
123ebf3d589Sguenther 		bwrite(fd, &(u.ustuck->m_id), sizeof u.ustuck->m_id);
124ebf3d589Sguenther 	bwrite(fd, pl_character, sizeof pl_character);
125ebf3d589Sguenther 	bwrite(fd, genocided, sizeof genocided);
126ebf3d589Sguenther 	bwrite(fd, fut_geno, sizeof fut_geno);
127df930be7Sderaadt 	savenames(fd);
128df930be7Sderaadt 	for(tmp = 1; tmp <= maxdlevel; tmp++) {
129df930be7Sderaadt 		extern int hackpid;
130df930be7Sderaadt 		extern boolean level_exists[];
131df930be7Sderaadt 
132df930be7Sderaadt 		if(tmp == dlevel || !level_exists[tmp]) continue;
133df930be7Sderaadt 		glo(tmp);
134*df69c215Sderaadt 		if((ofd = open(lock, O_RDONLY)) == -1) {
135df930be7Sderaadt 		    if(!hu) pline("Error while saving: cannot read %s.", lock);
136df930be7Sderaadt 		    (void) close(fd);
137df930be7Sderaadt 		    (void) unlink(SAVEF);
138df930be7Sderaadt 		    if(!hu) done("tricked");
139df930be7Sderaadt 		    return(0);
140df930be7Sderaadt 		}
141df930be7Sderaadt 		getlev(ofd, hackpid, tmp);
142df930be7Sderaadt 		(void) close(ofd);
143ebf3d589Sguenther 		bwrite(fd, &tmp, sizeof tmp);		/* level number */
144df930be7Sderaadt 		savelev(fd,tmp);			/* actual level */
145df930be7Sderaadt 		(void) unlink(lock);
146df930be7Sderaadt 	}
147df930be7Sderaadt 	(void) close(fd);
148df930be7Sderaadt 	glo(dlevel);
149df930be7Sderaadt 	(void) unlink(lock);	/* get rid of current level --jgm */
150df930be7Sderaadt 	glo(0);
151df930be7Sderaadt 	(void) unlink(lock);
152df930be7Sderaadt 	return(1);
153df930be7Sderaadt }
154df930be7Sderaadt 
1554a5fbbc4Spjanzen int
dorecover(int fd)1564a5fbbc4Spjanzen dorecover(int fd)
157df930be7Sderaadt {
1584a5fbbc4Spjanzen 	int nfd;
159df930be7Sderaadt 	int tmp;		/* not a register ! */
160df930be7Sderaadt 	unsigned mid;		/* idem */
161df930be7Sderaadt 	struct obj *otmp;
162df930be7Sderaadt 	extern boolean restoring;
163df930be7Sderaadt 
164df930be7Sderaadt 	restoring = TRUE;
165df930be7Sderaadt 	getlev(fd, 0, 0);
166df930be7Sderaadt 	invent = restobjchn(fd);
167df930be7Sderaadt 	for(otmp = invent; otmp; otmp = otmp->nobj)
168df930be7Sderaadt 		if(otmp->owornmask)
169df930be7Sderaadt 			setworn(otmp, otmp->owornmask);
170df930be7Sderaadt 	fcobj = restobjchn(fd);
171df930be7Sderaadt 	fallen_down = restmonchn(fd);
172df930be7Sderaadt 	mread(fd, (char *) &tmp, sizeof tmp);
173df930be7Sderaadt 	if(tmp != getuid()) {		/* strange ... */
174df930be7Sderaadt 		(void) close(fd);
175df930be7Sderaadt 		(void) unlink(SAVEF);
176df930be7Sderaadt 		puts("Saved game was not yours.");
177df930be7Sderaadt 		restoring = FALSE;
178df930be7Sderaadt 		return(0);
179df930be7Sderaadt 	}
180df930be7Sderaadt 	mread(fd, (char *) &flags, sizeof(struct flag));
181df930be7Sderaadt 	mread(fd, (char *) &dlevel, sizeof dlevel);
182df930be7Sderaadt 	mread(fd, (char *) &maxdlevel, sizeof maxdlevel);
183df930be7Sderaadt 	mread(fd, (char *) &moves, sizeof moves);
184df930be7Sderaadt 	mread(fd, (char *) &u, sizeof(struct you));
185df930be7Sderaadt 	if(u.ustuck)
186df930be7Sderaadt 		mread(fd, (char *) &mid, sizeof mid);
187df930be7Sderaadt 	mread(fd, (char *) pl_character, sizeof pl_character);
188df930be7Sderaadt 	mread(fd, (char *) genocided, sizeof genocided);
189df930be7Sderaadt 	mread(fd, (char *) fut_geno, sizeof fut_geno);
190df930be7Sderaadt 	restnames(fd);
191df930be7Sderaadt 	while(1) {
192df930be7Sderaadt 		if(read(fd, (char *) &tmp, sizeof tmp) != sizeof tmp)
193df930be7Sderaadt 			break;
194df930be7Sderaadt 		getlev(fd, 0, tmp);
195df930be7Sderaadt 		glo(tmp);
196*df69c215Sderaadt 		if((nfd = open(lock, O_CREAT | O_TRUNC | O_WRONLY, FMASK)) == -1)
197df930be7Sderaadt 			panic("Cannot open temp file %s!\n", lock);
198df930be7Sderaadt 		savelev(nfd,tmp);
199df930be7Sderaadt 		(void) close(nfd);
200df930be7Sderaadt 	}
20129a36e40Sderaadt 	(void) lseek(fd, (off_t)0, SEEK_SET);
202df930be7Sderaadt 	getlev(fd, 0, 0);
203df930be7Sderaadt 	(void) close(fd);
204df930be7Sderaadt 	(void) unlink(SAVEF);
205df930be7Sderaadt 	if(Punished) {
206df930be7Sderaadt 		for(otmp = fobj; otmp; otmp = otmp->nobj)
207df930be7Sderaadt 			if(otmp->olet == CHAIN_SYM) goto chainfnd;
208df930be7Sderaadt 		panic("Cannot find the iron chain?");
209df930be7Sderaadt 	chainfnd:
210df930be7Sderaadt 		uchain = otmp;
211df930be7Sderaadt 		if(!uball){
212df930be7Sderaadt 			for(otmp = fobj; otmp; otmp = otmp->nobj)
213df930be7Sderaadt 				if(otmp->olet == BALL_SYM && otmp->spe)
214df930be7Sderaadt 					goto ballfnd;
215df930be7Sderaadt 			panic("Cannot find the iron ball?");
216df930be7Sderaadt 		ballfnd:
217df930be7Sderaadt 			uball = otmp;
218df930be7Sderaadt 		}
219df930be7Sderaadt 	}
220df930be7Sderaadt 	if(u.ustuck) {
2214a5fbbc4Spjanzen 		struct monst *mtmp;
222df930be7Sderaadt 
223df930be7Sderaadt 		for(mtmp = fmon; mtmp; mtmp = mtmp->nmon)
224df930be7Sderaadt 			if(mtmp->m_id == mid) goto monfnd;
225df930be7Sderaadt 		panic("Cannot find the monster ustuck.");
226df930be7Sderaadt 	monfnd:
227df930be7Sderaadt 		u.ustuck = mtmp;
228df930be7Sderaadt 	}
229df930be7Sderaadt #ifndef QUEST
230df930be7Sderaadt 	setsee();  /* only to recompute seelx etc. - these weren't saved */
23154da88e4Spjanzen #endif /* QUEST */
232df930be7Sderaadt 	docrt();
233df930be7Sderaadt 	restoring = FALSE;
234df930be7Sderaadt 	return(1);
235df930be7Sderaadt }
236df930be7Sderaadt 
237df930be7Sderaadt struct obj *
restobjchn(int fd)2384a5fbbc4Spjanzen restobjchn(int fd)
239df930be7Sderaadt {
2404a5fbbc4Spjanzen 	struct obj *otmp, *otmp2;
2414a5fbbc4Spjanzen 	struct obj *first = 0;
242df930be7Sderaadt 	int xl;
243df930be7Sderaadt 	while(1) {
244df930be7Sderaadt 		mread(fd, (char *) &xl, sizeof(xl));
245df930be7Sderaadt 		if(xl == -1) break;
246df930be7Sderaadt 		otmp = newobj(xl);
247df930be7Sderaadt 		if(!first) first = otmp;
248df930be7Sderaadt 		else otmp2->nobj = otmp;
249df930be7Sderaadt 		mread(fd, (char *) otmp, (unsigned) xl + sizeof(struct obj));
250df930be7Sderaadt 		if(!otmp->o_id) otmp->o_id = flags.ident++;
251df930be7Sderaadt 		otmp2 = otmp;
252df930be7Sderaadt 	}
253df930be7Sderaadt 	if(first && otmp2->nobj){
254df930be7Sderaadt 		impossible("Restobjchn: error reading objchn.");
255df930be7Sderaadt 		otmp2->nobj = 0;
256df930be7Sderaadt 	}
257df930be7Sderaadt 	return(first);
258df930be7Sderaadt }
259df930be7Sderaadt 
260df930be7Sderaadt struct monst *
restmonchn(int fd)2614a5fbbc4Spjanzen restmonchn(int fd)
262df930be7Sderaadt {
2634a5fbbc4Spjanzen 	struct monst *mtmp, *mtmp2;
2644a5fbbc4Spjanzen 	struct monst *first = 0;
265df930be7Sderaadt 	int xl;
266df930be7Sderaadt 
267df930be7Sderaadt 	struct permonst *monbegin;
268df930be7Sderaadt 	long differ;
269df930be7Sderaadt 
270df930be7Sderaadt 	mread(fd, (char *)&monbegin, sizeof(monbegin));
271df930be7Sderaadt 	differ = (char *)(&mons[0]) - (char *)(monbegin);
272df930be7Sderaadt 
273df930be7Sderaadt 	while(1) {
274df930be7Sderaadt 		mread(fd, (char *) &xl, sizeof(xl));
275df930be7Sderaadt 		if(xl == -1) break;
276df930be7Sderaadt 		mtmp = newmonst(xl);
277df930be7Sderaadt 		if(!first) first = mtmp;
278df930be7Sderaadt 		else mtmp2->nmon = mtmp;
279df930be7Sderaadt 		mread(fd, (char *) mtmp, (unsigned) xl + sizeof(struct monst));
280df930be7Sderaadt 		if(!mtmp->m_id)
281df930be7Sderaadt 			mtmp->m_id = flags.ident++;
282df930be7Sderaadt 		mtmp->data = (struct permonst *)
283df930be7Sderaadt 			((char *) mtmp->data + differ);
284df930be7Sderaadt 		if(mtmp->minvent)
285df930be7Sderaadt 			mtmp->minvent = restobjchn(fd);
286df930be7Sderaadt 		mtmp2 = mtmp;
287df930be7Sderaadt 	}
288df930be7Sderaadt 	if(first && mtmp2->nmon){
289df930be7Sderaadt 		impossible("Restmonchn: error reading monchn.");
290df930be7Sderaadt 		mtmp2->nmon = 0;
291df930be7Sderaadt 	}
292df930be7Sderaadt 	return(first);
293df930be7Sderaadt }
294