Files
nethack/sys/vms/vmstty.c
PatR 284452796c redo tty resizing
Rest of 'not PR #1102'.  Resizing the terminal while getpos was in
operation recalculated the map from scratch instead of redrawing what
the core considers to already be shown.  And it was always operating
while an asynchronous signal was excuting which could potentially
clobber whatever was running at the time the signal arrived.

This uses same redrawing as the prior '^R during getpos()' fix.  It
also only performs the resize while tty_nhgetch() is waiting for
input.  If that is the situation at the time that the signal arrives
then it will resize immediately (while in the asynchronous signal
handler); if not, it will set a flag and tty_nhgetch() will do the
resize the next time it gets called.

This builds with TTY_PERM_INVENT enabled and doesn't seem to be any
worse than before, but there are bugs with that.  The only way I could
get perminv to appear was to save and restore, then perm_invent was
honored for both RC file and mO command.  And once I managed to get it
to display, moving an item from a lower case slot to slot 'A', made
that item vanish; nothing appeared in the invent's right hand panel.

Both of those misbehaviors already happen prior to this commit.  I
also saw an abort+panictrace if I resized while at the "Dump core?"
prompt when running the pre-commit code and didn't see that with the
post-commit code (although the prompt wasn't shown so I couldn't tell
that it was waiting for an answer).  The abort probably sounds scarier
than it warrants; I suspect that the pre-commit code just treated the
resize as answering 'y' for some reason, possibly a stale value in the
variable it uses.
2023-10-12 18:45:14 -07:00

617 lines
19 KiB
C

