]>
git.cameronkatri.com Git - bsdgames-darwin.git/blob - monop/execute.c
1 /* $NetBSD: execute.c,v 1.16 2008/02/24 01:41:14 dholland Exp $ */
4 * Copyright (c) 1980, 1993
5 * The Regents of the University of California. All rights reserved.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of the University nor the names of its contributors
16 * may be used to endorse or promote products derived from this software
17 * without specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
23 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32 #include <sys/cdefs.h>
35 static char sccsid
[] = "@(#)execute.c 8.1 (Berkeley) 5/31/93";
37 __RCSID("$NetBSD: execute.c,v 1.16 2008/02/24 01:41:14 dholland Exp $");
46 #include <sys/types.h>
52 #define MIN_FORMAT_VERSION 1
53 #define CUR_FORMAT_VERSION 1
54 #define MAX_FORMAT_VERSION 1
56 typedef struct stat STAT
;
57 typedef struct tm TIME
;
61 static bool new_play
; /* set if move on to new player */
63 static void show_move(void);
65 static void restore_reset(void);
66 static int restore_parseline(char *txt
);
67 static int restore_toplevel_attr(const char *attribute
, char *txt
);
68 static int restore_player_attr(const char *attribute
, char *txt
);
69 static int restore_deck_attr(const char *attribute
, char *txt
);
70 static int restore_square_attr(const char *attribute
, char *txt
);
71 static int getnum(const char *what
, char *txt
, int min
, int max
, int *ret
);
72 static int getnum_withbrace(const char *what
, char *txt
, int min
, int max
,
76 * This routine executes the given command by index number
82 new_play
= FALSE
; /* new_play is true if fixing */
89 printf("%s rolled doubles. Goes again\n", cur_p
->name
);
93 * This routine moves a piece around.
101 new_play
= was_jail
= FALSE
;
102 printf("roll is %d, %d\n", r1
=roll(1, 6), r2
=roll(1, 6));
103 if (cur_p
->loc
== JAIL
) {
105 if (!move_jail(r1
, r2
)) {
111 if (r1
== r2
&& ++num_doub
== 3) {
112 printf("That's 3 doubles. You go to jail\n");
119 if (r1
!= r2
|| was_jail
)
126 * This routine moves a normal move
134 old_loc
= cur_p
->loc
;
135 cur_p
->loc
= (cur_p
->loc
+ rl
) % N_SQRS
;
136 if (cur_p
->loc
< old_loc
&& rl
> 0) {
138 printf("You pass %s and get $200\n", board
[0].name
);
144 * This routine shows the results of a move
151 sqp
= &board
[cur_p
->loc
];
152 printf("That puts you on %s\n", sqp
->name
);
155 printf("That is a safe place\n");
175 if (sqp
->owner
< 0) {
176 printf("That would cost $%d\n", sqp
->cost
);
177 if (getyn("Do you want to buy? ") == 0) {
179 cur_p
->money
-= sqp
->cost
;
181 else if (num_play
> 2)
184 else if (sqp
->owner
== player
)
185 printf("You own it.\n");
192 * Reset the game state.
199 for (i
= 0; i
< N_SQRS
; i
++) {
201 if (board
[i
].type
== PRPTY
) {
202 board
[i
].desc
->morg
= 0;
203 board
[i
].desc
->houses
= 0;
204 } else if (board
[i
].type
== RR
|| board
[i
].type
== UTIL
) {
205 board
[i
].desc
->morg
= 0;
209 for (i
= 0; i
< 2; i
++) {
210 deck
[i
].top_card
= 0;
211 deck
[i
].gojf_used
= FALSE
;
215 for (i
= 0; i
< num_play
; i
++) {
223 for (i
= 0; i
< MAX_PL
+2; i
++) {
239 * This routine saves the current game for use at a later date
250 printf("Which file do you wish to save it in? ");
251 fgets(buf
, sizeof(buf
), stdin
);
254 sp
= strchr(buf
, '\n');
259 * check for existing files, and confirm overwrite if needed
262 if (stat(buf
, &sb
) == 0
263 && getyn("File exists. Do you wish to overwrite? ") > 0)
266 outf
= fopen(buf
, "w");
271 printf("\"%s\" ", buf
);
272 time(&t
); /* get current time */
275 fprintf(outf
, "NetBSD monop format v%d\n", CUR_FORMAT_VERSION
);
276 fprintf(outf
, "time %s", ctime(&t
)); /* ctime includes a \n */
277 fprintf(outf
, "numplayers %d\n", num_play
);
278 fprintf(outf
, "currentplayer %d\n", player
);
279 fprintf(outf
, "doubles %d\n", num_doub
);
282 for (i
= 0; i
< num_play
; i
++) {
283 fprintf(outf
, "player %d {\n", i
);
284 fprintf(outf
, " name %s\n", name_list
[i
]);
285 fprintf(outf
, " money %d\n", play
[i
].money
);
286 fprintf(outf
, " loc %d\n", play
[i
].loc
);
287 fprintf(outf
, " num_gojf %d\n", play
[i
].num_gojf
);
288 fprintf(outf
, " in_jail %d\n", play
[i
].in_jail
);
289 fprintf(outf
, "}\n");
293 for (i
= 0; i
< 2; i
++) {
294 fprintf(outf
, "deck %d {\n", i
);
295 fprintf(outf
, " numcards %d\n", deck
[i
].num_cards
);
296 fprintf(outf
, " topcard %d\n", deck
[i
].top_card
);
297 fprintf(outf
, " gojf_used %d\n", deck
[i
].gojf_used
);
298 fprintf(outf
, " offsets");
299 for (j
= 0; j
< deck
[i
].num_cards
; j
++)
300 fprintf(outf
, " %ld", (long)(deck
[i
].offsets
[j
]));
302 fprintf(outf
, "}\n");
306 for (i
= 0; i
< N_SQRS
; i
++) {
307 fprintf(outf
, "square %d {\n", i
);
308 fprintf(outf
, "owner %d\n", board
[i
].owner
);
309 if (board
[i
].owner
< 0) {
311 } else if (board
[i
].type
== PRPTY
) {
312 fprintf(outf
, "morg %d\n", board
[i
].desc
->morg
);
313 fprintf(outf
, "houses %d\n", board
[i
].desc
->houses
);
314 } else if (board
[i
].type
== RR
|| board
[i
].type
== UTIL
) {
315 fprintf(outf
, "morg %d\n", board
[i
].desc
->morg
);
317 fprintf(outf
, "}\n");
319 if (ferror(outf
) || fflush(outf
))
320 warnx("write error");
323 strcpy(buf
, ctime(&t
));
324 for (sp
= buf
; *sp
!= '\n'; sp
++)
327 printf("[%s]\n", buf
);
331 * This routine restores an old game from a file
338 printf("Which file do you wish to restore from? ");
339 fgets(buf
, sizeof(buf
), stdin
);
342 sp
= strchr(buf
, '\n');
349 * This does the actual restoring. It returns TRUE if the
350 * backup was successful, else false.
362 inf
= fopen(file
, "r");
367 printf("\"%s\" ", file
);
368 if (fstat(fileno(inf
), &sbuf
) < 0) {
369 err(1, "%s: fstat", file
);
372 /* Clear the game state to prevent brokenness on misordered files. */
375 /* Reset the parser */
378 /* Note: can't use buf[], file might point at it. (Lame...) */
379 while (fgets(readbuf
, sizeof(readbuf
), inf
)) {
381 * The input buffer is long enough to handle anything
382 * that's supposed to be in the output buffer, so if
383 * we get a partial line, complain.
385 sp
= strchr(readbuf
, '\n');
387 printf("file is corrupt: long lines.\n");
392 if (restore_parseline(readbuf
)) {
398 warnx("%s: read error", file
);
401 name_list
[num_play
] = "done";
404 * We could at this point crosscheck the following:
405 * - there are only two GOJF cards floating around
406 * - total number of houses and hotels does not exceed maximums
407 * - no props are both built and mortgaged
408 * but for now we don't.
411 strcpy(xbuf
, ctime(&sbuf
.st_mtime
));
412 for (sp
= xbuf
; *sp
!= '\n'; sp
++)
415 printf("[%s]\n", xbuf
);
420 * State of the restore parser
422 static int restore_version
;
429 static int restore_itemnum
;
432 * Reset the restore parser
437 restore_version
= -1;
438 restore_item
= RI_NONE
;
439 restore_itemnum
= -1;
443 * Handle one line of the save file
446 restore_parseline(char *txt
)
451 if (restore_version
< 0) {
452 /* Haven't seen the header yet. Demand it right away. */
453 if (!strncmp(txt
, "NetBSD monop format v", 21)) {
454 return getnum("format version", txt
+21,
459 printf("file is not a monop save file.\n");
463 /* Check for lines that are right braces. */
464 if (!strcmp(txt
, "}")) {
465 if (restore_item
== RI_NONE
) {
466 printf("mismatched close brace.\n");
469 restore_item
= RI_NONE
;
470 restore_itemnum
= -1;
474 /* Any other line must begin with a word, which is the attribute. */
479 s
= strchr(attribute
, ' ');
481 printf("file is corrupt: attribute %s lacks value.\n",
488 /* keep the remaining text for further handling */
491 switch (restore_item
) {
493 /* toplevel attributes */
494 return restore_toplevel_attr(attribute
, txt
);
497 /* player attributes */
498 return restore_player_attr(attribute
, txt
);
501 /* deck attributes */
502 return restore_deck_attr(attribute
, txt
);
505 /* board square attributes */
506 return restore_square_attr(attribute
, txt
);
509 printf("internal logic error\n");
514 restore_toplevel_attr(const char *attribute
, char *txt
)
516 if (!strcmp(attribute
, "time")) {
518 } else if (!strcmp(attribute
, "numplayers")) {
519 if (getnum("numplayers", txt
, 2, MAX_PL
, &num_play
) < 0) {
523 printf("numplayers: multiple settings\n");
526 play
= calloc(num_play
, sizeof(play
[0]));
530 } else if (!strcmp(attribute
, "currentplayer")) {
531 if (getnum("currentplayer", txt
, 0, num_play
-1, &player
) < 0) {
535 printf("currentplayer: before numplayers\n");
538 cur_p
= &play
[player
];
539 } else if (!strcmp(attribute
, "doubles")) {
540 if (getnum("doubles", txt
, 0, 2, &num_doub
) < 0) {
543 } else if (!strcmp(attribute
, "player")) {
544 if (getnum_withbrace("player", txt
, 0, num_play
-1,
545 &restore_itemnum
) < 0) {
548 restore_item
= RI_PLAYER
;
549 } else if (!strcmp(attribute
, "deck")) {
550 if (getnum_withbrace("deck", txt
, 0, 1,
551 &restore_itemnum
) < 0) {
554 restore_item
= RI_DECK
;
555 } else if (!strcmp(attribute
, "square")) {
556 if (getnum_withbrace("square", txt
, 0, N_SQRS
-1,
557 &restore_itemnum
) < 0) {
560 restore_item
= RI_SQUARE
;
562 printf("unknown attribute %s\n", attribute
);
569 restore_player_attr(const char *attribute
, char *txt
)
575 printf("player came before numplayers.\n");
578 pp
= &play
[restore_itemnum
];
580 if (!strcmp(attribute
, "name")) {
581 if (pp
->name
!= NULL
) {
582 printf("player has multiple names.\n");
585 /* XXX should really systematize the max name length */
586 if (strlen(txt
) > 256) {
589 pp
->name
= strdup(txt
);
590 if (pp
->name
== NULL
)
592 name_list
[restore_itemnum
] = pp
->name
;
593 } else if (!strcmp(attribute
, "money")) {
594 if (getnum(attribute
, txt
, 0, INT_MAX
, &pp
->money
) < 0) {
597 } else if (!strcmp(attribute
, "loc")) {
598 /* note: not N_SQRS-1 */
599 if (getnum(attribute
, txt
, 0, N_SQRS
, &tmp
) < 0) {
603 } else if (!strcmp(attribute
, "num_gojf")) {
604 if (getnum(attribute
, txt
, 0, 2, &tmp
) < 0) {
608 } else if (!strcmp(attribute
, "in_jail")) {
609 if (getnum(attribute
, txt
, 0, 3, &tmp
) < 0) {
613 if (pp
->in_jail
> 0 && pp
->loc
!= JAIL
) {
614 printf("player escaped from jail?\n");
618 printf("unknown attribute %s\n", attribute
);
625 restore_deck_attr(const char *attribute
, char *txt
)
631 dp
= &deck
[restore_itemnum
];
633 if (!strcmp(attribute
, "numcards")) {
634 if (getnum(attribute
, txt
, dp
->num_cards
, dp
->num_cards
,
638 } else if (!strcmp(attribute
, "topcard")) {
639 if (getnum(attribute
, txt
, 0, dp
->num_cards
,
640 &dp
->top_card
) < 0) {
643 } else if (!strcmp(attribute
, "gojf_used")) {
644 if (getnum(attribute
, txt
, 0, 1, &tmp
) < 0) {
648 } else if (!strcmp(attribute
, "offsets")) {
651 for (j
= 0; j
<dp
->num_cards
; j
++) {
652 dp
->offsets
[j
] = strtol(s
, &s
, 10);
655 printf("offsets: invalid values\n");
659 printf("unknown attribute %s\n", attribute
);
666 restore_square_attr(const char *attribute
, char *txt
)
668 SQUARE
*sp
= &board
[restore_itemnum
];
671 if (!strcmp(attribute
, "owner")) {
672 if (getnum(attribute
, txt
, -1, num_play
-1, &tmp
) < 0) {
677 add_list(tmp
, &play
[tmp
].own_list
, restore_itemnum
);
678 } else if (!strcmp(attribute
, "morg")) {
679 if (sp
->type
!= PRPTY
&& sp
->type
!= RR
&& sp
->type
!= UTIL
) {
680 printf("unownable property is mortgaged.\n");
683 if (getnum(attribute
, txt
, 0, 1, &tmp
) < 0) {
686 sp
->desc
->morg
= tmp
;
687 } else if (!strcmp(attribute
, "houses")) {
688 if (sp
->type
!= PRPTY
) {
689 printf("unbuildable property has houses.\n");
692 if (getnum(attribute
, txt
, 0, 5, &tmp
) < 0) {
695 sp
->desc
->houses
= tmp
;
697 printf("unknown attribute %s\n", attribute
);
704 getnum(const char *what
, char *txt
, int min
, int max
, int *ret
)
710 l
= strtol(txt
, &s
, 10);
711 if (errno
|| strlen(s
)>0) {
712 printf("%s: not a number.\n", what
);
715 if (l
< min
|| l
> max
) {
716 printf("%s: out of range.\n", what
);
723 getnum_withbrace(const char *what
, char *txt
, int min
, int max
, int *ret
)
726 s
= strchr(txt
, ' ');
728 printf("%s: expected open brace\n", what
);
735 printf("%s: expected open brace\n", what
);
739 printf("%s: garbage after open brace\n", what
);
742 return getnum(what
, txt
, min
, max
, ret
);