dgamelaunch/ttyrec.c

687 lines
18 KiB
C

/*
* Copyright (c) 1980 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. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. 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.
*/
/* 1999-02-22 Arkadiusz Mi¶kiewicz <misiek@misiek.eu.org>
* - added Native Language Support
*/
/* 2000-12-27 Satoru Takabayashi <satoru@namazu.org>
* - modify `script' to create `ttyrec'.
*/
/*
* script
*/
#include "dgamelaunch.h"
#include "config.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/time.h>
#include <sys/wait.h>
#include <fcntl.h>
#include <termios.h>
#include <time.h>
#include <signal.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#ifndef NOSTREAMS
# include <stropts.h>
#endif
#include <stdlib.h>
#include <stdint.h>
#include "ttyrec.h"
#include "io.h"
#ifndef XCASE
# define XCASE 0
#endif
static void query_encoding(int game, char *username);
int slave;
pid_t dgl_parent;
pid_t child, subchild;
pid_t input_child;
char* ipfile = NULL;
volatile int wait_for_menu = 0;
FILE *fscript;
int master;
struct termios tt;
struct winsize win;
int ancient_encoding = 0;
void
ttyrec_id(int game, char *username, char *ttyrec_filename)
{
int i;
time_t tstamp;
Header h;
char *buf = (char *)malloc(1024);
char tmpbuf[256];
if (!buf) return;
tstamp = time(NULL);
#define dCLRSCR "\033[2J"
#define dCRLF "\r\n"
snprintf(buf, 1024,
dCLRSCR "\033[1;1H" dCRLF
"Player: %s" dCRLF
"Game: %s" dCRLF
"Server: %s" dCRLF
"Filename: %s" dCRLF
"Time: (%lu) %s" dCRLF
dCLRSCR,
username,
myconfig[game]->game_name,
globalconfig.server_id,
ttyrec_filename,
tstamp, ctime(&tstamp)
);
#undef dCLRSCR
#undef dCRLF
h.len = strlen(buf);
gettimeofday (&h.tv, NULL);
(void) write_header(fscript, &h);
(void) fwrite(buf, 1, h.len, fscript);
free(buf);
}
int
ttyrec_main (int game, char *username, char *ttyrec_path, char* ttyrec_filename)
{
char dirname[100];
/* Note our PID to let children kill the main dgl process for idling */
dgl_parent = getpid();
child = subchild = input_child = 0;
if (!ttyrec_path) {
child = fork();
if (child < 0) {
perror ("fork");
fail ();
}
if (child == 0) {
execvp (myconfig[game]->game_path, myconfig[game]->bin_args);
} else {
int status;
(void) wait(&status);
}
return 0;
}
if (ttyrec_path[strlen(ttyrec_path)-1] == '/')
snprintf (dirname, 100, "%s%s", ttyrec_path, ttyrec_filename);
else
snprintf (dirname, 100, "%s/%s", ttyrec_path, ttyrec_filename);
ancient_encoding = myconfig[game]->encoding;
if (ancient_encoding == -1)
query_encoding(game, username);
atexit(&remove_ipfile);
if ((fscript = fopen (dirname, "w")) == NULL)
{
perror (dirname);
fail ();
}
setbuf (fscript, NULL);
fixtty ();
(void) signal (SIGCHLD, finish);
child = fork ();
if (child < 0)
{
perror ("fork");
fail ();
}
if (child == 0)
{
subchild = child = fork ();
if (child < 0)
{
perror ("fork");
fail ();
}
if (child)
{
close (slave);
ipfile = gen_inprogress_lock (game, child, ttyrec_filename);
ttyrec_id(game, username, ttyrec_filename);
dooutput (myconfig[game]->max_idle_time);
}
else
doshell (game, username);
}
(void) fclose (fscript);
wait_for_menu = 1;
input_child = fork();
if (input_child < 0)
{
perror ("fork2");
fail ();
}
if (!input_child)
doinput ();
else
{
while (wait_for_menu)
sleep(1);
}
remove_ipfile();
child = 0;
return 0;
}
void
doinput ()
{
register int cc;
char ibuf[BUFSIZ];
while ((cc = read (0, ibuf, BUFSIZ)) > 0)
(void) write (master, ibuf, cc);
done ();
}
void
finish (int sig)
{
int status;
register int pid;
register int die = 0;
(void)sig; /* unused */
while ((pid = wait3 (&status, WNOHANG, 0)) > 0)
{
if (pid == child)
die = 1;
}
if (die)
{
if (input_child)
{
// Need to kill the child that's writing input to pty.
kill(input_child, SIGTERM);
while ((pid = wait3(&status, WNOHANG, 0)) > 0);
}
else
done ();
}
wait_for_menu = 0;
}
void
game_idle_kill(int signal)
{
kill(child, SIGHUP);
kill(dgl_parent, SIGHUP);
}
static unsigned short charset_vt100[128] =
{
0x0000, 0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0006, 0x0007,
0x0008, 0x0009, 0x000a, 0x000b, 0x000c, 0x000d, 0x000e, 0x000f,
0x0010, 0x0011, 0x0012, 0x0013, 0x0014, 0x0015, 0x0016, 0x0017,
0x0018, 0x0019, 0x001a, 0x001b, 0x001c, 0x001d, 0x001e, 0x001f,
0x0020, 0x0021, 0x0022, 0x0023, 0x0024, 0x0025, 0x0026, 0x0027,
0x0028, 0x0029, 0x002a, 0x2192, 0x2190, 0x2191, 0x2193, 0x002f,
0x2588, 0x0031, 0x0032, 0x0033, 0x0034, 0x0035, 0x0036, 0x0037,
0x0038, 0x0039, 0x003a, 0x003b, 0x003c, 0x003d, 0x003e, 0x003f,
0x0040, 0x0041, 0x0042, 0x0043, 0x0044, 0x0045, 0x0046, 0x0047,
0x0048, 0x0049, 0x004a, 0x004b, 0x004c, 0x004d, 0x004e, 0x004f,
0x0050, 0x0051, 0x0052, 0x0053, 0x0054, 0x0055, 0x0056, 0x0057,
0x0058, 0x0059, 0x005a, 0x005b, 0x005c, 0x005d, 0x005e, 0x00a0,
#if 0
0x25c6, 0x2592, 0x2409, 0x240c, 0x240d, 0x240a, 0x00b0, 0x00b1,
0x2591, 0x240b, 0x2518, 0x2510, 0x250c, 0x2514, 0x253c, 0xf800,
0xf801, 0x2500, 0xf803, 0xf804, 0x251c, 0x2524, 0x2534, 0x252c,
0x2502, 0x2264, 0x2265, 0x03c0, 0x2260, 0x00a3, 0x00b7, 0x007f,
#endif
0x2666, 0x2592, 0x2409, 0x240c, 0x240d, 0x240a, 0x00b0, 0x00b1,
0x2424, 0x240b, 0x2518, 0x2510, 0x250c, 0x2514, 0x253c, 0x23ba,
0x23bb, 0x2500, 0x23bc, 0x23bd, 0x251c, 0x2524, 0x2534, 0x252c,
0x2502, 0x2264, 0x2265, 0x03c0, 0x2260, 0x00a3, 0x00b7, 0x0020,
};
static unsigned short charset_cp437[256] =
{
// Real IBM charset has no control codes, but they are needed by
// terminals.
0x0000, 0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0006, 0x0007,
0x0008, 0x0009, 0x000a, 0x000b, 0x000c, 0x000d, 0x000e, 0x000f,
0x0010, 0x0011, 0x0012, 0x0013, 0x0014, 0x0015, 0x0016, 0x0017,
0x0018, 0x0019, 0x001a, 0x001b, 0x001c, 0x001d, 0x001e, 0x001f,
#if 0
0x0000, 0x263a, 0x263b, 0x2665, 0x2666, 0x2663, 0x2660, 0x2022,
0x25d8, 0x25cb, 0x25d9, 0x2642, 0x2640, 0x266a, 0x266b, 0x263c,
0x25b6, 0x25c0, 0x2195, 0x203c, 0x00b6, 0x00a7, 0x25ac, 0x21a8,
0x2191, 0x2193, 0x2192, 0x2190, 0x221f, 0x2194, 0x25b2, 0x25bc,
#endif
0x0020, 0x0021, 0x0022, 0x0023, 0x0024, 0x0025, 0x0026, 0x0027,
0x0028, 0x0029, 0x002a, 0x002b, 0x002c, 0x002d, 0x002e, 0x002f,
0x0030, 0x0031, 0x0032, 0x0033, 0x0034, 0x0035, 0x0036, 0x0037,
0x0038, 0x0039, 0x003a, 0x003b, 0x003c, 0x003d, 0x003e, 0x003f,
0x0040, 0x0041, 0x0042, 0x0043, 0x0044, 0x0045, 0x0046, 0x0047,
0x0048, 0x0049, 0x004a, 0x004b, 0x004c, 0x004d, 0x004e, 0x004f,
0x0050, 0x0051, 0x0052, 0x0053, 0x0054, 0x0055, 0x0056, 0x0057,
0x0058, 0x0059, 0x005a, 0x005b, 0x005c, 0x005d, 0x005e, 0x005f,
0x0060, 0x0061, 0x0062, 0x0063, 0x0064, 0x0065, 0x0066, 0x0067,
0x0068, 0x0069, 0x006a, 0x006b, 0x006c, 0x006d, 0x006e, 0x006f,
0x0070, 0x0071, 0x0072, 0x0073, 0x0074, 0x0075, 0x0076, 0x0077,
0x0078, 0x0079, 0x007a, 0x007b, 0x007c, 0x007d, 0x007e, 0x2302,
0x00c7, 0x00fc, 0x00e9, 0x00e2, 0x00e4, 0x00e0, 0x00e5, 0x00e7,
0x00ea, 0x00eb, 0x00e8, 0x00ef, 0x00ee, 0x00ec, 0x00c4, 0x00c5,
0x00c9, 0x00e6, 0x00c6, 0x00f4, 0x00f6, 0x00f2, 0x00fb, 0x00f9,
0x00ff, 0x00d6, 0x00dc, 0x00a2, 0x00a3, 0x00a5, 0x20a7, 0x0192,
0x00e1, 0x00ed, 0x00f3, 0x00fa, 0x00f1, 0x00d1, 0x00aa, 0x00ba,
0x00bf, 0x2310, 0x00ac, 0x00bd, 0x00bc, 0x00a1, 0x00ab, 0x00bb,
0x2591, 0x2592, 0x2593, 0x2502, 0x2524, 0x2561, 0x2562, 0x2556,
0x2555, 0x2563, 0x2551, 0x2557, 0x255d, 0x255c, 0x255b, 0x2510,
0x2514, 0x2534, 0x252c, 0x251c, 0x2500, 0x253c, 0x255e, 0x255f,
0x255a, 0x2554, 0x2569, 0x2566, 0x2560, 0x2550, 0x256c, 0x2567,
0x2568, 0x2564, 0x2565, 0x2559, 0x2558, 0x2552, 0x2553, 0x256b,
0x256a, 0x2518, 0x250c, 0x2588, 0x2584, 0x258c, 0x2590, 0x2580,
0x03b1, 0x00df, 0x0393, 0x03c0, 0x03a3, 0x03c3, 0x00b5, 0x03c4,
0x03a6, 0x0398, 0x03a9, 0x03b4, 0x221e, 0x03c6, 0x03b5, 0x2229,
0x2261, 0x00b1, 0x2265, 0x2264, 0x2320, 0x2321, 0x00f7, 0x2248,
0x00b0, 0x2219, 0x00b7, 0x221a, 0x207f, 0x00b2, 0x25a0, 0x00a0,
};
// there must be at least 4 bytes free, NOT CHECKED!
static int wctoutf8(char *d, uint32_t s)
{
if (s < 0x80)
{
d[0] = s;
return 1;
}
if (s < 0x800)
{
d[0] = ( s >> 6) | 0xc0;
d[1] = ( s & 0x3f) | 0x80;
return 2;
}
if (s < 0x10000)
{
d[0] = ( s >> 12) | 0xe0;
d[1] = ((s >> 6) & 0x3f) | 0x80;
d[2] = ( s & 0x3f) | 0x80;
return 3;
}
if (s < 0x110000)
{
d[0] = ( s >> 18) | 0xf0;
d[1] = ((s >> 12) & 0x3f) | 0x80;
d[2] = ((s >> 6) & 0x3f) | 0x80;
d[3] = ( s & 0x3f) | 0x80;
return 4;
}
// Invalid char marker (U+FFFD).
d[0] = 0xef;
d[1] = 0xbf;
d[2] = 0xbd;
return 3;
}
void
dooutput (int max_idle_time)
{
int cc, i, len;
time_t tvec, time ();
char obuf[BUFSIZ], ubuf[BUFSIZ*4+2], *ctime (), *out;
int galt = 0; // vt100 G switch
setbuf (stdout, NULL);
(void) close (0);
tvec = time ((time_t *) NULL);
/* Set up SIGALRM handler to kill idle games */
signal(SIGALRM, game_idle_kill);
for (;;)
{
Header h;
cc = read (master, obuf, BUFSIZ);
if (cc <= 0)
break;
if (max_idle_time)
alarm(max_idle_time);
switch (ancient_encoding)
{
case 0: // UTF-8
default:
h.len = cc;
gettimeofday (&h.tv, NULL);
(void) write (1, obuf, cc);
(void) write_header (fscript, &h);
(void) fwrite (obuf, 1, cc, fscript);
break;
case 1: // IBM
out = ubuf;
// Old Crawl emits useless toggles of vt100 mode, even though it
// never uses them in this mode. And they break stuff...
for (i = 0; i < cc; i++)
{
if (galt == 2) // ignore "ESC ( 0", "ESC ( B" and anything such
{
galt = 0;
continue;
}
else if (galt == 1)
if (obuf[i] == '(')
{
galt = 2;
continue;
}
else
{
galt = 0; // false alarm, emit ESC and continue
*out ++ = 27;
}
else if (obuf[i] == 27)
{
galt = 1;
continue;
}
else if (obuf[i] == 14 || obuf[i] == 15)
continue;
out += wctoutf8(out, charset_cp437[(unsigned char)obuf[i]]);
}
h.len = len = out - ubuf;
gettimeofday(&h.tv, NULL);
write(1, ubuf, len);
write_header(fscript, &h);
fwrite(ubuf, 1, len, fscript);
break;
case 2: // DEC
out = ubuf;
for (i = 0; i < cc; i++)
{
if (obuf[i] == 14)
galt = 1;
else if (obuf[i] == 15)
galt = 0;
else if (obuf[i] & 0x80) // strictly 7-bit
out += wctoutf8(out, 0xFFFD); // or we could assume some other charset
else if (galt)
out += wctoutf8(out, charset_vt100[(int)obuf[i]]);
else
*out++ = obuf[i];
}
h.len = len = out - ubuf;
gettimeofday(&h.tv, NULL);
write(1, ubuf, len);
write_header(fscript, &h);
fwrite(ubuf, 1, len, fscript);
break;
}
}
done ();
}
void
doshell (int game, char *username)
{
getslave ();
(void) close (master);
(void) fclose (fscript);
(void) dup2 (slave, 0);
(void) dup2 (slave, 1);
(void) dup2 (slave, 2);
(void) close (slave);
/*
if (myconfig[game]->mkdir)
(void) mkdir(myconfig[game]->mkdir, 0755);
if (myconfig[game]->chdir)
(void) chdir(myconfig[game]->chdir);
*/
execvp (myconfig[game]->game_path, myconfig[game]->bin_args);
fail ();
}
void
fixtty ()
{
struct termios rtt;
rtt = tt;
rtt.c_iflag = 0;
rtt.c_lflag &= ~(ISIG | ICANON | XCASE | ECHO | ECHOE | ECHOK | ECHONL);
rtt.c_oflag = OPOST;
rtt.c_cc[VINTR] = _POSIX_VDISABLE;
rtt.c_cc[VQUIT] = _POSIX_VDISABLE;
rtt.c_cc[VERASE] = _POSIX_VDISABLE;
rtt.c_cc[VKILL] = _POSIX_VDISABLE;
rtt.c_cc[VEOF] = 1;
rtt.c_cc[VEOL] = 0;
rtt.c_cc[VMIN] = 1;
rtt.c_cc[VTIME] = 0;
(void) tcsetattr (0, TCSAFLUSH, &rtt);
}
void
fail ()
{
(void) kill (0, SIGTERM);
done ();
}
void
done ()
{
time_t tvec, time ();
char *ctime ();
if (subchild)
{
tvec = time ((time_t *) NULL);
(void) fclose (fscript);
(void) close (master);
}
else
{
(void) tcsetattr (0, TCSAFLUSH, &tt);
}
remove_ipfile();
graceful_exit (0);
}
void
getslave ()
{
(void) setsid ();
/* grantpt( master);
unlockpt(master);
if ((slave = open((const char *)ptsname(master), O_RDWR)) < 0) {
perror((const char *)ptsname(master));
fail();
perror("open(fd, O_RDWR)");
fail();
} */
#ifndef NOSTREAMS
if (isastream (slave))
{
if (ioctl (slave, I_PUSH, "ptem") < 0)
{
perror ("ioctl(fd, I_PUSH, ptem)");
fail ();
}
if (ioctl (slave, I_PUSH, "ldterm") < 0)
{
perror ("ioctl(fd, I_PUSH, ldterm)");
fail ();
}
#ifndef _HPUX_SOURCE
if (ioctl (slave, I_PUSH, "ttcompat") < 0)
{
perror ("ioctl(fd, I_PUSH, ttcompat)");
fail ();
}
#endif
}
#endif /* !NOSTREAMS */
(void) ioctl (0, TIOCGWINSZ, (char *) &win);
}
void
remove_ipfile (void)
{
if (ipfile != NULL) {
unlink (ipfile);
free(ipfile);
ipfile = NULL;
}
signal(SIGALRM, SIG_IGN);
}
int encoding_by_name(const char *enc)
{
if (!strcasecmp(enc, "UTF-8") || !strcasecmp(enc, "UNICODE"))
return 0;
if (!strcasecmp(enc, "IBM") || !strcasecmp(enc, "CP437"))
return 1;
if (!strcasecmp(enc, "DEC"))
return 2;
if (!strcasecmp(enc, "ASCII"))
return 1; // what to do with invalid chars?
return -1;
}
static void call_print_charset(char *exe, char **args)
{
char **args2, **a;
int nargs = 0;
for (args2 = args; *args2; args2++)
nargs++;
args2 = calloc(nargs + 1, sizeof(char*));
for (a = args2; *args; args++)
*a++ = *args;
*a++ = "--print-charset";
*a = 0;
execvp(exe, args2);
}
static void query_encoding(int game, char *username)
{
int son;
int p[2];
int null;
struct timeval tv;
fd_set se;
char buf[128];
if (pipe(p))
perror("pipe");
switch(son = fork())
{
case -1:
perror("fork");
fail();
case 0:
null = open("/dev/null", O_RDONLY);
if (null != -1)
dup2(null, 0), close(null);
else
{
fprintf(stderr, "Error: can't open /dev/null\n");
// non-fatal, but if the child opens some other file, it might
// get confused
close(0);
}
dup2(p[1], 1);
close(p[1]);
close(p[0]);
call_print_charset(myconfig[game]->game_path, myconfig[game]->bin_args);
exit(1);
}
close(p[1]);
FD_ZERO(&se);
FD_SET(p[0],&se);
tv.tv_sec = 60; // FIXME: a huge delay, in case there's a db rebuild
tv.tv_usec = 0;
if (select(p[0]+1,&se,NULL,NULL,&tv) != 1)
{
fprintf(stderr, "Error: can't obtain charset info.\nPress any key...\n");
read(0, buf, 1);
close(p[0]); // SIGPIPE
kill(son, SIGTERM); // and SIGTERM for a good measure
waitpid(son, 0, 0);
ancient_encoding = 0;
return;
}
// Sending _one_ short message over a pipe is atomic on all systems I know.
// Don't assume this on about any other file descriptor.
read(p[0], buf, sizeof(buf)-1);
buf[sizeof(buf)-1] = 0;
close(p[0]);
kill(son, SIGTERM);
waitpid(son, 0, 0);
if (strchr(buf, '\n'))
*strchr(buf, '\n') = 0;
ancient_encoding = encoding_by_name(buf);
if (ancient_encoding == -1)
{
fprintf(stderr, "Error: unknown encoding \"%s\"\nPress any key...\n", buf);
read(0, buf, 1);
}
}