/* NetHack 3.7 vmstty.c $NHDT-Date: 1596498309 2020/08/03 23:45:09 $ $NHDT-Branch: NetHack-3.7 $:$NHDT-Revision: 1.21 $ */
/* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
/*-Copyright (c) Robert Patrick Rankin, 2011. */
/* NetHack may be freely redistributed. See license for details. */
/* tty.c - (VMS) version */
#define NEED_VARARGS
#include "hack.h"
#include "wintty.h"
#include "tcap.h"
#ifdef VMSVSI
#include <lib$routines.h>
#include <smg$routines.h>
#include <starlet.h>
#include <elfdef.h>
#endif
#include <descrip.h>
#include <iodef.h>
#ifndef __GNUC__
#include <smgdef.h>
#include <ttdef.h>
#include <tt2def.h>
#else /* values needed from missing include files */
#define SMG$K_TRM_UP 274
#define SMG$K_TRM_DOWN 275
#define SMG$K_TRM_LEFT 276
#define SMG$K_TRM_RIGHT 277
#define TT$M_MECHTAB 0x00000100 /* hardware tab support */
#define TT$M_MECHFORM 0x00080000 /* hardware form-feed support */
#define TT$M_NOBRDCST 0x00020000 /* disable broadcast messages, but */
#define TT2$M_BRDCSTMBX 0x00000010 /* catch them in associated mailbox */
#define TT2$M_APP_KEYPAD 0x00800000 /* application vs numeric keypad mode */
#endif /* __GNUC__ */
#ifdef USE_QIO_INPUT
#include <ssdef.h>
#endif
#include <errno.h>
#include <signal.h>
#ifndef VMSVSI
unsigned long lib$disable_ctrl(), lib$enable_ctrl();
unsigned long sys$assign(), sys$dassgn(), sys$qiow();
#endif
#ifndef USE_QIO_INPUT
#ifndef VMSVSI
unsigned long smg$create_virtual_keyboard(), smg$delete_virtual_keyboard(),
smg$read_keystroke(), smg$cancel_input();
#endif
#else
static short parse_function_key(int);
#endif
static void setctty(void);
static void resettty(void);
#define vms_ok(sts) ((sts) &1)
#define META(c) ((c) | 0x80) /* 8th bit */
#define CTRL(c) ((c) & 0x1F)
#define CMASK(c) (1 << CTRL(c))
#define LIB$M_CLI_CTRLT CMASK('T') /* 0x00100000 */
#define LIB$M_CLI_CTRLY CMASK('Y') /* 0x02000000 */
#define ESC '\033'
#define CSI META(ESC) /* '\233' */
#define SS3 META(CTRL('O')) /* '\217' */
extern short ospeed;
char erase_char, intr_char, kill_char;
static boolean settty_needed = FALSE, bombing = FALSE;
static unsigned long kb = 0;
#ifdef USE_QIO_INPUT
static char inputbuf[15 + 1], *inp = 0;
static int inc = 0;
#endif
#define QIO_FUNC IO$_TTYREADALL | IO$M_NOECHO | IO$M_TRMNOECHO
#ifdef MAIL
#define TT_SPECIAL_HANDLING (TT$M_MECHTAB | TT$M_MECHFORM | TT$M_NOBRDCST)
#define TT2_SPECIAL_HANDLING (TT2$M_BRDCSTMBX)
#else
#define TT_SPECIAL_HANDLING (TT$M_MECHTAB | TT$M_MECHFORM)
#define TT2_SPECIAL_HANDLING (0)
#endif
#define Uword unsigned short
#define Ubyte unsigned char
struct _rd_iosb { /* i/o status block for read */
Uword status, trm_offset;
Uword terminator, trm_siz;
};
struct _wr_iosb { /* i/o status block for write */
Uword status, byte_cnt;
unsigned : 32;
};
struct _sm_iosb { /* i/o status block for sense-mode qio */
Uword status;
Ubyte xmt_speed, rcv_speed;
Ubyte cr_fill, lf_fill, parity;
unsigned : 8;
};
struct _sm_bufr { /* sense-mode characteristics buffer */
Ubyte class, type; /* class==DC$_TERM, type==(various) */
Uword buf_siz; /* aka page width */
#define page_width buf_siz /* number of columns */
unsigned tt_char : 24; /* primary characteristics */
unsigned page_length : 8; /* number of lines */
unsigned tt2_char : 32; /* secondary characteristics */
};
static struct {
struct _sm_iosb io;
struct _sm_bufr sm;
} sg = { { 0 }, { 0 } };
static unsigned short tt_chan = 0;
static unsigned long tt_char_restore = 0, tt_char_active = 0,
tt2_char_restore = 0, tt2_char_active = 0;
static unsigned long ctrl_mask = 0;
#ifdef DEBUG
extern int nh_vms_getchar(void);
/* rename the real vms_getchar and interpose this one in front of it */
int
vms_getchar(void)
{
static int althack = 0, altprefix;
char *nhalthack;
int res;
if (!althack) {
/* one-time init */
nhalthack = nh_getenv("NH_ALTHACK");
althack = nhalthack ? 1 : -1;
if (althack > 0)
altprefix = *nhalthack;
}
#define vms_getchar nh_vms_getchar
res = vms_getchar();
if (althack > 0 && res == altprefix) {
res = vms_getchar();
if (res != ESC)
res = META(res);
}
return res;
}
#endif /*DEBUG*/
int
vms_getchar(void)
{
short key;
#ifdef USE_QIO_INPUT
struct _rd_iosb iosb;
unsigned long sts;
unsigned char kb_buf;
#else /* SMG input */
static volatile int recurse = 0; /* SMG is not AST re-entrant! */
#endif
if (gp.program_state.done_hup) {
/* hangup has occurred; do not attempt to get further user input */
return ESC;
}
#ifdef USE_QIO_INPUT
if (inc > 0) {
/* we have buffered character(s) from previous read */
kb_buf = *inp++;
--inc;
sts = SS$_NORMAL;
} else {
sts = sys$qiow(0, tt_chan, QIO_FUNC, &iosb, (void (*) ()) 0, 0,
&kb_buf, sizeof kb_buf, 0, 0, 0, 0);
}
if (vms_ok(sts)) {
if (kb_buf == CTRL('C')) {
if (intr_char)
gsignal(SIGINT);
key = (short) kb_buf;
} else if (kb_buf == '\r') { /* <return> */
key = (short) '\n';
} else if (kb_buf == ESC || kb_buf == CSI || kb_buf == SS3) {
switch (parse_function_key((int) kb_buf)) {
case SMG$K_TRM_UP:
key = gc.Cmd.dirchars[2];
break;
case SMG$K_TRM_DOWN:
key = gc.Cmd.dirchars[6];
break;
case SMG$K_TRM_LEFT:
key = gc.Cmd.dirchars[0];
break;
case SMG$K_TRM_RIGHT:
key = gc.Cmd.dirchars[4];
break;
default:
key = ESC;
break;
}
} else {
key = (short) kb_buf;
}
} else if (sts == SS$_HANGUP || iosb.status == SS$_HANGUP
|| sts == SS$_DEVOFFLINE) {
gsignal(SIGHUP);
key = ESC;
} else /*(this should never happen)*/
key = getchar();
#else /*!USE_QIO_INPUT*/
if (recurse++ == 0 && kb != 0) {
smg$read_keystroke(&kb, &key);
switch (key) {
case SMG$K_TRM_UP:
key = gc.Cmd.move_N;
break;
case SMG$K_TRM_DOWN:
key = gc.Cmd.move_S;
break;
case SMG$K_TRM_LEFT:
key = gc.Cmd.move_W;
break;
case SMG$K_TRM_RIGHT:
key = gc.Cmd.move_E;
break;
case '\r':
key = '\n';
break;
default:
if (key > 255)
key = ESC;
break;
}
} else {
/* abnormal input--either SMG didn't initialize properly or
* vms_getchar() has been called recursively (via SIGINT handler).
*/
if (kb != 0) /* must have been a recursive call */
smg$cancel_input(&kb); /* from an interrupt handler */
key = getchar();
}
--recurse;
#endif /* USE_QIO_INPUT */
return (int) key;
}
#ifdef USE_QIO_INPUT
/*
* We've just gotten an <escape> character. Do a timed read to
* get any other characters, then try to parse them as an escape
* sequence. This isn't perfect, since there's no guarantee
* that a full escape sequence will be available, or even if one
* is, it might actually by regular input from a fast typist or
* a stalled input connection. {For packetized environments,
* cross plural(body_part(FINGER)) and hope for best. :-}
*
* This is needed to preserve compatibility with SMG interface
* for two reasons:
* 1) retain support for arrow keys, and
* 2) treat other VTxxx function keys as <esc> for aborting
* various NetHack prompts.
* The second reason is compelling; otherwise remaining chars of
* an escape sequence get treated as inappropriate user commands.
*
* SMG code values for these key sequences fall in the range of
* 256 thru 3xx. The assignments are not particularly intuitive.
*/
/*=
-- Summary of VTxxx-style keyboards and transmitted escape sequences. --
Keypad codes are prefixed by 7 bit (\033 O) or 8 bit SS3:
keypad: PF1 PF2 PF3 PF4 codes: P Q R S
7 8 9 - w x y m
4 5 6 . t u v n
1 2 3 :en-: q r s : :
...0... , :ter: ...p... l :M:
Arrows are prefixed by either SS3 or CSI (either 7 or 8 bit), depending on
whether the terminal is in application or numeric mode (ditto for PF keys):
arrows: <up> <dwn> <lft> <rgt> A B D C
Additional function keys (vk201/vk401) generate CSI nn ~ (nn is 1 or 2 digits):
vk201 keys: F6 F7 F8 F9 F10 F11 F12 F13 F14 Help Do F17 F18 F19 F20
'nn' digits: 17 18 19 20 21 23 24 25 26 28 29 31 32 33 34
alternate: ^C ^[ ^H ^J (when in VT100 mode)
edit keypad: <fnd> <ins> <rmv> digits: 1 2 3
<sel> <prv> <nxt> 4 5 6
VT52 mode: arrows and PF keys send ESCx where x is in A-D or P-S.
=*/
static const char *arrow_or_PF = "ABCDPQRS", /* suffix char */
*smg_keypad_codes = "PQRSpqrstuvwxyMmlnABDC";
/* PF1..PF4,KP0..KP9,enter,dash,comma,dot,up-arrow,down,left,right */
/* Ultimate return value is (index into smg_keypad_codes[] + 256). */
static short
parse_function_key(register int c)
{
struct _rd_iosb iosb;
unsigned long sts;
char seq_buf[15 + 1]; /* plenty room for escape sequence + slop */
short result = ESC; /* translate to <escape> by default */
/*
* Read whatever we can from type-ahead buffer (1 second timeout).
* If the user typed an actual <escape> to deliberately abort
* something, he or she should be able to tolerate the necessary
* restriction of a negligible pause before typing anything else.
* We might already have [at least some of] an escape sequence from a
* previous read, particularly if user holds down the arrow keys...
*/
if (inc > 0)
strncpy(seq_buf, inp, inc);
if (inc < (int) (sizeof seq_buf) - 1) {
sts = sys$qiow(0, tt_chan, QIO_FUNC | IO$M_TIMED, &iosb,
(void (*) ()) 0, 0, seq_buf + inc,
sizeof seq_buf - 1 - inc, 1, 0, 0, 0);
if (vms_ok(sts))
sts = iosb.status;
} else
sts = SS$_NORMAL;
if (vms_ok(sts) || sts == SS$_TIMEOUT) {
register int cnt = iosb.trm_offset + iosb.trm_siz + inc;
register char *p = seq_buf;
if (c == ESC) /* check for 7-bit vt100/ANSI, or vt52 */
if (*p == '[' || *p == 'O')
c = META(CTRL(*p++)), cnt--;
else if (strchr(arrow_or_PF, *p))
c = SS3; /*CSI*/
if (cnt > 0 && (c == SS3 || (c == CSI && strchr(arrow_or_PF, *p)))) {
register char *q = strchr(smg_keypad_codes, *p);
if (q)
result = 256 + (q - smg_keypad_codes);
p++, --cnt; /* one more char consumed */
} else if (cnt > 1 && c == CSI) {
static short /* "CSI nn ~" -> F_keys[nn] */
F_keys[35] = {
ESC, /*(filler)*/
311, 312, 313, 314, 315, 316, /* E1-E6 */
ESC, ESC, ESC, ESC, /*(more filler)*/
281, 282, 283, 284, 285, ESC, /* F1-F5 */
286, 287, 288, 289, 290, ESC, /* F6-F10*/
291, 292, 293, 294, ESC, /*F11-F14*/
295, 296, ESC, /*<help>,<do>, aka F15,F16*/
297, 298, 299, 300 /*F17-F20*/
}; /* note: there are several missing nn in CSI nn ~ values */
int nn;
char *q;
*(p + cnt) = '\0'; /* terminate string */
q = strchr(p, '~');
if (q && sscanf(p, "%d~", &nn) == 1) {
if (nn > 0 && nn < SIZE(F_keys))
result = F_keys[nn];
cnt -= (++q - p);
p = q;
}
}
if (cnt > 0)
strncpy((inp = inputbuf), p, (inc = cnt));
else
inc = 0, inp = 0;
}
return result;
}
#endif /* USE_QIO_INPUT */
static void
setctty(void)
{
struct _sm_iosb iosb;
unsigned long status;
status = sys$qiow(0, tt_chan, IO$_SETMODE, &iosb, (void (*) ()) 0, 0,
&sg.sm, sizeof sg.sm, 0, 0, 0, 0);
if (vms_ok(status))
status = iosb.status;
if (vms_ok(status)) {
/* try to force terminal into synch with TTDRIVER's setting */
number_pad((sg.sm.tt2_char & TT2$M_APP_KEYPAD) ? -1 : 1);
} else {
raw_print("");
errno = EVMSERR, vaxc$errno = status;
perror("NetHack(setctty: setmode)");
wait_synch();
}
}
/* atexit() routine */
static void
resettty(void)
{
if (settty_needed) {
bombing = TRUE; /* don't clear screen; preserve traceback info */
settty((char *) 0);
}
(void) sys$dassgn(tt_chan), tt_chan = 0;
}
/*
* Get initial state of terminal, set ospeed (for termcap routines)
* and switch off tab expansion if necessary.
* Called by init_nhwindows() and resume_nhwindows() in wintty.c
* (for initial startup and for returning from '!' or ^Z).
*/
void
gettty(void)
{
static char dev_tty[] = "TT:";
static $DESCRIPTOR(tty_dsc, dev_tty);
int err = 0;
unsigned long status, zero = 0;
if (tt_chan == 0) { /* do this stuff once only */
iflags.cbreak = OFF, iflags.echo = ON; /* until setup is complete */
status = sys$assign(&tty_dsc, &tt_chan, 0, 0);
if (!vms_ok(status)) {
raw_print(""), err++;
errno = EVMSERR, vaxc$errno = status;
perror("NetHack(gettty: $assign)");
}
atexit(resettty); /* register an exit handler to reset things */
}
status = sys$qiow(0, tt_chan, IO$_SENSEMODE, &sg.io, (void (*) ()) 0, 0,
&sg.sm, sizeof sg.sm, 0, 0, 0, 0);
if (vms_ok(status))
status = sg.io.status;
if (!vms_ok(status)) {
raw_print(""), err++;
errno = EVMSERR, vaxc$errno = status;
perror("NetHack(gettty: sensemode)");
}
ospeed = sg.io.xmt_speed;
erase_char = '\177'; /* <rubout>, aka <delete> */
kill_char = CTRL('U');
intr_char = CTRL('C');
(void) lib$enable_ctrl(&zero, &ctrl_mask);
/* Use the systems's values for lines and columns if it has any idea. */
if (sg.sm.page_length)
LI = sg.sm.page_length;
if (sg.sm.page_width)
CO = sg.sm.page_width;
/* suppress tab and form-feed expansion, in case termcap uses them */
tt_char_restore = sg.sm.tt_char;
tt_char_active = sg.sm.tt_char |= TT_SPECIAL_HANDLING;
tt2_char_restore = sg.sm.tt2_char;
tt2_char_active = sg.sm.tt2_char |= TT2_SPECIAL_HANDLING;
#if 0 /*[ defer until setftty() ]*/
setctty();
#endif
if (err)
wait_synch();
}
/* reset terminal to original state */
void
settty(const char *s)
{
if (!bombing)
end_screen();
if (s)
raw_print(s);
if (settty_needed) {
disable_broadcast_trapping();
#if 0 /* let SMG's exit handler do the cleanup (as per doc) */
/* #ifndef USE_QIO_INPUT */
if (kb)
smg$delete_virtual_keyboard(&kb), kb = 0;
#endif /* 0 (!USE_QIO_INPUT) */
if (ctrl_mask)
(void) lib$enable_ctrl(&ctrl_mask, 0);
iflags.echo = ON;
iflags.cbreak = OFF;
/* reset original tab, form-feed, broadcast settings */
sg.sm.tt_char = tt_char_restore;
sg.sm.tt2_char = tt2_char_restore;
setctty();
settty_needed = FALSE;
}
}
/* same as settty, with no clearing of the screen */
void
shuttty(const char *s)
{
bombing = TRUE;
settty(s);
bombing = FALSE;
}
void
setftty(void)
{
unsigned long mask = LIB$M_CLI_CTRLT | LIB$M_CLI_CTRLY;
(void) lib$disable_ctrl(&mask, 0);
if (kb == 0) { /* do this stuff once only */
#ifdef USE_QIO_INPUT
kb = tt_chan;
#else /*!USE_QIO_INPUT*/
smg$create_virtual_keyboard(&kb);
#endif /*USE_QIO_INPUT*/
init_broadcast_trapping();
}
enable_broadcast_trapping(); /* no-op if !defined(MAIL) */
iflags.cbreak = (kb != 0) ? ON : OFF;
iflags.echo = (kb != 0) ? OFF : ON;
/* disable tab & form-feed expansion; prepare for broadcast trapping */
sg.sm.tt_char = tt_char_active;
sg.sm.tt2_char = tt2_char_active;
setctty();
start_screen();
settty_needed = TRUE;
}
/* enable kbd interupts if enabled when game started */
void
intron(void)
{
intr_char = CTRL('C');
}
/* disable kbd interrupts if required*/
void
introff(void)
{
intr_char = 0;
}
#ifdef TIMED_DELAY
#ifndef VMSVSI
extern unsigned long lib$emul(const long *, const long *, const long *,
long *);
extern unsigned long sys$schdwk(), sys$hiber();
#endif
#define VMS_UNITS_PER_SECOND 10000000L /* hundreds of nanoseconds, 1e-7 */
/* constant for conversion from milliseconds to VMS delta time (negative) */
static const long mseconds_to_delta = VMS_UNITS_PER_SECOND / 1000L * -1L;
/* sleep for specified number of milliseconds (note: the timer used
generally only has 10-millisecond resolution at the hardware level...) */
void
msleep(unsigned mseconds) /* milliseconds */
{
long pid = 0L, zero = 0L, msec, qtime[2];
msec = (long) mseconds;
if (msec > 0 &&
/* qtime{0:63} = msec{0:31} * mseconds_to_delta{0:31} + zero{0:31} */
vms_ok(lib$emul(&msec, &mseconds_to_delta, &zero, qtime))) {
/* schedule a wake-up call, then go to sleep */
if (vms_ok(sys$schdwk(&pid, (genericptr_t) 0, qtime, (long *) 0)))
(void) sys$hiber();
}
}
#endif /* TIMED_DELAY */
/* fatal error */
/*VARARGS1*/
void error
VA_DECL(const char *, s)
{
VA_START(s);
VA_INIT(s, const char *);
if (settty_needed)
settty((char *) 0);
Vprintf(s, VA_ARGS);
(void) putchar('\n');
VA_END();
#ifndef SAVE_ON_FATAL_ERROR
/* prevent vmsmain's exit handler byebye() from calling hangup() */
/* sethanguphandler((void (*)(int) )) SIG_DFL; */
sethanguphandler((SIG_RET_TYPE) SIG_DFL);
#endif
exit(EXIT_FAILURE);
}
#ifdef SIGWINCH
/* called by resize_tty(wintty.c) after receiving a SIGWINCH signal;
terminal size has changed and we should update LI and CO (from termcap) */
void
getwindowsz(void)
{
/*
* gettty() has code to do this, but it can't be used directly because
* it fetches terminal state in order to reset that upon termination.
* We need to avoid clobbering other saved state with values used by
* game-in-progress. For now, do nothing.
*/
return;
}
#endif
#ifdef ENHANCED_SYMBOLS
/*
* set in tty_start_screen() and allows
* OS-specific changes that may be
* required for support of utf8.
* Currently a placeholder for VMS.
*/
void
tty_utf8graphics_fixup(void)
{
return;
}
#endif /* ENHANCED_SYMBOLS */
/*vmstty.c */