1*df69c215Sderaadt /* $OpenBSD: save.c,v 1.14 2019/06/28 13:32:52 deraadt Exp $ */
2df930be7Sderaadt /* $NetBSD: save.c,v 1.4 1995/03/24 05:02:13 cgd Exp $ */
3df930be7Sderaadt
4df930be7Sderaadt /*
5df930be7Sderaadt * Copyright (c) 1983, 1993
6df930be7Sderaadt * The Regents of the University of California. All rights reserved.
7df930be7Sderaadt *
8df930be7Sderaadt * Redistribution and use in source and binary forms, with or without
9df930be7Sderaadt * modification, are permitted provided that the following conditions
10df930be7Sderaadt * are met:
11df930be7Sderaadt * 1. Redistributions of source code must retain the above copyright
12df930be7Sderaadt * notice, this list of conditions and the following disclaimer.
13df930be7Sderaadt * 2. Redistributions in binary form must reproduce the above copyright
14df930be7Sderaadt * notice, this list of conditions and the following disclaimer in the
15df930be7Sderaadt * documentation and/or other materials provided with the distribution.
167a09557bSmillert * 3. Neither the name of the University nor the names of its contributors
17df930be7Sderaadt * may be used to endorse or promote products derived from this software
18df930be7Sderaadt * without specific prior written permission.
19df930be7Sderaadt *
20df930be7Sderaadt * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
21df930be7Sderaadt * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22df930be7Sderaadt * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23df930be7Sderaadt * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
24df930be7Sderaadt * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25df930be7Sderaadt * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26df930be7Sderaadt * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27df930be7Sderaadt * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28df930be7Sderaadt * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29df930be7Sderaadt * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30df930be7Sderaadt * SUCH DAMAGE.
31df930be7Sderaadt */
32df930be7Sderaadt
33182ed36cSmestre #include <sys/stat.h>
34182ed36cSmestre
35182ed36cSmestre #include <err.h>
36182ed36cSmestre #include <errno.h>
37182ed36cSmestre #include <fcntl.h>
38182ed36cSmestre #include <string.h>
39ee7acb09Stb #include <time.h>
40182ed36cSmestre #include <unistd.h>
41182ed36cSmestre
42df930be7Sderaadt #include "mille.h"
43df930be7Sderaadt
44df930be7Sderaadt /*
45df930be7Sderaadt * @(#)save.c 1.2 (Berkeley) 3/28/83
46df930be7Sderaadt */
47df930be7Sderaadt
48df930be7Sderaadt typedef struct stat STAT;
49df930be7Sderaadt
50df930be7Sderaadt /*
51a0b17b52Spjanzen * This routine saves the current game for use at a later date.
52a0b17b52Spjanzen * Returns FALSE if it couldn't be done.
53df930be7Sderaadt */
54a0b17b52Spjanzen bool
save(void)5556f9a7dcSmestre save(void)
56a0b17b52Spjanzen {
57a0b17b52Spjanzen char *sp;
58a0b17b52Spjanzen int outf;
59a0b17b52Spjanzen time_t *tp;
60a0b17b52Spjanzen char buf[256];
61df930be7Sderaadt time_t tme;
62df930be7Sderaadt STAT junk;
63a0b17b52Spjanzen bool rv;
64df930be7Sderaadt
65a0b17b52Spjanzen sp = NULL;
66df930be7Sderaadt tp = &tme;
677faebae9Spjanzen if (Fromfile && getyn(SAMEFILEPROMPT))
687faebae9Spjanzen strlcpy(buf, Fromfile, sizeof(buf));
697faebae9Spjanzen else {
70df930be7Sderaadt over:
71df930be7Sderaadt prompt(FILEPROMPT);
72df930be7Sderaadt leaveok(Board, FALSE);
73df930be7Sderaadt refresh();
74df930be7Sderaadt sp = buf;
75a0b17b52Spjanzen while ((*sp = readch()) != '\n' && *sp != '\r' &&
764e075216Spjanzen (sp - buf < (int)sizeof(buf))) {
77df930be7Sderaadt if (*sp == killchar())
78df930be7Sderaadt goto over;
79df930be7Sderaadt else if (*sp == erasechar()) {
80df930be7Sderaadt if (--sp < buf)
81df930be7Sderaadt sp = buf;
82df930be7Sderaadt else {
83df930be7Sderaadt addch('\b');
84df930be7Sderaadt /*
85df930be7Sderaadt * if the previous char was a control
86df930be7Sderaadt * char, cover up two characters.
87df930be7Sderaadt */
88df930be7Sderaadt if (*sp < ' ')
89df930be7Sderaadt addch('\b');
90df930be7Sderaadt clrtoeol();
91df930be7Sderaadt }
92df930be7Sderaadt }
93df930be7Sderaadt else {
94df930be7Sderaadt addstr(unctrl(*sp));
95df930be7Sderaadt ++sp;
96df930be7Sderaadt }
97df930be7Sderaadt refresh();
98df930be7Sderaadt }
99df930be7Sderaadt *sp = '\0';
100df930be7Sderaadt leaveok(Board, TRUE);
101df930be7Sderaadt }
102df930be7Sderaadt
103df930be7Sderaadt /*
104df930be7Sderaadt * check for existing files, and confirm overwrite if needed
105df930be7Sderaadt */
106df930be7Sderaadt
107df930be7Sderaadt if (sp == buf || (!Fromfile && stat(buf, &junk) > -1
108df930be7Sderaadt && getyn(OVERWRITEFILEPROMPT) == FALSE))
109df930be7Sderaadt return FALSE;
110df930be7Sderaadt
111*df69c215Sderaadt if ((outf = open(buf, O_CREAT | O_TRUNC | O_WRONLY, 0644)) == -1) {
112df930be7Sderaadt error(strerror(errno));
113df930be7Sderaadt return FALSE;
114df930be7Sderaadt }
115df930be7Sderaadt mvwaddstr(Score, ERR_Y, ERR_X, buf);
116df930be7Sderaadt wrefresh(Score);
117df930be7Sderaadt time(tp); /* get current time */
118a0b17b52Spjanzen rv = varpush(outf, writev);
119a0b17b52Spjanzen close(outf);
120a0b17b52Spjanzen if (!rv)
121a0b17b52Spjanzen unlink(buf);
122a0b17b52Spjanzen else {
12342ceebb3Sderaadt strlcpy(buf, ctime(tp), sizeof buf);
124df930be7Sderaadt for (sp = buf; *sp != '\n'; sp++)
125df930be7Sderaadt continue;
126df930be7Sderaadt *sp = '\0';
127df930be7Sderaadt wprintw(Score, " [%s]", buf);
128a0b17b52Spjanzen }
129df930be7Sderaadt wclrtoeol(Score);
130df930be7Sderaadt wrefresh(Score);
131a0b17b52Spjanzen return rv;
132df930be7Sderaadt }
133df930be7Sderaadt
134df930be7Sderaadt /*
135df930be7Sderaadt * This does the actual restoring. It returns TRUE if the
136df930be7Sderaadt * backup was made on exiting, in which case certain things must
137df930be7Sderaadt * be cleaned up before the game starts.
138df930be7Sderaadt */
139a0b17b52Spjanzen bool
rest_f(const char * file)14056f9a7dcSmestre rest_f(const char *file)
141a0b17b52Spjanzen {
142a0b17b52Spjanzen char *sp;
143a0b17b52Spjanzen int inf;
144df930be7Sderaadt char buf[80];
145df930be7Sderaadt STAT sbuf;
146df930be7Sderaadt
147*df69c215Sderaadt if ((inf = open(file, O_RDONLY)) == -1)
148a0b17b52Spjanzen err(1, "%s", file);
149*df69c215Sderaadt if (fstat(inf, &sbuf) == -1) /* get file stats */
150a0b17b52Spjanzen err(1, "%s", file);
151a0b17b52Spjanzen varpush(inf, readv);
152df930be7Sderaadt close(inf);
15342ceebb3Sderaadt strlcpy(buf, ctime(&sbuf.st_mtime), sizeof buf);
154df930be7Sderaadt for (sp = buf; *sp != '\n'; sp++)
155df930be7Sderaadt continue;
156df930be7Sderaadt *sp = '\0';
157df930be7Sderaadt /*
158df930be7Sderaadt * initialize some necessary values
159df930be7Sderaadt */
16042ceebb3Sderaadt (void)snprintf(Initstr, sizeof Initstr, "%s [%s]\n", file, buf);
161df930be7Sderaadt Fromfile = file;
162df930be7Sderaadt return !On_exit;
163df930be7Sderaadt }
164