2006-12-18 12:32:42 +08:00
|
|
|
/****************************************************************************
|
2010-05-02 08:59:02 +08:00
|
|
|
* Copyright (c) 1998-2009,2010 Free Software Foundation, Inc. *
|
2006-12-18 12:32:42 +08:00
|
|
|
* *
|
|
|
|
* 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. *
|
|
|
|
****************************************************************************/
|
1997-05-15 12:00:00 +08:00
|
|
|
/*
|
2010-11-14 09:43:00 +08:00
|
|
|
* $Id: rain.c,v 1.38 2010/11/13 20:11:46 tom Exp $
|
1997-05-15 12:00:00 +08:00
|
|
|
*/
|
|
|
|
#include <test.priv.h>
|
|
|
|
|
|
|
|
/* rain 11/3/1980 EPS/CITHEP */
|
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
#ifdef USE_PTHREADS
|
|
|
|
#include <pthread.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
WANT_USE_WINDOW();
|
|
|
|
|
2008-03-23 09:02:37 +08:00
|
|
|
#define MAX_THREADS 10
|
|
|
|
#define MAX_DROP 5
|
2008-03-09 08:57:07 +08:00
|
|
|
|
|
|
|
struct DATA;
|
|
|
|
|
|
|
|
typedef void (*DrawPart) (struct DATA *);
|
|
|
|
|
|
|
|
typedef struct DATA {
|
|
|
|
int y, x;
|
|
|
|
#ifdef USE_PTHREADS
|
|
|
|
DrawPart func;
|
|
|
|
int state;
|
|
|
|
#endif
|
|
|
|
} DATA;
|
|
|
|
|
2008-03-23 09:02:37 +08:00
|
|
|
#ifdef USE_PTHREADS
|
|
|
|
pthread_cond_t cond_next_drop;
|
|
|
|
pthread_mutex_t mutex_next_drop;
|
|
|
|
static int used_threads;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
pthread_t myself;
|
|
|
|
long counter;
|
|
|
|
} STATS;
|
|
|
|
|
|
|
|
static STATS drop_threads[MAX_THREADS];
|
|
|
|
#endif
|
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
static void
|
|
|
|
onsig(int n GCC_UNUSED)
|
|
|
|
{
|
|
|
|
curs_set(1);
|
|
|
|
endwin();
|
|
|
|
ExitProgram(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2010-11-14 09:43:00 +08:00
|
|
|
static double
|
2008-03-09 08:57:07 +08:00
|
|
|
ranf(void)
|
|
|
|
{
|
|
|
|
long r = (rand() & 077777);
|
2010-11-14 09:43:00 +08:00
|
|
|
return ((double) r / 32768.);
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
random_x(void)
|
|
|
|
{
|
2010-11-14 09:43:00 +08:00
|
|
|
return (int) (((double) (COLS - 4) * ranf()) + 2);
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
random_y(void)
|
|
|
|
{
|
2010-11-14 09:43:00 +08:00
|
|
|
return (int) (((double) (LINES - 4) * ranf()) + 2);
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
static int
|
|
|
|
next_j(int j)
|
1998-03-01 12:21:12 +08:00
|
|
|
{
|
2002-10-13 11:35:53 +08:00
|
|
|
if (j == 0)
|
2008-03-09 08:57:07 +08:00
|
|
|
j = MAX_DROP - 1;
|
2002-10-13 11:35:53 +08:00
|
|
|
else
|
|
|
|
--j;
|
1998-03-01 12:21:12 +08:00
|
|
|
if (has_colors()) {
|
2002-10-13 11:35:53 +08:00
|
|
|
int z = (int) (3 * ranf());
|
2010-11-14 09:43:00 +08:00
|
|
|
chtype color = (chtype) COLOR_PAIR(z);
|
1998-03-01 12:21:12 +08:00
|
|
|
if (z)
|
|
|
|
color |= A_BOLD;
|
2009-08-30 06:41:54 +08:00
|
|
|
(void) attrset(color);
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
|
|
|
return j;
|
|
|
|
}
|
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
static void
|
|
|
|
part1(DATA * drop)
|
|
|
|
{
|
2010-05-02 08:59:02 +08:00
|
|
|
MvAddCh(drop->y, drop->x, '.');
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
part2(DATA * drop)
|
|
|
|
{
|
2010-05-02 08:59:02 +08:00
|
|
|
MvAddCh(drop->y, drop->x, 'o');
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
part3(DATA * drop)
|
|
|
|
{
|
2010-05-02 08:59:02 +08:00
|
|
|
MvAddCh(drop->y, drop->x, 'O');
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
part4(DATA * drop)
|
|
|
|
{
|
2010-05-02 08:59:02 +08:00
|
|
|
MvAddCh(drop->y - 1, drop->x, '-');
|
|
|
|
MvAddStr(drop->y, drop->x - 1, "|.|");
|
|
|
|
MvAddCh(drop->y + 1, drop->x, '-');
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
part5(DATA * drop)
|
|
|
|
{
|
2010-05-02 08:59:02 +08:00
|
|
|
MvAddCh(drop->y - 2, drop->x, '-');
|
|
|
|
MvAddStr(drop->y - 1, drop->x - 1, "/ \\");
|
|
|
|
MvAddStr(drop->y, drop->x - 2, "| O |");
|
|
|
|
MvAddStr(drop->y + 1, drop->x - 1, "\\ /");
|
|
|
|
MvAddCh(drop->y + 2, drop->x, '-');
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
part6(DATA * drop)
|
|
|
|
{
|
2010-05-02 08:59:02 +08:00
|
|
|
MvAddCh(drop->y - 2, drop->x, ' ');
|
|
|
|
MvAddStr(drop->y - 1, drop->x - 1, " ");
|
|
|
|
MvAddStr(drop->y, drop->x - 2, " ");
|
|
|
|
MvAddStr(drop->y + 1, drop->x - 1, " ");
|
|
|
|
MvAddCh(drop->y + 2, drop->x, ' ');
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef USE_PTHREADS
|
|
|
|
static void
|
|
|
|
napsome(void)
|
|
|
|
{
|
|
|
|
napms(60);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2008-03-23 09:02:37 +08:00
|
|
|
* This runs inside the use_window() mutex.
|
2008-03-09 08:57:07 +08:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
really_draw(WINDOW *win, void *arg)
|
|
|
|
{
|
|
|
|
DATA *data = (DATA *) arg;
|
|
|
|
|
|
|
|
(void) win;
|
|
|
|
next_j(data->state);
|
|
|
|
data->func(data);
|
2008-03-23 09:02:37 +08:00
|
|
|
refresh();
|
2008-03-09 08:57:07 +08:00
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
draw_part(void (*func) (DATA *), int state, DATA * data)
|
|
|
|
{
|
|
|
|
data->func = func;
|
|
|
|
data->state = state;
|
|
|
|
use_window(stdscr, really_draw, (void *) data);
|
|
|
|
napsome();
|
|
|
|
}
|
|
|
|
|
2008-03-23 09:02:37 +08:00
|
|
|
/*
|
|
|
|
* Tell the threads that one of them can start work on a new raindrop.
|
|
|
|
* They may all be busy if we're sending requests too rapidly.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
put_next_drop(void)
|
|
|
|
{
|
|
|
|
pthread_cond_signal(&cond_next_drop);
|
|
|
|
pthread_mutex_unlock(&mutex_next_drop);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait until we're assigned the task of drawing a new raindrop.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
get_next_drop(void)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&mutex_next_drop);
|
|
|
|
pthread_cond_wait(&cond_next_drop, &mutex_next_drop);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
static void *
|
|
|
|
draw_drop(void *arg)
|
|
|
|
{
|
|
|
|
DATA mydata;
|
2008-03-23 09:02:37 +08:00
|
|
|
int mystats;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find myself in the list of threads so we can count the number of loops.
|
|
|
|
*/
|
|
|
|
for (mystats = 0; mystats < MAX_THREADS; ++mystats) {
|
2010-09-26 09:21:45 +08:00
|
|
|
#ifdef __MINGW32__
|
|
|
|
if (drop_threads[mystats].myself.p == pthread_self().p)
|
|
|
|
#else
|
2008-03-23 09:02:37 +08:00
|
|
|
if (drop_threads[mystats].myself == pthread_self())
|
2010-09-26 09:21:45 +08:00
|
|
|
#endif
|
2008-03-23 09:02:37 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (mystats < MAX_THREADS)
|
|
|
|
drop_threads[mystats].counter++;
|
2008-03-09 08:57:07 +08:00
|
|
|
|
2008-03-23 09:02:37 +08:00
|
|
|
/*
|
|
|
|
* Make a copy of caller's data. We're cheating for the cases after
|
|
|
|
* the first loop since we still have a pointer into the main thread
|
|
|
|
* to the data which it uses for setting up this thread (but it has
|
|
|
|
* been modified to use different coordinates).
|
|
|
|
*/
|
|
|
|
mydata = *(DATA *) arg;
|
2008-03-09 08:57:07 +08:00
|
|
|
|
2008-03-23 09:02:37 +08:00
|
|
|
draw_part(part1, 0, &mydata);
|
|
|
|
draw_part(part2, 1, &mydata);
|
|
|
|
draw_part(part3, 2, &mydata);
|
|
|
|
draw_part(part4, 3, &mydata);
|
|
|
|
draw_part(part5, 4, &mydata);
|
|
|
|
draw_part(part6, 0, &mydata);
|
|
|
|
} while (get_next_drop());
|
2008-03-09 08:57:07 +08:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
2008-03-23 09:02:37 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The description of pthread_create() is misleading, since it implies that
|
|
|
|
* threads will exit cleanly after their function returns.
|
|
|
|
*
|
|
|
|
* Since they do not (and the number of threads is limited by system
|
|
|
|
* resources), make a limited number of threads, and signal any that are
|
|
|
|
* waiting when we want a thread past that limit.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
start_drop(DATA * data)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (!used_threads) {
|
|
|
|
/* mutex and condition for signalling thread */
|
|
|
|
pthread_mutex_init(&mutex_next_drop, NULL);
|
|
|
|
pthread_cond_init(&cond_next_drop, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (used_threads < MAX_THREADS) {
|
|
|
|
rc = pthread_create(&(drop_threads[used_threads].myself),
|
|
|
|
NULL,
|
|
|
|
draw_drop,
|
|
|
|
data);
|
|
|
|
++used_threads;
|
|
|
|
} else {
|
|
|
|
rc = put_next_drop();
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
2008-03-09 08:57:07 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static int
|
|
|
|
get_input(void)
|
|
|
|
{
|
2008-03-23 09:02:37 +08:00
|
|
|
return USING_WINDOW(stdscr, wgetch);
|
2008-03-09 08:57:07 +08:00
|
|
|
}
|
|
|
|
|
1997-05-15 12:00:00 +08:00
|
|
|
int
|
2008-03-09 08:57:07 +08:00
|
|
|
main(int argc GCC_UNUSED,
|
|
|
|
char *argv[]GCC_UNUSED)
|
1997-05-15 12:00:00 +08:00
|
|
|
{
|
2008-03-23 09:02:37 +08:00
|
|
|
bool done = FALSE;
|
2008-03-09 08:57:07 +08:00
|
|
|
DATA drop;
|
2008-03-23 09:02:37 +08:00
|
|
|
#ifndef USE_PTHREADS
|
2008-03-09 08:57:07 +08:00
|
|
|
DATA last[MAX_DROP];
|
2008-03-23 09:02:37 +08:00
|
|
|
#endif
|
2008-03-09 08:57:07 +08:00
|
|
|
int j = 0;
|
2002-10-13 11:35:53 +08:00
|
|
|
|
|
|
|
setlocale(LC_ALL, "");
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2006-12-18 12:32:42 +08:00
|
|
|
CATCHALL(onsig);
|
1997-05-15 12:00:00 +08:00
|
|
|
|
|
|
|
initscr();
|
1998-03-01 12:21:12 +08:00
|
|
|
if (has_colors()) {
|
|
|
|
int bg = COLOR_BLACK;
|
|
|
|
start_color();
|
2000-10-21 12:42:11 +08:00
|
|
|
#if HAVE_USE_DEFAULT_COLORS
|
1998-03-01 12:21:12 +08:00
|
|
|
if (use_default_colors() == OK)
|
2002-10-13 11:35:53 +08:00
|
|
|
bg = -1;
|
1998-03-01 12:21:12 +08:00
|
|
|
#endif
|
2010-11-14 09:43:00 +08:00
|
|
|
init_pair(1, COLOR_BLUE, (short) bg);
|
|
|
|
init_pair(2, COLOR_CYAN, (short) bg);
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
1997-05-15 12:00:00 +08:00
|
|
|
nl();
|
|
|
|
noecho();
|
|
|
|
curs_set(0);
|
1998-03-01 12:21:12 +08:00
|
|
|
timeout(0);
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2008-03-23 09:02:37 +08:00
|
|
|
#ifndef USE_PTHREADS
|
2008-03-09 08:57:07 +08:00
|
|
|
for (j = MAX_DROP; --j >= 0;) {
|
|
|
|
last[j].x = random_x();
|
|
|
|
last[j].y = random_y();
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
2008-05-25 08:49:27 +08:00
|
|
|
j = 0;
|
2008-03-23 09:02:37 +08:00
|
|
|
#endif
|
1998-03-01 12:21:12 +08:00
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
while (!done) {
|
|
|
|
drop.x = random_x();
|
|
|
|
drop.y = random_y();
|
1998-03-01 12:21:12 +08:00
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
#ifdef USE_PTHREADS
|
2008-03-23 09:02:37 +08:00
|
|
|
if (start_drop(&drop) != 0) {
|
2008-03-09 08:57:07 +08:00
|
|
|
beep();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* The non-threaded code draws parts of each drop on each loop.
|
|
|
|
*/
|
|
|
|
part1(&drop);
|
1998-03-01 12:21:12 +08:00
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
part2(&last[j]);
|
1998-03-01 12:21:12 +08:00
|
|
|
|
|
|
|
j = next_j(j);
|
2008-03-09 08:57:07 +08:00
|
|
|
part3(&last[j]);
|
1998-03-01 12:21:12 +08:00
|
|
|
|
|
|
|
j = next_j(j);
|
2008-03-09 08:57:07 +08:00
|
|
|
part4(&last[j]);
|
1998-03-01 12:21:12 +08:00
|
|
|
|
|
|
|
j = next_j(j);
|
2008-03-09 08:57:07 +08:00
|
|
|
part5(&last[j]);
|
1998-03-01 12:21:12 +08:00
|
|
|
|
|
|
|
j = next_j(j);
|
2008-03-09 08:57:07 +08:00
|
|
|
part6(&last[j]);
|
2002-10-13 11:35:53 +08:00
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
last[j] = drop;
|
|
|
|
#endif
|
2002-10-13 11:35:53 +08:00
|
|
|
|
2008-03-09 08:57:07 +08:00
|
|
|
switch (get_input()) {
|
2002-10-13 11:35:53 +08:00
|
|
|
case ('q'):
|
|
|
|
case ('Q'):
|
2008-03-09 08:57:07 +08:00
|
|
|
done = TRUE;
|
|
|
|
break;
|
1999-10-24 12:32:42 +08:00
|
|
|
case 's':
|
|
|
|
nodelay(stdscr, FALSE);
|
|
|
|
break;
|
|
|
|
case ' ':
|
|
|
|
nodelay(stdscr, TRUE);
|
|
|
|
break;
|
1998-03-01 12:21:12 +08:00
|
|
|
#ifdef KEY_RESIZE
|
2002-10-13 11:35:53 +08:00
|
|
|
case (KEY_RESIZE):
|
1998-03-01 12:21:12 +08:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
napms(50);
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
|
|
|
curs_set(1);
|
|
|
|
endwin();
|
2008-03-23 09:02:37 +08:00
|
|
|
#ifdef USE_PTHREADS
|
|
|
|
printf("Counts per thread:\n");
|
|
|
|
for (j = 0; j < MAX_THREADS; ++j)
|
|
|
|
printf(" %d:%ld\n", j, drop_threads[j].counter);
|
|
|
|
#endif
|
2008-03-09 08:57:07 +08:00
|
|
|
ExitProgram(EXIT_SUCCESS);
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|