123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371 |
- /* $NetBSD: worm.c,v 1.25 2004/01/27 20:30:31 jsm Exp $ */
- /*
- * Copyright (c) 1980, 1993
- * The Regents of the University of California. All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. Neither the name of the University nor the names of its contributors
- * may be used to endorse or promote products derived from this software
- * without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
- * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- */
- #include <sys/cdefs.h>
- #ifndef lint
- __COPYRIGHT("@(#) Copyright (c) 1980, 1993\n\
- The Regents of the University of California. All rights reserved.\n");
- #endif /* not lint */
- #ifndef lint
- #if 0
- static char sccsid[] = "@(#)worm.c 8.1 (Berkeley) 5/31/93";
- #else
- __RCSID("$NetBSD: worm.c,v 1.25 2004/01/27 20:30:31 jsm Exp $");
- #endif
- #endif /* not lint */
- /*
- * Worm. Written by Michael Toy
- * UCSC
- */
- #include <ctype.h>
- #include <curses.h>
- #include <err.h>
- #include <signal.h>
- #include <stdlib.h>
- #include <termios.h>
- #include <unistd.h>
- #define newlink() (struct body *) malloc(sizeof (struct body));
- #define HEAD '@'
- #define BODY 'o'
- #define LENGTH 7
- #define RUNLEN 8
- #define CNTRL(p) (p-'A'+1)
- WINDOW *tv;
- WINDOW *stw;
- struct body {
- int x;
- int y;
- struct body *prev;
- struct body *next;
- } *head, *tail, goody;
- int growing = 0;
- int running = 0;
- int slow = 0;
- int score = 0;
- int start_len = LENGTH;
- int visible_len;
- int lastch;
- char outbuf[BUFSIZ];
- void crash(void) __attribute__((__noreturn__));
- void display(const struct body *, char);
- int main(int, char **);
- void leave(int) __attribute__((__noreturn__));
- void life(void);
- void newpos(struct body *);
- void process(int);
- void prize(void);
- int rnd(int);
- void setup(void);
- void wake(int);
- int
- main(argc, argv)
- int argc;
- char **argv;
- {
- /* Revoke setgid privileges */
- setregid(getgid(), getgid());
- setbuf(stdout, outbuf);
- srand(getpid());
- signal(SIGALRM, wake);
- signal(SIGINT, leave);
- signal(SIGQUIT, leave);
- initscr();
- cbreak();
- noecho();
- #ifdef KEY_LEFT
- keypad(stdscr, TRUE);
- #endif
- slow = (baudrate() <= 1200);
- clear();
- if (COLS < 18 || LINES < 5) {
- /*
- * Insufficient room for the line with " Worm" and the
- * score if fewer than 18 columns; insufficient room for
- * anything much if fewer than 5 lines.
- */
- endwin();
- errx(1, "screen too small");
- }
- if (argc == 2)
- start_len = atoi(argv[1]);
- if ((start_len <= 0) || (start_len > ((LINES-3) * (COLS-2)) / 3))
- start_len = LENGTH;
- stw = newwin(1, COLS-1, 0, 0);
- tv = newwin(LINES-1, COLS-1, 1, 0);
- box(tv, '*', '*');
- scrollok(tv, FALSE);
- scrollok(stw, FALSE);
- wmove(stw, 0, 0);
- wprintw(stw, " Worm");
- refresh();
- wrefresh(stw);
- wrefresh(tv);
- life(); /* Create the worm */
- prize(); /* Put up a goal */
- while(1)
- {
- if (running)
- {
- running--;
- process(lastch);
- }
- else
- {
- fflush(stdout);
- process(getch());
- }
- }
- }
- void
- life()
- {
- struct body *bp, *np;
- int i, j = 1;
- np = NULL;
- head = newlink();
- if (head == NULL)
- err(1, NULL);
- head->x = start_len % (COLS-5) + 2;
- head->y = LINES / 2;
- head->next = NULL;
- display(head, HEAD);
- for (i = 0, bp = head; i < start_len; i++, bp = np) {
- np = newlink();
- if (np == NULL)
- err(1, NULL);
- np->next = bp;
- bp->prev = np;
- if (((bp->x <= 2) && (j == 1)) || ((bp->x >= COLS-4) && (j == -1))) {
- j *= -1;
- np->x = bp->x;
- np->y = bp->y + 1;
- } else {
- np->x = bp->x - j;
- np->y = bp->y;
- }
- display(np, BODY);
- }
- tail = np;
- tail->prev = NULL;
- visible_len = start_len + 1;
- }
- void
- display(pos, chr)
- const struct body *pos;
- char chr;
- {
- wmove(tv, pos->y, pos->x);
- waddch(tv, chr);
- }
- void
- leave(dummy)
- int dummy;
- {
- endwin();
- if (dummy == 0){ /* called via crash() */
- printf("\nWell, you ran into something and the game is over.\n");
- printf("Your final score was %d\n\n", score);
- }
- exit(0);
- }
- void
- wake(dummy)
- int dummy __attribute__((__unused__));
- {
- signal(SIGALRM, wake);
- fflush(stdout);
- process(lastch);
- }
- int
- rnd(range)
- int range;
- {
- return abs((rand()>>5)+(rand()>>5)) % range;
- }
- void
- newpos(bp)
- struct body * bp;
- {
- if (visible_len == (LINES-3) * (COLS-3) - 1) {
- endwin();
- printf("\nYou won!\n");
- printf("Your final score was %d\n\n", score);
- exit(0);
- }
- do {
- bp->y = rnd(LINES-3)+ 1;
- bp->x = rnd(COLS-3) + 1;
- wmove(tv, bp->y, bp->x);
- } while(winch(tv) != ' ');
- }
- void
- prize()
- {
- int value;
- value = rnd(9) + 1;
- newpos(&goody);
- waddch(tv, value+'0');
- wrefresh(tv);
- }
- void
- process(ch)
- int ch;
- {
- int x,y;
- struct body *nh;
- alarm(0);
- x = head->x;
- y = head->y;
- switch(ch)
- {
- #ifdef KEY_LEFT
- case KEY_LEFT:
- #endif
- case 'h':
- x--; break;
- #ifdef KEY_DOWN
- case KEY_DOWN:
- #endif
- case 'j':
- y++; break;
- #ifdef KEY_UP
- case KEY_UP:
- #endif
- case 'k':
- y--; break;
- #ifdef KEY_RIGHT
- case KEY_RIGHT:
- #endif
- case 'l':
- x++; break;
- case 'H': x--; running = RUNLEN; ch = tolower(ch); break;
- case 'J': y++; running = RUNLEN/2; ch = tolower(ch); break;
- case 'K': y--; running = RUNLEN/2; ch = tolower(ch); break;
- case 'L': x++; running = RUNLEN; ch = tolower(ch); break;
- case '\f': setup(); return;
- case ERR:
- case CNTRL('C'):
- case CNTRL('D'):
- crash();
- return;
- default: if (! running) alarm(1);
- return;
- }
- lastch = ch;
- if (growing == 0)
- {
- display(tail, ' ');
- tail->next->prev = NULL;
- nh = tail->next;
- free(tail);
- tail = nh;
- visible_len--;
- }
- else growing--;
- display(head, BODY);
- wmove(tv, y, x);
- if (isdigit(ch = winch(tv)))
- {
- growing += ch-'0';
- prize();
- score += growing;
- running = 0;
- wmove(stw, 0, COLS - 12);
- wprintw(stw, "Score: %3d", score);
- wrefresh(stw);
- }
- else if(ch != ' ') crash();
- nh = newlink();
- if (nh == NULL)
- err(1, NULL);
- nh->next = NULL;
- nh->prev = head;
- head->next = nh;
- nh->y = y;
- nh->x = x;
- display(nh, HEAD);
- head = nh;
- visible_len++;
- if (!(slow && running))
- {
- wmove(tv, head->y, head->x);
- wrefresh(tv);
- }
- if (!running)
- alarm(1);
- }
- void
- crash()
- {
- leave(0);
- }
- void
- setup()
- {
- clear();
- refresh();
- touchwin(stw);
- wrefresh(stw);
- touchwin(tv);
- wrefresh(tv);
- alarm(1);
- }
|