NetBSD/games/hunt/huntd/answer.c

436 lines
10 KiB
C
Raw Normal View History

2009-08-27 04:36:32 +04:00
/* $NetBSD: answer.c,v 1.16 2009/08/27 00:36:32 dholland Exp $ */
1997-10-04 13:00:13 +04:00
/*
2003-06-11 16:00:21 +04:00
* Copyright (c) 1983-2003, 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:
*
* + Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* + 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.
* + Neither the name of the University of California, San Francisco 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 COPYRIGHT HOLDERS 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 COPYRIGHT
* OWNER 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.
1997-10-04 13:00:13 +04:00
*/
#include <sys/cdefs.h>
#ifndef lint
2009-08-27 04:36:32 +04:00
__RCSID("$NetBSD: answer.c,v 1.16 2009/08/27 00:36:32 dholland Exp $");
#endif /* not lint */
2009-07-04 08:29:54 +04:00
#include <ctype.h>
#include <errno.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include "hunt.h"
1997-10-04 13:00:13 +04:00
2009-07-04 08:29:54 +04:00
#define SCOREDECAY 15
1997-10-04 13:00:13 +04:00
2009-07-04 08:29:54 +04:00
static char Ttyname[NAMELEN];
1997-10-04 13:00:13 +04:00
static IDENT *get_ident(uint32_t, uint32_t, char *, char);
static void stmonitor(PLAYER *);
static void stplayer(PLAYER *, int);
int
2009-07-04 05:58:57 +04:00
answer(void)
1997-10-04 13:00:13 +04:00
{
2009-07-04 08:29:54 +04:00
PLAYER *pp;
int newsock;
static u_long mode;
static char name[NAMELEN];
static char team;
static int enter_status;
static socklen_t socklen;
static uint32_t machine;
static uint32_t uid;
static SOCKET sockstruct;
char *cp1, *cp2;
int flags;
2009-08-27 04:36:32 +04:00
uint32_t version;
2009-07-04 08:29:54 +04:00
int i;
#ifdef INTERNET
1997-10-04 13:00:13 +04:00
socklen = sizeof sockstruct;
2009-07-04 08:29:54 +04:00
#else
1997-10-04 13:00:13 +04:00
socklen = sizeof sockstruct - 1;
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
errno = 0;
newsock = accept(Socket, (struct sockaddr *) &sockstruct, &socklen);
if (newsock < 0)
{
if (errno == EINTR)
return FALSE;
2009-07-04 08:29:54 +04:00
#ifdef LOG
1997-10-04 13:00:13 +04:00
syslog(LOG_ERR, "accept: %m");
2009-07-04 08:29:54 +04:00
#else
1997-10-04 13:00:13 +04:00
perror("accept");
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
cleanup(1);
}
2009-07-04 08:29:54 +04:00
#ifdef INTERNET
1997-10-04 13:00:13 +04:00
machine = ntohl(((struct sockaddr_in *) &sockstruct)->sin_addr.s_addr);
2009-07-04 08:29:54 +04:00
#else
1997-10-04 13:00:13 +04:00
if (machine == 0)
machine = gethostid();
2009-07-04 08:29:54 +04:00
#endif
2009-08-27 04:36:32 +04:00
version = htonl((uint32_t) HUNT_VERSION);
(void) write(newsock, &version, LONGLEN);
(void) read(newsock, &uid, LONGLEN);
2008-01-28 06:23:29 +03:00
uid = ntohl(uid);
1997-10-04 13:00:13 +04:00
(void) read(newsock, name, NAMELEN);
(void) read(newsock, &team, 1);
(void) read(newsock, &enter_status, LONGLEN);
1997-10-04 13:00:13 +04:00
enter_status = ntohl((unsigned long) enter_status);
(void) read(newsock, Ttyname, NAMELEN);
(void) read(newsock, &mode, sizeof mode);
1997-10-04 13:00:13 +04:00
mode = ntohl(mode);
/*
* Ensure null termination.
*/
name[sizeof(name)-1] = '\0';
Ttyname[sizeof(Ttyname)-1] = '\0';
1997-10-04 13:00:13 +04:00
/*
* Turn off blocking I/O, so a slow or dead terminal won't stop
* the game. All subsequent reads check how many bytes they read.
*/
flags = fcntl(newsock, F_GETFL, 0);
flags |= O_NDELAY;
(void) fcntl(newsock, F_SETFL, flags);
/*
* Make sure the name contains only printable characters
* since we use control characters for cursor control
* between driver and player processes
*/
for (cp1 = cp2 = name; *cp1 != '\0'; cp1++)
if (isprint((unsigned char)*cp1) || *cp1 == ' ')
1997-10-04 13:00:13 +04:00
*cp2++ = *cp1;
*cp2 = '\0';
2009-07-04 08:29:54 +04:00
#ifdef INTERNET
1997-10-04 13:00:13 +04:00
if (mode == C_MESSAGE) {
char buf[BUFSIZ + 1];
int n;
if (team == ' ')
2009-06-29 01:12:10 +04:00
(void) snprintf(buf, sizeof(buf), "%s: ", name);
1997-10-04 13:00:13 +04:00
else
2009-06-29 01:12:10 +04:00
(void) snprintf(buf, sizeof(buf), "%s[%c]: ", name,
team);
1997-10-04 13:00:13 +04:00
n = strlen(buf);
for (pp = Player; pp < End_player; pp++) {
cgoto(pp, HEIGHT, 0);
outstr(pp, buf, n);
}
while ((n = read(newsock, buf, BUFSIZ)) > 0)
for (pp = Player; pp < End_player; pp++)
outstr(pp, buf, n);
for (pp = Player; pp < End_player; pp++) {
ce(pp);
sendcom(pp, REFRESH);
sendcom(pp, READY, 0);
(void) fflush(pp->p_output);
}
(void) close(newsock);
return FALSE;
}
else
2009-07-04 08:29:54 +04:00
#endif
#ifdef MONITOR
1997-10-04 13:00:13 +04:00
if (mode == C_MONITOR)
if (End_monitor < &Monitor[MAXMON]) {
1997-10-04 13:00:13 +04:00
pp = End_monitor++;
i = pp - Monitor + MAXPL + 3;
} else {
1997-10-04 13:00:13 +04:00
socklen = 0;
(void) write(newsock, &socklen,
1997-10-04 13:00:13 +04:00
sizeof socklen);
(void) close(newsock);
return FALSE;
}
else
2009-07-04 08:29:54 +04:00
#endif
if (End_player < &Player[MAXPL]) {
1997-10-04 13:00:13 +04:00
pp = End_player++;
i = pp - Player + 3;
} else {
1997-10-04 13:00:13 +04:00
socklen = 0;
(void) write(newsock, &socklen,
1997-10-04 13:00:13 +04:00
sizeof socklen);
(void) close(newsock);
return FALSE;
}
#ifdef MONITOR
if (mode == C_MONITOR && team == ' ')
team = '*';
#endif
pp->p_ident = get_ident(machine, uid, name, team);
pp->p_output = fdopen(newsock, "w");
pp->p_death[0] = '\0';
pp->p_fd = newsock;
fdset[i].fd = newsock;
fdset[i].events = POLLIN;
1997-10-04 13:00:13 +04:00
pp->p_y = 0;
pp->p_x = 0;
2009-07-04 08:29:54 +04:00
#ifdef MONITOR
1997-10-04 13:00:13 +04:00
if (mode == C_MONITOR)
stmonitor(pp);
else
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
stplayer(pp, enter_status);
return TRUE;
}
2009-07-04 08:29:54 +04:00
#ifdef MONITOR
static void
2009-07-04 05:58:57 +04:00
stmonitor(PLAYER *pp)
1997-10-04 13:00:13 +04:00
{
2009-07-04 08:29:54 +04:00
int line;
PLAYER *npp;
1997-10-04 13:00:13 +04:00
memcpy(pp->p_maze, Maze, sizeof Maze);
drawmaze(pp);
2009-06-29 01:12:10 +04:00
(void) snprintf(Buf, sizeof(Buf), "%5.5s%c%-10.10s %c", " ",
stat_char(pp),
1997-10-04 13:00:13 +04:00
pp->p_ident->i_name, pp->p_ident->i_team);
line = STAT_MON_ROW + 1 + (pp - Monitor);
for (npp = Player; npp < End_player; npp++) {
cgoto(npp, line, STAT_NAME_COL);
outstr(npp, Buf, STAT_NAME_LEN);
}
for (npp = Monitor; npp < End_monitor; npp++) {
cgoto(npp, line, STAT_NAME_COL);
outstr(npp, Buf, STAT_NAME_LEN);
}
sendcom(pp, REFRESH);
sendcom(pp, READY, 0);
(void) fflush(pp->p_output);
}
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
static void
2009-07-04 05:58:57 +04:00
stplayer(PLAYER *newpp, int enter_status)
1997-10-04 13:00:13 +04:00
{
2009-07-04 08:29:54 +04:00
int x, y;
PLAYER *pp;
1997-10-04 13:00:13 +04:00
Nplayer++;
for (y = 0; y < UBOUND; y++)
for (x = 0; x < WIDTH; x++)
newpp->p_maze[y][x] = Maze[y][x];
for ( ; y < DBOUND; y++) {
for (x = 0; x < LBOUND; x++)
newpp->p_maze[y][x] = Maze[y][x];
for ( ; x < RBOUND; x++)
newpp->p_maze[y][x] = SPACE;
for ( ; x < WIDTH; x++)
newpp->p_maze[y][x] = Maze[y][x];
}
for ( ; y < HEIGHT; y++)
for (x = 0; x < WIDTH; x++)
newpp->p_maze[y][x] = Maze[y][x];
do {
x = rand_num(WIDTH - 1) + 1;
y = rand_num(HEIGHT - 1) + 1;
} while (Maze[y][x] != SPACE);
newpp->p_over = SPACE;
newpp->p_x = x;
newpp->p_y = y;
newpp->p_undershot = FALSE;
2009-07-04 08:29:54 +04:00
#ifdef FLY
1997-10-04 13:00:13 +04:00
if (enter_status == Q_FLY) {
newpp->p_flying = rand_num(20);
newpp->p_flyx = 2 * rand_num(6) - 5;
newpp->p_flyy = 2 * rand_num(6) - 5;
newpp->p_face = FLYER;
}
else
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
{
newpp->p_flying = -1;
newpp->p_face = rand_dir();
}
newpp->p_damage = 0;
newpp->p_damcap = MAXDAM;
newpp->p_nchar = 0;
newpp->p_ncount = 0;
newpp->p_nexec = 0;
newpp->p_ammo = ISHOTS;
2009-07-04 08:29:54 +04:00
#ifdef BOOTS
1997-10-04 13:00:13 +04:00
newpp->p_nboots = 0;
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
if (enter_status == Q_SCAN) {
newpp->p_scan = SCANLEN;
newpp->p_cloak = 0;
}
else {
newpp->p_scan = 0;
newpp->p_cloak = CLOAKLEN;
}
newpp->p_ncshot = 0;
do {
x = rand_num(WIDTH - 1) + 1;
y = rand_num(HEIGHT - 1) + 1;
} while (Maze[y][x] != SPACE);
Maze[y][x] = GMINE;
2009-07-04 08:29:54 +04:00
#ifdef MONITOR
1997-10-04 13:00:13 +04:00
for (pp = Monitor; pp < End_monitor; pp++)
check(pp, y, x);
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
do {
x = rand_num(WIDTH - 1) + 1;
y = rand_num(HEIGHT - 1) + 1;
} while (Maze[y][x] != SPACE);
Maze[y][x] = MINE;
2009-07-04 08:29:54 +04:00
#ifdef MONITOR
1997-10-04 13:00:13 +04:00
for (pp = Monitor; pp < End_monitor; pp++)
check(pp, y, x);
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
2009-06-29 01:12:10 +04:00
(void) snprintf(Buf, sizeof(Buf), "%5.2f%c%-10.10s %c",
newpp->p_ident->i_score,
1997-10-04 13:00:13 +04:00
stat_char(newpp), newpp->p_ident->i_name,
newpp->p_ident->i_team);
y = STAT_PLAY_ROW + 1 + (newpp - Player);
for (pp = Player; pp < End_player; pp++) {
if (pp != newpp) {
char smallbuf[16];
1997-10-04 13:00:13 +04:00
pp->p_ammo += NSHOTS;
newpp->p_ammo += NSHOTS;
cgoto(pp, y, STAT_NAME_COL);
outstr(pp, Buf, STAT_NAME_LEN);
2009-06-29 01:12:10 +04:00
(void) snprintf(smallbuf, sizeof(smallbuf),
"%3d", pp->p_ammo);
1997-10-04 13:00:13 +04:00
cgoto(pp, STAT_AMMO_ROW, STAT_VALUE_COL);
outstr(pp, smallbuf, 3);
}
}
2009-07-04 08:29:54 +04:00
#ifdef MONITOR
1997-10-04 13:00:13 +04:00
for (pp = Monitor; pp < End_monitor; pp++) {
cgoto(pp, y, STAT_NAME_COL);
outstr(pp, Buf, STAT_NAME_LEN);
}
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
drawmaze(newpp);
drawplayer(newpp, TRUE);
look(newpp);
2009-07-04 08:29:54 +04:00
#ifdef FLY
1997-10-04 13:00:13 +04:00
if (enter_status == Q_FLY)
/* Make sure that the position you enter in will be erased */
showexpl(newpp->p_y, newpp->p_x, FLYER);
2009-07-04 08:29:54 +04:00
#endif
1997-10-04 13:00:13 +04:00
sendcom(newpp, REFRESH);
sendcom(newpp, READY, 0);
(void) fflush(newpp->p_output);
}
/*
* rand_dir:
* Return a random direction
*/
int
2009-07-04 05:58:57 +04:00
rand_dir(void)
1997-10-04 13:00:13 +04:00
{
switch (rand_num(4)) {
case 0:
return LEFTS;
case 1:
return RIGHT;
case 2:
return BELOW;
case 3:
return ABOVE;
}
/* NOTREACHED */
return(-1);
1997-10-04 13:00:13 +04:00
}
/*
* get_ident:
* Get the score structure of a player
*/
static IDENT *
2009-07-04 05:58:57 +04:00
get_ident(uint32_t machine, uint32_t uid, char *name, char team)
1997-10-04 13:00:13 +04:00
{
2009-07-04 08:29:54 +04:00
IDENT *ip;
static IDENT punt;
1997-10-04 13:00:13 +04:00
for (ip = Scores; ip != NULL; ip = ip->i_next)
if (ip->i_machine == machine
&& ip->i_uid == uid
&& ip->i_team == team
&& strncmp(ip->i_name, name, NAMELEN) == 0)
break;
if (ip != NULL) {
if (ip->i_entries < SCOREDECAY)
ip->i_entries++;
else
ip->i_kills = (ip->i_kills * (SCOREDECAY - 1))
/ SCOREDECAY;
ip->i_score = ip->i_kills / (double) ip->i_entries;
}
else {
ip = malloc(sizeof(*ip));
1997-10-04 13:00:13 +04:00
if (ip == NULL) {
/* Fourth down, time to punt */
ip = &punt;
}
ip->i_machine = machine;
ip->i_team = team;
ip->i_uid = uid;
strncpy(ip->i_name, name, NAMELEN);
ip->i_kills = 0;
ip->i_entries = 1;
ip->i_score = 0;
ip->i_absorbed = 0;
ip->i_faced = 0;
ip->i_shot = 0;
ip->i_robbed = 0;
ip->i_slime = 0;
ip->i_missed = 0;
ip->i_ducked = 0;
ip->i_gkills = ip->i_bkills = ip->i_deaths = 0;
ip->i_stillb = ip->i_saved = 0;
ip->i_next = Scores;
Scores = ip;
}
return ip;
}