dgamelaunch/ttyplay.c

564 lines
13 KiB
C

/*
* Copyright (c) 2000 Satoru Takabayashi <satoru@namazu.org>
* 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.
*/
#include "config.h"
#define _GNU_SOURCE /* need sighandler_t */
#include <sys/types.h>
#include <sys/time.h>
#include <sys/stat.h>
#ifdef HAVE_KQUEUE
#include <sys/event.h>
#endif
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <unistd.h>
#include <termios.h>
#include <string.h>
#include <curses.h>
#include <signal.h>
#include <errno.h>
#include "dgamelaunch.h"
#include "ttyplay.h"
#include "ttyrec.h"
#include "io.h"
#include "stripgfx.h"
#ifdef __MACH__
typedef void (*sighandler_t)(int);
#endif
int stripped = NO_GRAPHICS;
static int got_sigwinch = 0;
static int term_resizex = -1;
static int term_resizey = -1;
void
ttyplay_sigwinch_func(int sig)
{
signal(SIGWINCH, ttyplay_sigwinch_func);
got_sigwinch = 1;
}
struct timeval
timeval_diff (struct timeval tv1, struct timeval tv2)
{
struct timeval diff;
diff.tv_sec = tv2.tv_sec - tv1.tv_sec;
diff.tv_usec = tv2.tv_usec - tv1.tv_usec;
if (diff.tv_usec < 0)
{
diff.tv_sec--;
diff.tv_usec += 1000000;
}
return diff;
}
struct timeval
timeval_div (struct timeval tv1, double n)
{
double x = ((double) tv1.tv_sec + (double) tv1.tv_usec / 1000000.0) / n;
struct timeval div;
div.tv_sec = (int) x;
div.tv_usec = (x - (int) x) * 1000000;
return div;
}
double
ttywait (struct timeval prev, struct timeval cur, double speed)
{
struct timeval diff = timeval_diff (prev, cur);
assert (speed != 0);
diff = timeval_div (diff, speed);
select (1, NULL, NULL, NULL, &diff); /* skip if a user hits any key */
return speed;
}
double
ttynowait (struct timeval prev, struct timeval cur, double speed)
{
return 0; /* Speed isn't important. */
}
int
kbhit(void)
{
int i = 0;
nodelay(stdscr, TRUE);
timeout(0);
i = wgetch(stdscr);
nodelay(stdscr, FALSE);
if (i == -1)
i = 0;
else
ungetch(i);
return (i);
}
int
ttyplay_keyboard_action(int c)
{
struct termios t;
switch (c)
{
case ERR:
case 'q':
return READ_QUIT;
case 'r':
if (term_resizex > 0 && term_resizey > 0) {
printf ("\033[8;%d;%dt", term_resizey, term_resizex);
return READ_RESTART;
}
break;
case 's':
switch (stripped)
{
case NO_GRAPHICS: populate_gfx_array ((stripped = DEC_GRAPHICS)); break;
case DEC_GRAPHICS: populate_gfx_array ((stripped = IBM_GRAPHICS)); break;
case IBM_GRAPHICS: populate_gfx_array ((stripped = NO_GRAPHICS)); break;
}
return READ_RESTART;
case 'm':
tcgetattr (0, &t);
if (!loggedin)
{
initcurses();
loginprompt(1);
}
if (loggedin)
{
initcurses ();
domailuser (chosen_name);
}
endwin ();
tcsetattr (0, TCSANOW, &t);
return READ_RESTART;
case '?':
tcgetattr (0, &t);
initcurses();
(void) runmenuloop(dgl_find_menu("watchmenu_help"));
endwin ();
tcsetattr (0, TCSANOW, &t);
return READ_RESTART;
}
return (READ_DATA);
}
int
ttyread (FILE * fp, Header * h, char **buf, int pread)
{
long offset;
int kb = kbhit();
if (kb == ERR) return READ_QUIT;
else if (kb) {
const int c = dgl_getch();
const int action = ttyplay_keyboard_action(c);
if (action != READ_DATA)
return (action);
}
/* do this BEFORE header read, hlen bug */
offset = ftell (fp);
if (read_header (fp, h) == 0)
{
return READ_EOF;
}
/* length should never be longer than one BUFSIZ */
if (h->len > BUFSIZ)
{
fprintf (stderr, "h->len too big (%ld) limit %ld\n",
(long)h->len, (long)BUFSIZ);
return READ_QUIT;
}
*buf = malloc (h->len + 1);
if (*buf == NULL)
{
perror ("malloc");
return READ_QUIT;
}
if (fread (*buf, 1, h->len, fp) != h->len)
{
fseek (fp, offset, SEEK_SET);
return READ_EOF;
}
(*buf)[h->len] = 0;
return READ_DATA;
}
int
ttypread (FILE * fp, Header * h, char **buf, int pread)
{
int n;
#ifdef HAVE_KQUEUE
struct kevent evt[2];
static int kq = -1;
#endif
struct timeval w = { 0, 100000 };
struct timeval origw = { 0, 100000 };
int counter = 0;
fd_set readfs;
int doread = 0;
int action = READ_DATA;
#ifdef HAVE_KQUEUE
if (kq == -1)
kq = kqueue ();
if (kq == -1)
{
printf ("kqueue() failed.\n");
return READ_QUIT;
}
#endif
/*
* Read persistently just like tail -f.
*/
while ((action = ttyread (fp, h, buf, 1)) == READ_EOF)
{
idle_alarm_reset();
fflush(stdout);
clearerr (fp);
#ifdef HAVE_KQUEUE
n = -1;
if (kq != -2)
{
EV_SET (&evt[0], STDIN_FILENO, EVFILT_READ, EV_ADD | EV_ONESHOT, 0, 0, NULL);
EV_SET (&evt[1], fileno (fp), EVFILT_READ, EV_ADD | EV_ONESHOT, 0, 0, NULL);
n = kevent (kq, evt, 2, evt, 1, NULL);
doread = (n >= 1 && evt[0].ident == STDIN_FILENO &&
evt[0].filter == EVFILT_READ) ||
(n >= 2 && evt[1].ident == STDIN_FILENO &&
evt[1].filter == EVFILT_READ);
if (n == -1)
{
/*
* Perhaps kevent(2) doesn't work on this fstype,
* use select(2) instead. Never use kevent again, assuming all
* active ttyrecs are on the same fstype.
*/
close(kq);
kq = -2;
}
}
if (n == -1)
#endif
{
if (counter++ > (20 * 60 * 10))
{
/*
* The reason for this timeout is that the select() method uses
* some CPU in waiting. The kqueue() method does not do that, so it
* does not need the timeout.
*/
endwin ();
printf ("Exiting due to 20 minutes of inactivity.\n");
return READ_QUIT;
}
FD_ZERO (&readfs);
FD_SET (STDIN_FILENO, &readfs);
n = select (1, &readfs, NULL, NULL, &w);
w = origw;
doread = n >= 1 && FD_ISSET (0, &readfs);
}
if (n == -1)
{
if ((errno == EINTR) && got_sigwinch) {
got_sigwinch = 0;
return READ_RESTART;
} else {
printf("select()/kevent() failed.\n");
return READ_QUIT;
}
}
if (doread)
{ /* user hits a character? */
const int c = dgl_getch();
action = ttyplay_keyboard_action(c);
if (action != READ_DATA)
return action;
}
}
return (action);
}
void
ttywrite (char *buf, int len)
{
int i;
for (i = 0; i < len; i++)
{
if (stripped != NO_GRAPHICS)
buf[i] = strip_gfx (buf[i]);
}
fwrite (buf, 1, len, stdout);
}
void
ttynowrite (char *buf, int len)
{
/* do nothing */
}
int
ttyplay (FILE * fp, double speed, ReadFunc read_func,
WriteFunc write_func, WaitFunc wait_func, off_t offset)
{
int first_time = 1;
int r = READ_EOF;
struct timeval prev;
/* for dtype's attempt to get the last clrscr and playback from there */
if (offset != -1)
{
fseek (fp, offset, SEEK_SET);
}
while (1)
{
char *buf;
Header h;
r = read_func (fp, &h, &buf, 0);
if (r != READ_DATA)
{
break;
}
if (!first_time)
{
speed = wait_func (prev, h.tv, speed);
}
first_time = 0;
write_func (buf, h.len);
prev = h.tv;
free (buf);
}
return r;
}
static off_t
find_last_string_in_file(FILE * fp, const char *seq)
{
char buf[512];
struct stat mystat;
off_t offset = 0L;
const long readsz = sizeof(buf);
int bytes_read = 0;
const int seqlen = strlen(seq);
const char *reset_pos = seq + seqlen - 1;
const char *match_pos = reset_pos;
fstat(fileno (fp), &mystat);
offset = mystat.st_size - readsz;
if (offset < 0)
offset = 0;
while (1)
{
const char *search_pos = 0;
fseeko(fp, offset, SEEK_SET);
bytes_read = fread(buf, 1, readsz, fp);
if (bytes_read <= 0)
break;
search_pos = buf + bytes_read - 1;
while (search_pos >= buf)
{
int matched = *search_pos == *match_pos;
if (!matched && match_pos != reset_pos)
{
match_pos = reset_pos;
matched = *search_pos == *match_pos;
}
if (matched)
{
if (match_pos == seq)
return offset + (search_pos - buf);
--match_pos;
}
--search_pos;
}
// If we've reached the start of the file, exit.
if (!offset)
break;
offset -= readsz;
if (offset < 0)
offset = 0;
}
return 0;
}
static off_t
find_seek_offset_clrscr (FILE * fp)
{
off_t raw_seek_offset = 0;
off_t seek_offset_clrscr;
raw_seek_offset = find_last_string_in_file(fp, "\033[2J");
seek_offset_clrscr = 0;
/* now find last filepos that is less than seek offset */
fseek (fp, 0, SEEK_SET);
while (1)
{
char *buf;
Header h;
long offset;
if (ttyread (fp, &h, &buf, 0) != READ_DATA)
{
break;
}
free (buf);
offset = ftell(fp);
if (offset < raw_seek_offset)
seek_offset_clrscr = ftell (fp);
else
break;
}
return seek_offset_clrscr;
}
#if 0 /* not used anymore */
void
ttyskipall (FILE * fp)
{
/*
* Skip all records.
*/
ttyplay (fp, 0, ttyread, ttynowrite, ttynowait, 0);
}
#endif
void
ttyplayback (FILE * fp, double speed, ReadFunc read_func, WaitFunc wait_func)
{
ttyplay (fp, speed, ttyread, ttywrite, wait_func, 0);
}
void
ttypeek (FILE * fp, double speed)
{
int r;
do
{
setvbuf (fp, NULL, _IOFBF, 0);
r = ttyplay(fp, 0, ttyread, ttywrite, ttynowait, find_seek_offset_clrscr(fp));
if (r == READ_EOF) {
clearerr (fp);
setvbuf (fp, NULL, _IONBF, 0);
fflush (stdout);
r = ttyplay (fp, speed, ttypread, ttywrite, ttynowait, -1);
}
} while (r == READ_RESTART);
}
int
ttyplay_main (char *ttyfile, int mode, int resizex, int resizey)
{
double speed = 1.0;
ReadFunc read_func = ttyread;
WaitFunc wait_func = ttywait;
FILE *input = stdin;
struct termios old, new;
sighandler_t old_sigwinch;
populate_gfx_array (stripped);
input = efopen (ttyfile, "r");
tcgetattr (0, &old); /* Get current terminal state */
new = old; /* Make a copy */
new.c_lflag &= ~(ICANON | ECHO | ECHONL); /* unbuffered, no echo */
new.c_cc[VMIN] = 1;
new.c_cc[VTIME] = 0;
tcsetattr (0, TCSANOW, &new); /* Make it current */
if (resizex > 0 && resizey > 0) {
term_resizex = resizex;
term_resizey = resizey;
}
got_sigwinch = 0;
old_sigwinch = signal(SIGWINCH, ttyplay_sigwinch_func);
if (mode == 1)
ttypeek (input, speed);
else
ttyplayback (input, speed, read_func, wait_func);
tcsetattr (0, TCSANOW, &old); /* Return terminal state */
fclose (input);
if (old_sigwinch != SIG_ERR)
signal(SIGWINCH, old_sigwinch);
term_resizex = term_resizey = -1;
printf("\033[2J"); /* clear screen afterwards */
return 0;
}