marlene/darlene.c
Kelvin Sherlock 4da75367c1 (re)set the controlling terminal.
This is necessary to read input from the console (via the Event Manager) since opening a pseudo terminal screws up the controlling terminal (and there is no O_NOCTTY flag).
2019-10-03 00:05:21 -04:00

352 lines
6.9 KiB
C

/*
* The original concept. Not the original code :)
*
*/
#include <Event.h>
#include <gsos.h>
#include <Locator.h>
#include <Memory.h>
#include <shell.h>
#include <texttool.h>
#include <errno.h>
#include <fcntl.h>
#include <libutil.h>
#include <sgtty.h>
#include <signal.h>
#include <string.h>
#include <sys/ioctl.compat.h>
#include <sys/ioctl.h>
#include <sys/wait.h>
#include <unistd.h>
#include <pwd.h>
#include <gno/gno.h>
#include <gno/kerntool.h>
#include "vt100.h"
extern void screen_init(void);
extern void screen_on(void);
extern void screen_off(void);
void display_pstr(const char *);
void display_cstr(const char *);
static char out_buffer[1024];
static char out_buffer_size = 0;
static int master = -1;
static void flush(void) {
if (out_buffer_size) {
write(master, out_buffer, out_buffer_size);
out_buffer_size = 0;
}
}
void send(char *data, unsigned size) {
if (out_buffer_size + size > sizeof(out_buffer)) {
flush();
}
if (size > sizeof(out_buffer) / 2) {
write(master, data, size);
return;
}
memcpy(out_buffer + out_buffer_size, data, size);
out_buffer_size += size;
}
int
openpty2(int *amaster, int *aslave, char *name, struct sgttyb *sg, struct winsize *winp) {
char ptyname[] = ".ptyq0";
char ttyname[] = ".ttyq0";
int master;
int slave;
unsigned i;
for (i = 0; i < 16; ++i) {
char c = "0123456789abcdef"[i];
ptyname[5] = c;
ttyname[5] = c;
master = open(ptyname, O_RDWR);
if (master < 0) continue;
slave = open(ttyname, O_RDWR);
if (slave < 0) {
close(master);
continue;
}
*amaster = master;
*aslave = slave;
if (name) strcpy(name, ttyname);
if (sg) stty(slave, sg);
if (winp) ioctl(slave, TIOCSWINSZ, (char *)winp);
return 0;
}
errno = ENOENT;
return -1;
}
#define _write(fd, msg) write(fd, msg, sizeof(msg) - 1);
static SetGSPB term_var = {
3, "\x04\x00TERM", "\x05\x00vt100", 1
};
static char **child_argv = NULL;
static struct passwd *pwd = NULL;
static char *whoami = "";
#pragma databank 1
static int _child(int master, int slave) {
static unsigned zero = 0;
int ok;
close(master);
ok = login_tty(slave);
if (ok < 0) {
_write(slave, "login_tty failed.\r\n");
close(slave);
return 1;
}
/* work around GNO bug where environment leaks into parent */
PushVariablesGS(&zero);
SetGS(&term_var);
if (child_argv) {
execvp(child_argv[0], child_argv);
} else {
static char buffer[64] = "login -f ";
strcpy(buffer + 9, whoami);
/* should sniff the correct TERM value from /etc/ttys */
_execve(":usr:bin:login", buffer);
_execve(":usr:sbin:login", buffer);
_execve(":bin:gsh", "gsh");
}
_write(STDERR_FILENO, "_execve failed.\r\n");
return 1;
}
#undef _write
#pragma databank 0
int
forkpty2(int *amaster, char *name, struct sgttyb *sg, struct winsize *winp) {
int ok;
int master, slave, pid;
ok = openpty2(&master, &slave, name, sg, winp);
if (ok < 0) return -1;
pid = fork2(_child, 512, 0, "darlene child", 2, master, slave);
if (pid < 0) {
close(master);
close(slave);
return -1;
}
*amaster = master;
close(slave);
return pid;
}
#pragma databank 1
static void sigchild(int sig, int x) {
//display_cstr("\r\nsigchild\r\n");
PostEvent(app4Evt,0);
}
#pragma databank 0
int main(int argc, char **argv) {
static EventRecord event;
int pid;
unsigned i;
Word MyID;
unsigned vt100_flags = vtDEFAULT;
Handle dpHandle = NULL;
Handle shrHandle = NULL;
Handle shdHandle = NULL;
master = -1;
kernStatus();
if (_toolErr) {
ErrWriteCString("GNO/ME required.\r\n");
return 1;
}
if (!isatty(STDIN_FILENO)) {
ErrWriteCString("stdin required.\r\n");
return 1;
}
term_var.value = "\x05\x00vt100";
child_argv = NULL;
for (i = 1; i < argc; ++i) {
char *cp = argv[i];
if (cp[0] != '-') break;
if (strcmp(cp, "--") == 0) {
break;
} else if (strcmp(cp, "--vt52") == 0) {
term_var.value = "\x04\x00vt52";
vt100_flags &= ~vtDECANM;
} else if (strcmp(cp, "--vt100") == 0) {
term_var.value = "\x05\x00vt100";
vt100_flags |= vtDECANM;
} else {
ErrWriteCString("Unknown option: ");
ErrWriteCString(cp);
ErrWriteCString("\r\n");
return 1;
}
}
argc -= i;
argv += i;
if (argc) {
child_argv = malloc(argc * 4 + 4);
/* <= argc so trailing NULL copies */
for(i = 0; i <= argc; ++i) child_argv[i] = argv[i];
}
pwd = getpwuid(getuid());
whoami = pwd ? pwd->pw_name : "";
MyID = MMStartUp();
dpHandle = NewHandle(0x0100, MyID,
attrBank | attrPage |attrNoCross | attrFixed | attrLocked,
0x000000);
shdHandle = NewHandle(0x8000, MyID,
attrAddr | attrFixed | attrLocked,
(void *)0x012000);
shrHandle = NewHandle(0x8000, MyID,
attrAddr | attrFixed | attrLocked,
(void *)0xe12000);
if (!dpHandle || !shdHandle || !shrHandle) {
ErrWriteCString("Unable to allocate memory.\r\n");
if (dpHandle) DisposeHandle(dpHandle);
if (shdHandle) DisposeHandle(shdHandle);
if (shrHandle) DisposeHandle(shrHandle);
return 1;
}
EMStartUp((Word)*dpHandle, 0x14, 0, 0, 0, 0, MyID);
screen_init();
vt100_init(vt100_flags);
signal(SIGCHLD,sigchild);
pid = forkpty2(&master, NULL, NULL, NULL);
if ( pid < 0) {
display_cstr("Unable to forkpty.\r\n");
goto _exit;
}
/* reset the controlling terminal, which was clobbered by opening a pty */
/* standard TIOCSCTTY causes ORCA/C shift errors */
#undef TIOCSCTTY
#define TIOCSCTTY (0x20000000ul | ('t' << 8) | 97)
ioctl(STDIN_FILENO, TIOCSCTTY, (void *)0);
for(;;) {
static char buffer[1024];
int fio = 0;
int ok;
ok = ioctl(master, FIONREAD, &fio);
if (fio > sizeof(buffer)) fio = sizeof(buffer);
if (fio > 0) {
fio = read(master, buffer, fio);
if (fio > 0) vt100_process(buffer, fio);
}
if (GetNextEvent(everyEvent, &event)) {
fio = 1;
if (event.what == keyDownEvt) {
unsigned char key = event.message;
if (event.modifiers & appleKey) {
switch (key) {
case 'Q': // quit
case 'q':
goto _exit1;
break;
case 'V': // paste...
case 'v':
break;
case 'Z': // suspend (gnome)
case 'z': {
/* EMStartUp puts the tty in RAW mode. */
static struct sgttyb sb;
gtty(1,&sb);
sb.sg_flags &= ~RAW;
stty(1,&sb);
screen_off();
Kkill(Kgetpid(), SIGSTOP, &errno);
sb.sg_flags |= RAW;
stty(1,&sb);
screen_on();
}
break;
}
} else {
vt100_event(&event);
}
}
if (event.what == app4Evt) {
/* child signal received! */
union wait wt;
ok = waitpid(pid, &wt, WNOHANG);
if (ok <= 0) continue;
display_cstr("\r\nChild exited.\r\n");
break;
}
}
flush();
if (fio <= 0) asm { cop 0x7f }
}
_exit1:
flush();
if (master > 0) close(master);
_exit:
// flush q
FlushEvents(everyEvent, 0);
display_cstr("\n\rPress any key to exit.\n\r");
while (!GetNextEvent(keyDownMask | autoKeyMask, &event)) ;
EMShutDown();
DisposeHandle(dpHandle);
DisposeHandle(shdHandle);
DisposeHandle(shrHandle);
screen_off();
return 0;
}