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