/**************************************************************************** * Copyright (c) 1998-2012,2013 Free Software Foundation, Inc. * * * * Permission is hereby granted, free of charge, to any person obtaining a * * copy of this software and associated documentation files (the * * "Software"), to deal in the Software without restriction, including * * without limitation the rights to use, copy, modify, merge, publish, * * distribute, distribute with modifications, sublicense, and/or sell * * copies of the Software, and to permit persons to whom the Software is * * furnished to do so, subject to the following conditions: * * * * The above copyright notice and this permission notice shall be included * * in all copies or substantial portions of the Software. * * * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, * * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR * * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR * * THE USE OR OTHER DEALINGS IN THE SOFTWARE. * * * * Except as contained in this notice, the name(s) of the above copyright * * holders shall not be used in advertising or otherwise to promote the * * sale, use or other dealings in this Software without prior written * * authorization. * ****************************************************************************/ /* @@@ @@@ @@@@@@@@@@ @@@@@@@@@@@ @@@@@@@@@@@@ @@@ @@@ @@@@@@@@@@@@ @@@@@@@@@@@@ @@@@@@@@@@@@@ @@@ @@@ @@@@ @@@@ @@@@ @@@@ @@@ @@@@ @@@ @@ @@@ @@@ @@@ @@@ @@@ @@@ @@@ @@@ @@@@ @@@ @@@ @@@ @@@ @@@ @@@ @@@ @@@@ @@@@ @@@@ @@@ @@@ @@@ @@@ @@@ @@@ @@@@@@@@@@@@ @@@@ @@@@ @@@ @@@ @@@ @@@ @@@@ @@@@ @@@@@@@@@@@@ @@@ @@@ @@@ @@@ @@ @@ @@@@@@@@@@ @@@ @@@ @@@ @@@ Eric P. Scott Caltech High Energy Physics October, 1980 Hacks to turn this into a test frame for cursor movement: Eric S. Raymond <esr@snark.thyrsus.com> January, 1995 July 1995 (esr): worms is now in living color! :-) Options: -f fill screen with copies of 'WORM' at start. -l <n> set worm length -n <n> set number of worms -t make worms leave droppings -T <start> <end> set trace interval -S set single-stepping during trace interval -N suppress cursor-movement optimization This program makes a good torture-test for the ncurses cursor-optimization code. You can use -T to set the worm move interval over which movement traces will be dumped. The program stops and waits for one character of input at the beginning and end of the interval. $Id: worm.c,v 1.64 2013/04/27 19:50:17 tom Exp $ */ #include <test.priv.h> #ifdef USE_PTHREADS #include <pthread.h> #endif WANT_USE_WINDOW(); #define MAX_WORMS 40 #define MAX_LENGTH 1024 static chtype flavor[] = { 'O', '*', '#', '$', '%', '0', '@', }; static const int xinc[] = { 1, 1, 1, 0, -1, -1, -1, 0 }, yinc[] = { -1, 0, 1, 1, 1, 0, -1, -1 }; typedef struct worm { int orientation; int head; int *xpos; int *ypos; chtype attrs; #ifdef USE_PTHREADS pthread_t thread; #endif } WORM; static unsigned long sequence = 0; static bool quitting = FALSE; static WORM worm[MAX_WORMS]; static int **refs; static int last_x, last_y; static const char *field; static int length = 16, number = 3; static chtype trail = ' '; static unsigned pending; #ifdef TRACE static int generation, trace_start, trace_end; #endif /* TRACE */ /* *INDENT-OFF* */ static const struct options { int nopts; int opts[3]; } normal[8]={ { 3, { 7, 0, 1 } }, { 3, { 0, 1, 2 } }, { 3, { 1, 2, 3 } }, { 3, { 2, 3, 4 } }, { 3, { 3, 4, 5 } }, { 3, { 4, 5, 6 } }, { 3, { 5, 6, 7 } }, { 3, { 6, 7, 0 } } }, upper[8]={ { 1, { 1, 0, 0 } }, { 2, { 1, 2, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 2, { 4, 5, 0 } }, { 1, { 5, 0, 0 } }, { 2, { 1, 5, 0 } } }, left[8]={ { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 2, { 2, 3, 0 } }, { 1, { 3, 0, 0 } }, { 2, { 3, 7, 0 } }, { 1, { 7, 0, 0 } }, { 2, { 7, 0, 0 } } }, right[8]={ { 1, { 7, 0, 0 } }, { 2, { 3, 7, 0 } }, { 1, { 3, 0, 0 } }, { 2, { 3, 4, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 2, { 6, 7, 0 } } }, lower[8]={ { 0, { 0, 0, 0 } }, { 2, { 0, 1, 0 } }, { 1, { 1, 0, 0 } }, { 2, { 1, 5, 0 } }, { 1, { 5, 0, 0 } }, { 2, { 5, 6, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } } }, upleft[8]={ { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 1, { 3, 0, 0 } }, { 2, { 1, 3, 0 } }, { 1, { 1, 0, 0 } } }, upright[8]={ { 2, { 3, 5, 0 } }, { 1, { 3, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 1, { 5, 0, 0 } } }, lowleft[8]={ { 3, { 7, 0, 1 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 1, { 1, 0, 0 } }, { 2, { 1, 7, 0 } }, { 1, { 7, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } } }, lowright[8]={ { 0, { 0, 0, 0 } }, { 1, { 7, 0, 0 } }, { 2, { 5, 7, 0 } }, { 1, { 5, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } }, { 0, { 0, 0, 0 } } }; /* *INDENT-ON* */ static void failed(const char *s) { perror(s); endwin(); ExitProgram(EXIT_FAILURE); } static void cleanup(void) { USING_WINDOW(stdscr, wrefresh); curs_set(1); endwin(); } static void onsig(int sig GCC_UNUSED) { cleanup(); ExitProgram(EXIT_FAILURE); } static double ranf(void) { long r = (rand() & 077777); return ((double) r / 32768.); } static int draw_worm(WINDOW *win, void *data) { WORM *w = (WORM *) data; const struct options *op; unsigned mask = (unsigned) (~(1 << (w - worm))); chtype attrs = w->attrs | ((mask & pending) ? A_REVERSE : 0); int x; int y; int h; bool done = FALSE; if ((x = w->xpos[h = w->head]) < 0) { wmove(win, y = w->ypos[h] = last_y, x = w->xpos[h] = 0); waddch(win, attrs); refs[y][x]++; } else { y = w->ypos[h]; } if (x > last_x) x = last_x; if (y > last_y) y = last_y; if (++h == length) h = 0; if (w->xpos[w->head = h] >= 0) { int x1, y1; x1 = w->xpos[h]; y1 = w->ypos[h]; if (y1 < LINES && x1 < COLS && --refs[y1][x1] == 0) { wmove(win, y1, x1); waddch(win, trail); } } op = &(x == 0 ? (y == 0 ? upleft : (y == last_y ? lowleft : left)) : (x == last_x ? (y == 0 ? upright : (y == last_y ? lowright : right)) : (y == 0 ? upper : (y == last_y ? lower : normal))))[w->orientation]; switch (op->nopts) { case 0: done = TRUE; break; case 1: w->orientation = op->opts[0]; break; default: w->orientation = op->opts[(int) (ranf() * (double) op->nopts)]; break; } if (!done) { x += xinc[w->orientation]; y += yinc[w->orientation]; wmove(win, y, x); if (y < 0) y = 0; waddch(win, attrs); w->ypos[h] = y; w->xpos[h] = x; refs[y][x]++; } return done; } #ifdef USE_PTHREADS static bool quit_worm(int bitnum) { pending |= (1 << bitnum); napms(10); /* let the other thread(s) have a chance */ pending &= ~(1 << bitnum); return quitting; } static void * start_worm(void *arg) { unsigned long compare = 0; Trace(("start_worm")); while (!quit_worm(((struct worm *) arg) - worm)) { while (compare < sequence) { ++compare; #if HAVE_USE_WINDOW use_window(stdscr, draw_worm, arg); #else draw_worm(stdscr, arg); #endif } } Trace(("...start_worm (done)")); return NULL; } #endif static bool draw_all_worms(void) { bool done = FALSE; int n; struct worm *w; #ifdef USE_PTHREADS static bool first = TRUE; if (first) { first = FALSE; for (n = 0, w = &worm[0]; n < number; n++, w++) { int rc; rc = pthread_create(&(w->thread), NULL, start_worm, w); } } #else for (n = 0, w = &worm[0]; n < number; n++, w++) { if ( #if HAVE_USE_WINDOW USING_WINDOW2(stdscr, draw_worm, w) #else draw_worm(stdscr, w) #endif ) done = TRUE; } #endif return done; } static int get_input(void) { int ch; ch = USING_WINDOW(stdscr, wgetch); return ch; } #ifdef KEY_RESIZE static int update_refs(WINDOW *win) { int x, y; (void) win; if (last_x != COLS - 1) { for (y = 0; y <= last_y; y++) { refs[y] = typeRealloc(int, (size_t) COLS, refs[y]); if (!refs[y]) failed("update_refs"); for (x = last_x + 1; x < COLS; x++) refs[y][x] = 0; } last_x = COLS - 1; } if (last_y != LINES - 1) { for (y = LINES; y <= last_y; y++) free(refs[y]); refs = typeRealloc(int *, (size_t) LINES, refs); for (y = last_y + 1; y < LINES; y++) { refs[y] = typeMalloc(int, (size_t) COLS); if (!refs[y]) failed("update_refs"); for (x = 0; x < COLS; x++) refs[y][x] = 0; } last_y = LINES - 1; } return OK; } #endif int main(int argc, char *argv[]) { int x, y; int n; struct worm *w; int *ip; bool done = FALSE; int max_refs; setlocale(LC_ALL, ""); for (x = 1; x < argc; x++) { char *p; p = argv[x]; if (*p == '-') p++; switch (*p) { case 'f': field = "WORM"; break; case 'l': if (++x == argc) goto usage; if ((length = atoi(argv[x])) < 2 || length > MAX_LENGTH) { fprintf(stderr, "%s: Invalid length\n", *argv); ExitProgram(EXIT_FAILURE); } break; case 'n': if (++x == argc) goto usage; if ((number = atoi(argv[x])) < 1 || number > MAX_WORMS) { fprintf(stderr, "%s: Invalid number of worms\n", *argv); ExitProgram(EXIT_FAILURE); } break; case 't': trail = '.'; break; #ifdef TRACE case 'T': trace_start = atoi(argv[++x]); trace_end = atoi(argv[++x]); break; case 'N': _nc_optimize_enable ^= OPTIMIZE_ALL; /* declared by ncurses */ break; #endif /* TRACE */ default: usage: fprintf(stderr, "usage: %s [-field] [-length #] [-number #] [-trail]\n", *argv); ExitProgram(EXIT_FAILURE); } } signal(SIGINT, onsig); initscr(); noecho(); cbreak(); nonl(); curs_set(0); last_y = LINES - 1; last_x = COLS - 1; #ifdef A_COLOR if (has_colors()) { int bg = COLOR_BLACK; start_color(); #if HAVE_USE_DEFAULT_COLORS if (use_default_colors() == OK) bg = -1; #endif #define SET_COLOR(num, fg) \ init_pair(num+1, (short) fg, (short) bg); \ flavor[num] |= (chtype) COLOR_PAIR(num+1) | A_BOLD SET_COLOR(0, COLOR_GREEN); SET_COLOR(1, COLOR_RED); SET_COLOR(2, COLOR_CYAN); SET_COLOR(3, COLOR_WHITE); SET_COLOR(4, COLOR_MAGENTA); SET_COLOR(5, COLOR_BLUE); SET_COLOR(6, COLOR_YELLOW); } #endif /* A_COLOR */ max_refs = LINES; refs = typeMalloc(int *, (size_t) max_refs); for (y = 0; y < max_refs; y++) { refs[y] = typeMalloc(int, (size_t) COLS); for (x = 0; x < COLS; x++) { refs[y][x] = 0; } } #ifdef BADCORNER /* if addressing the lower right corner doesn't work in your curses */ refs[last_y][last_x] = 1; #endif /* BADCORNER */ for (n = number, w = &worm[0]; --n >= 0; w++) { w->attrs = flavor[(unsigned) n % SIZEOF(flavor)]; w->orientation = 0; w->head = 0; if (!(ip = typeMalloc(int, (size_t) (length + 1)))) { fprintf(stderr, "%s: out of memory\n", *argv); ExitProgram(EXIT_FAILURE); } w->xpos = ip; for (x = length; --x >= 0;) *ip++ = -1; if (!(ip = typeMalloc(int, (size_t) (length + 1)))) { fprintf(stderr, "%s: out of memory\n", *argv); ExitProgram(EXIT_FAILURE); } w->ypos = ip; for (y = length; --y >= 0;) *ip++ = -1; } if (field) { const char *p; p = field; for (y = last_y; --y >= 0;) { for (x = COLS; --x >= 0;) { addch((chtype) (*p++)); if (!*p) p = field; } } } USING_WINDOW(stdscr, wrefresh); nodelay(stdscr, TRUE); while (!done) { int ch; ++sequence; if ((ch = get_input()) > 0) { #ifdef TRACE if (trace_start || trace_end) { if (generation == trace_start) { trace(TRACE_CALLS); get_input(); } else if (generation == trace_end) { trace(0); get_input(); } generation++; } #endif #ifdef KEY_RESIZE if (ch == KEY_RESIZE) { USING_WINDOW(stdscr, update_refs); } #endif /* * Make it simple to put this into single-step mode, or resume * normal operation -T.Dickey */ if (ch == 'q') { quitting = TRUE; done = TRUE; continue; } else if (ch == 's') { nodelay(stdscr, FALSE); } else if (ch == ' ') { nodelay(stdscr, TRUE); } } done = draw_all_worms(); napms(10); USING_WINDOW(stdscr, wrefresh); } Trace(("Cleanup")); cleanup(); #ifdef NO_LEAKS for (y = 0; y < max_refs; y++) { free(refs[y]); } free(refs); for (n = number, w = &worm[0]; --n >= 0; w++) { free(w->xpos); free(w->ypos); } #endif #ifdef USE_PTHREADS /* * Do this just in case one of the threads did not really exit. */ Trace(("join all threads")); for (n = 0; n < number; n++) { pthread_join(worm[n].thread, NULL); } #endif ExitProgram(EXIT_SUCCESS); }
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
package | Folder | 0755 |
|
|
Makefile.in | File | 4.92 KB | 0644 |
|
README | File | 35.89 KB | 0644 |
|
aclocal.m4 | File | 108.83 KB | 0644 |
|
background.c | File | 6.84 KB | 0644 |
|
blue.c | File | 12.4 KB | 0644 |
|
bs.6 | File | 3.91 KB | 0644 |
|
bs.c | File | 29.95 KB | 0644 |
|
bulgarian-utf8.txt | File | 340 B | 0644 |
|
cardfile.c | File | 13.26 KB | 0644 |
|
cardfile.dat | File | 394 B | 0644 |
|
chgat.c | File | 9.03 KB | 0644 |
|
clip_printw.c | File | 9.1 KB | 0644 |
|
color_name.h | File | 3.32 KB | 0644 |
|
color_set.c | File | 3.25 KB | 0644 |
|
configure | File | 449.42 KB | 0644 |
|
configure.in | File | 9.57 KB | 0644 |
|
demo_altkeys.c | File | 4.77 KB | 0644 |
|
demo_defkey.c | File | 7.4 KB | 0644 |
|
demo_forms.c | File | 13.01 KB | 0644 |
|
demo_keyok.c | File | 3.24 KB | 0644 |
|
demo_menus.c | File | 18.74 KB | 0644 |
|
demo_panels.c | File | 16.89 KB | 0644 |
|
demo_termcap.c | File | 10.09 KB | 0644 |
|
demo_terminfo.c | File | 10.05 KB | 0644 |
|
ditto.c | File | 11.04 KB | 0644 |
|
dots.c | File | 4.52 KB | 0644 |
|
dots_mvcur.c | File | 4.66 KB | 0644 |
|
echochar.c | File | 4.35 KB | 0644 |
|
edit_field.c | File | 11.21 KB | 0644 |
|
edit_field.h | File | 2.57 KB | 0644 |
|
filter.c | File | 4.97 KB | 0644 |
|
firework.c | File | 5.53 KB | 0644 |
|
firstlast.c | File | 3.72 KB | 0644 |
|
foldkeys.c | File | 7.44 KB | 0644 |
|
gdc.6 | File | 3.03 KB | 0644 |
|
gdc.c | File | 8.74 KB | 0644 |
|
hanoi.c | File | 8.58 KB | 0644 |
|
hashtest.c | File | 6.58 KB | 0644 |
|
inch_wide.c | File | 7.21 KB | 0644 |
|
inchs.c | File | 7.15 KB | 0644 |
|
ins_wide.c | File | 12.25 KB | 0644 |
|
insdelln.c | File | 9.16 KB | 0644 |
|
inserts.c | File | 10.62 KB | 0644 |
|
install-sh | File | 6.96 KB | 0644 |
|
key_names.c | File | 3.22 KB | 0644 |
|
keynames.c | File | 3.07 KB | 0644 |
|
knight.c | File | 18.54 KB | 0644 |
|
linedata.h | File | 3.16 KB | 0644 |
|
linux-color.dat | File | 2.56 KB | 0644 |
|
listused.sh | File | 5.46 KB | 0644 |
|
lrtest.c | File | 5.24 KB | 0644 |
|
make-tar.sh | File | 4.8 KB | 0644 |
|
mk-test.awk | File | 4.42 KB | 0644 |
|
modules | File | 5.07 KB | 0644 |
|
movewindow.c | File | 16.44 KB | 0644 |
|
ncurses.c | File | 154.98 KB | 0644 |
|
ncurses_tst.hin | File | 2.88 KB | 0644 |
|
newdemo.c | File | 7.38 KB | 0644 |
|
programs | File | 5.28 KB | 0644 |
|
railroad.c | File | 5.66 KB | 0644 |
|
rain.c | File | 8.44 KB | 0644 |
|
redraw.c | File | 4.66 KB | 0644 |
|
savescreen.c | File | 7.49 KB | 0644 |
|
savescreen.sh | File | 2.66 KB | 0644 |
|
tclock.c | File | 5.26 KB | 0644 |
|
test.priv.h | File | 19.55 KB | 0644 |
|
test_add_wchstr.c | File | 14.08 KB | 0644 |
|
test_addchstr.c | File | 12.14 KB | 0644 |
|
test_addstr.c | File | 10.3 KB | 0644 |
|
test_addwstr.c | File | 12.37 KB | 0644 |
|
test_arrays.c | File | 3.83 KB | 0644 |
|
test_get_wstr.c | File | 8.94 KB | 0644 |
|
test_getstr.c | File | 8.88 KB | 0644 |
|
test_instr.c | File | 6.49 KB | 0644 |
|
test_inwstr.c | File | 6.69 KB | 0644 |
|
test_opaque.c | File | 10.42 KB | 0644 |
|
test_vid_puts.c | File | 3.97 KB | 0644 |
|
test_vidputs.c | File | 3.92 KB | 0644 |
|
testaddch.c | File | 3.46 KB | 0644 |
|
testcurs.c | File | 16.69 KB | 0644 |
|
testscanw.c | File | 2.88 KB | 0644 |
|
tracemunch | File | 5.01 KB | 0644 |
|
view.c | File | 13.84 KB | 0644 |
|
widechars-utf8.txt | File | 389 B | 0644 |
|
widechars.h | File | 3.19 KB | 0644 |
|
worm.c | File | 14.01 KB | 0644 |
|
xmas.c | File | 33.47 KB | 0644 |
|
xterm-16color.dat | File | 2.99 KB | 0644 |
|
xterm-256color.dat | File | 6.88 KB | 0644 |
|
xterm-88color.dat | File | 4.11 KB | 0644 |
|