Added termcap support
This commit is contained in:
parent
35b3050d88
commit
d7bb7c9cf3
7 changed files with 745 additions and 1 deletions
|
@ -25,3 +25,5 @@ Epilogue.def
|
||||||
Streams.def
|
Streams.def
|
||||||
ArraySort.def
|
ArraySort.def
|
||||||
StripUnix.def
|
StripUnix.def
|
||||||
|
Termcap.def
|
||||||
|
XXTermcap.def
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
tail_m2.a
|
tail_m2.a
|
||||||
|
Termcap.mod
|
||||||
CSP.mod
|
CSP.mod
|
||||||
PascalIO.mod
|
PascalIO.mod
|
||||||
RealInOut.mod
|
RealInOut.mod
|
||||||
|
@ -17,6 +18,7 @@ Strings.mod
|
||||||
ArraySort.mod
|
ArraySort.mod
|
||||||
catch.c
|
catch.c
|
||||||
Traps.mod
|
Traps.mod
|
||||||
|
XXTermcap.c
|
||||||
Arguments.c
|
Arguments.c
|
||||||
LtoUset.e
|
LtoUset.e
|
||||||
StrAss.c
|
StrAss.c
|
||||||
|
|
|
@ -6,7 +6,8 @@ SOURCES = ASCII.def EM.def MathLib0.def Processes.def \
|
||||||
random.def Semaphores.def Unix.def RealConver.def \
|
random.def Semaphores.def Unix.def RealConver.def \
|
||||||
Strings.def InOut.def Terminal.def \
|
Strings.def InOut.def Terminal.def \
|
||||||
Mathlib.def PascalIO.def Traps.def CSP.def \
|
Mathlib.def PascalIO.def Traps.def CSP.def \
|
||||||
Epilogue.def Streams.def ArraySort.def StripUnix.def
|
Epilogue.def Streams.def ArraySort.def StripUnix.def \
|
||||||
|
Termcap.def XXTermcap.def
|
||||||
|
|
||||||
all:
|
all:
|
||||||
clean:
|
clean:
|
||||||
|
|
45
lang/m2/libm2/Termcap.def
Normal file
45
lang/m2/libm2/Termcap.def
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
(*
|
||||||
|
(c) copyright 1988 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
||||||
|
See the copyright notice in the ACK home directory, in the file "Copyright".
|
||||||
|
*)
|
||||||
|
|
||||||
|
(*
|
||||||
|
Module: Interface to termcap database
|
||||||
|
From: Unix manual chapter 3
|
||||||
|
Version: $Header$
|
||||||
|
*)
|
||||||
|
|
||||||
|
(*
|
||||||
|
Use this like the C-version. In this Modula-2 version, some of the buffers,
|
||||||
|
that are explicit in the C-version, are hidden. These buffers are initialized
|
||||||
|
by a call to Tgetent. The "ARRAY OF CHAR" parameters must be null-terminated.
|
||||||
|
You can call them with a constant string argument, as these are always
|
||||||
|
null-terminated in our Modula-2 implementation. Also, the strings returned
|
||||||
|
by these routines are null-terminated.
|
||||||
|
*)
|
||||||
|
|
||||||
|
DEFINITION MODULE Termcap;
|
||||||
|
|
||||||
|
TYPE STR = ARRAY[1..32] OF CHAR;
|
||||||
|
PTSTR = POINTER TO STR;
|
||||||
|
PUTPROC = PROCEDURE(CHAR);
|
||||||
|
|
||||||
|
VAR UP, BC: PTSTR; (* must be null-terminated *)
|
||||||
|
ospeed: INTEGER; (* see termcap(3), tty(4) *)
|
||||||
|
PC: CHAR; (* is initially 0C *)
|
||||||
|
|
||||||
|
PROCEDURE Tgetent(name: ARRAY OF CHAR) : INTEGER;
|
||||||
|
|
||||||
|
PROCEDURE Tgetnum(id: ARRAY OF CHAR): INTEGER;
|
||||||
|
|
||||||
|
PROCEDURE Tgetflag(id: ARRAY OF CHAR): BOOLEAN;
|
||||||
|
|
||||||
|
PROCEDURE Tgoto(cm: PTSTR; col, line: INTEGER): PTSTR;
|
||||||
|
(* Result exists until next call to Tgoto *)
|
||||||
|
|
||||||
|
PROCEDURE Tgetstr(id: ARRAY OF CHAR): PTSTR;
|
||||||
|
(* Result exists until next call to Tgetent *)
|
||||||
|
|
||||||
|
PROCEDURE Tputs(cp: PTSTR; affcnt: INTEGER; p: PUTPROC);
|
||||||
|
|
||||||
|
END Termcap.
|
83
lang/m2/libm2/Termcap.mod
Normal file
83
lang/m2/libm2/Termcap.mod
Normal file
|
@ -0,0 +1,83 @@
|
||||||
|
(*
|
||||||
|
(c) copyright 1988 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
||||||
|
See the copyright notice in the ACK home directory, in the file "Copyright".
|
||||||
|
*)
|
||||||
|
|
||||||
|
(*
|
||||||
|
Module: Interface to termcap database
|
||||||
|
From: Unix manual chapter 3
|
||||||
|
Version: $Header$
|
||||||
|
*)
|
||||||
|
|
||||||
|
(*$R-*)
|
||||||
|
IMPLEMENTATION MODULE Termcap;
|
||||||
|
|
||||||
|
IMPORT XXTermcap;
|
||||||
|
FROM SYSTEM IMPORT ADR, ADDRESS;
|
||||||
|
|
||||||
|
VAR Buf, Buf1 : ARRAY [1..1024] OF CHAR;
|
||||||
|
Initialized: BOOLEAN;
|
||||||
|
BufCnt : INTEGER;
|
||||||
|
|
||||||
|
PROCEDURE Tgetent(name: ARRAY OF CHAR) : INTEGER;
|
||||||
|
VAR i: INTEGER;
|
||||||
|
BEGIN
|
||||||
|
i := XXTermcap.tgetent(ADR(Buf), ADR(name));
|
||||||
|
IF i > 0 THEN
|
||||||
|
Initialized := TRUE
|
||||||
|
END;
|
||||||
|
BufCnt := 1;
|
||||||
|
RETURN i;
|
||||||
|
END Tgetent;
|
||||||
|
|
||||||
|
PROCEDURE Tgetnum(id: ARRAY OF CHAR): INTEGER;
|
||||||
|
BEGIN
|
||||||
|
IF NOT Initialized THEN
|
||||||
|
RETURN -1;
|
||||||
|
END;
|
||||||
|
RETURN XXTermcap.tgetnum(ADR(id));
|
||||||
|
END Tgetnum;
|
||||||
|
|
||||||
|
PROCEDURE Tgetflag(id: ARRAY OF CHAR): BOOLEAN;
|
||||||
|
BEGIN
|
||||||
|
IF NOT Initialized THEN
|
||||||
|
RETURN FALSE;
|
||||||
|
END;
|
||||||
|
RETURN XXTermcap.tgetflag(ADR(id)) = 1;
|
||||||
|
END Tgetflag;
|
||||||
|
|
||||||
|
PROCEDURE Tgoto(cm: PTSTR; col, line: INTEGER): PTSTR;
|
||||||
|
BEGIN
|
||||||
|
XXTermcap.UP := UP;
|
||||||
|
XXTermcap.BC := BC;
|
||||||
|
RETURN XXTermcap.tgoto(cm, col, line);
|
||||||
|
END Tgoto;
|
||||||
|
|
||||||
|
PROCEDURE Tgetstr(id: ARRAY OF CHAR): PTSTR;
|
||||||
|
VAR a, a2: ADDRESS;
|
||||||
|
b: CARDINAL;
|
||||||
|
BEGIN
|
||||||
|
IF NOT Initialized THEN
|
||||||
|
RETURN NIL;
|
||||||
|
END;
|
||||||
|
a := ADR(Buf1[BufCnt]);
|
||||||
|
a2 := XXTermcap.tgetstr(ADR(id), ADR(a));
|
||||||
|
IF a2 = NIL THEN
|
||||||
|
RETURN NIL;
|
||||||
|
END;
|
||||||
|
b := a - a2;
|
||||||
|
INC(BufCnt, b);
|
||||||
|
RETURN a2;
|
||||||
|
END Tgetstr;
|
||||||
|
|
||||||
|
PROCEDURE Tputs(cp: PTSTR; affcnt: INTEGER; p: PUTPROC);
|
||||||
|
BEGIN
|
||||||
|
XXTermcap.PC := PC;
|
||||||
|
XXTermcap.ospeed := ospeed;
|
||||||
|
XXTermcap.tputs(cp, affcnt, XXTermcap.PUTPROC(p));
|
||||||
|
END Tputs;
|
||||||
|
|
||||||
|
BEGIN
|
||||||
|
Initialized := FALSE;
|
||||||
|
PC := 0C;
|
||||||
|
END Termcap.
|
564
lang/m2/libm2/XXTermcap.c
Normal file
564
lang/m2/libm2/XXTermcap.c
Normal file
|
@ -0,0 +1,564 @@
|
||||||
|
/*
|
||||||
|
* termcap.c 1.1 20/7/87 agc Joypace Ltd
|
||||||
|
*
|
||||||
|
* Copyright Joypace Ltd, London, UK, 1987. All rights reserved.
|
||||||
|
* This file may be freely distributed provided that this notice
|
||||||
|
* remains attached.
|
||||||
|
*
|
||||||
|
* A public domain implementation of the termcap(3) routines.
|
||||||
|
*
|
||||||
|
* Made fully functional by Ceriel J.H. Jacobs.
|
||||||
|
*
|
||||||
|
* BUGS:
|
||||||
|
* - does not check termcap entry sizes
|
||||||
|
* - not fully tested
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define CAPABLEN 2
|
||||||
|
|
||||||
|
#define ISSPACE(c) ((c) == ' ' || (c) == '\t' || (c) == '\r' || (c) == '\n')
|
||||||
|
#define ISDIGIT(x) ((x) >= '0' && (x) <= '9')
|
||||||
|
|
||||||
|
short ospeed; /* output speed */
|
||||||
|
char PC; /* padding character */
|
||||||
|
char *BC; /* back cursor movement */
|
||||||
|
char *UP; /* up cursor movement */
|
||||||
|
|
||||||
|
static char *capab; /* the capability itself */
|
||||||
|
static int check_for_tc();
|
||||||
|
static int match_name();
|
||||||
|
|
||||||
|
#define NULL 0
|
||||||
|
|
||||||
|
/* Some things from C-library, needed here because the C-library is not
|
||||||
|
loaded with Modula-2 programs
|
||||||
|
*/
|
||||||
|
|
||||||
|
static char *
|
||||||
|
strcat(s1, s2)
|
||||||
|
register char *s1, *s2;
|
||||||
|
{
|
||||||
|
/* Append s2 to the end of s1. */
|
||||||
|
|
||||||
|
char *original = s1;
|
||||||
|
|
||||||
|
/* Find the end of s1. */
|
||||||
|
while (*s1 != 0) s1++;
|
||||||
|
|
||||||
|
/* Now copy s2 to the end of s1. */
|
||||||
|
while (*s1++ = *s2++) /* nothing */ ;
|
||||||
|
return(original);
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *
|
||||||
|
strcpy(s1, s2)
|
||||||
|
register char *s1, *s2;
|
||||||
|
{
|
||||||
|
/* Copy s2 to s1. */
|
||||||
|
char *original = s1;
|
||||||
|
|
||||||
|
while (*s1++ = *s2++) /* nothing */;
|
||||||
|
return(original);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
strlen(s)
|
||||||
|
char *s;
|
||||||
|
{
|
||||||
|
/* Return length of s. */
|
||||||
|
|
||||||
|
char *original = s;
|
||||||
|
|
||||||
|
while (*s != 0) s++;
|
||||||
|
return(s - original);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
strcmp(s1, s2)
|
||||||
|
register char *s1, *s2;
|
||||||
|
{
|
||||||
|
/* Compare 2 strings. */
|
||||||
|
|
||||||
|
for(;;) {
|
||||||
|
if (*s1 != *s2) {
|
||||||
|
if (!*s1) return -1;
|
||||||
|
if (!*s2) return 1;
|
||||||
|
return(*s1 - *s2);
|
||||||
|
}
|
||||||
|
if (*s1++ == 0) return(0);
|
||||||
|
s2++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
strncmp(s1, s2, n)
|
||||||
|
register char *s1, *s2;
|
||||||
|
int n;
|
||||||
|
{
|
||||||
|
/* Compare two strings, but at most n characters. */
|
||||||
|
|
||||||
|
while (n-- > 0) {
|
||||||
|
if (*s1 != *s2) {
|
||||||
|
if (!*s1) return -1;
|
||||||
|
if (!*s2) return 1;
|
||||||
|
return(*s1 - *s2);
|
||||||
|
}
|
||||||
|
if (*s1++ == 0) break;
|
||||||
|
s2++;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *
|
||||||
|
getenv(name)
|
||||||
|
register char *name;
|
||||||
|
{
|
||||||
|
extern char **environ;
|
||||||
|
register char **v = environ, *p, *q;
|
||||||
|
|
||||||
|
if (v == 0 || name == 0) return 0;
|
||||||
|
while ((p = *v++) != 0) {
|
||||||
|
q = name;
|
||||||
|
while (*q && *q++ == *p++) /* nothing */ ;
|
||||||
|
if (*q || *p != '=') continue;
|
||||||
|
return(p+1);
|
||||||
|
}
|
||||||
|
return(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *
|
||||||
|
fgets(buf, count, fd)
|
||||||
|
char *buf;
|
||||||
|
{
|
||||||
|
static char bf[1024];
|
||||||
|
static int cnt = 0;
|
||||||
|
static char *pbf = &bf[0];
|
||||||
|
register char *c = buf;
|
||||||
|
|
||||||
|
|
||||||
|
while (--count) {
|
||||||
|
if (pbf >= &bf[cnt]) {
|
||||||
|
if ((cnt = read(fd, bf, 1024)) <= 0) {
|
||||||
|
if (c == buf) return (char *) NULL;
|
||||||
|
*c = 0;
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
pbf = &bf[0];
|
||||||
|
}
|
||||||
|
*c = *pbf++;
|
||||||
|
if (*c++ == '\n') {
|
||||||
|
*--c = 0;
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*c = 0;
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* tgetent - get the termcap entry for terminal name, and put it
|
||||||
|
* in bp (which must be an array of 1024 chars). Returns 1 if
|
||||||
|
* termcap entry found, 0 if not found, and -1 if file not found.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
tgetent(bp, name)
|
||||||
|
char *bp;
|
||||||
|
char *name;
|
||||||
|
{
|
||||||
|
int fp;
|
||||||
|
char *file;
|
||||||
|
char *cp;
|
||||||
|
short len = strlen(name);
|
||||||
|
char buf[1024];
|
||||||
|
|
||||||
|
capab = bp;
|
||||||
|
if ((file = getenv("TERMCAP")) != (char *) NULL) {
|
||||||
|
if (*file != '/' &&
|
||||||
|
(cp = getenv("TERM")) != NULL && strcmp(name, cp) == 0) {
|
||||||
|
(void) strcpy(bp, file);
|
||||||
|
return(1);
|
||||||
|
}
|
||||||
|
else file = "/etc/termcap";
|
||||||
|
} else
|
||||||
|
file = "/etc/termcap";
|
||||||
|
if ((fp = open(file, 0)) < 0)
|
||||||
|
return(-1);
|
||||||
|
while (fgets(buf, 1024, fp) != NULL) {
|
||||||
|
if (buf[0] == '#') continue;
|
||||||
|
while (*(cp = &buf[strlen(buf) - 2]) == '\\')
|
||||||
|
if (fgets(cp, 1024, fp) == NULL)
|
||||||
|
return (0);
|
||||||
|
if (match_name(buf, name)) {
|
||||||
|
strcpy(bp, buf);
|
||||||
|
close(fp);
|
||||||
|
return(check_for_tc());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
close(fp);
|
||||||
|
return(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Compare the terminal name with each termcap entry name; Return 1 if a
|
||||||
|
* match is found.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
match_name(buf, name)
|
||||||
|
char *buf;
|
||||||
|
char *name;
|
||||||
|
{
|
||||||
|
register char *tp = buf;
|
||||||
|
register char *np;
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
for (np = name; *np && *tp == *np; np++, tp++) { }
|
||||||
|
if (*np == 0 && (*tp == '|' || *tp == ':' || *tp == 0))
|
||||||
|
return(1);
|
||||||
|
while (*tp != 0 && *tp != '|' && *tp != ':') tp++;
|
||||||
|
if (*tp++ != '|') return (0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Handle tc= definitions recursively.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
check_for_tc()
|
||||||
|
{
|
||||||
|
static int count = 0;
|
||||||
|
char *savcapab = capab;
|
||||||
|
char buf[1024];
|
||||||
|
char terminalname[128];
|
||||||
|
register char *p = capab + strlen(capab) - 2, *q;
|
||||||
|
|
||||||
|
while (*p != ':')
|
||||||
|
if (--p < capab)
|
||||||
|
return(0); /* no : in termcap entry */
|
||||||
|
if (p[1] != 't' || p[2] != 'c')
|
||||||
|
return(1);
|
||||||
|
if (count > 16) return(0); /* recursion in tc= definitions */
|
||||||
|
count++;
|
||||||
|
strcpy(terminalname, &p[4]);
|
||||||
|
q = terminalname;
|
||||||
|
while (*q && *q != ':') q++;
|
||||||
|
*q = 0;
|
||||||
|
if (tgetent(buf, terminalname) != 1) {
|
||||||
|
--count;
|
||||||
|
return(0);
|
||||||
|
}
|
||||||
|
--count;
|
||||||
|
for (q = buf; *q && *q != ':'; q++) { }
|
||||||
|
strcpy(p, q);
|
||||||
|
capab = savcapab;
|
||||||
|
return(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* tgetnum - get the numeric terminal capability corresponding
|
||||||
|
* to id. Returns the value, -1 if invalid.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
tgetnum(id)
|
||||||
|
char *id;
|
||||||
|
{
|
||||||
|
char *cp;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if ((cp = capab) == NULL || id == NULL)
|
||||||
|
return(-1);
|
||||||
|
while (*++cp != ':')
|
||||||
|
;
|
||||||
|
while (*cp) {
|
||||||
|
cp++;
|
||||||
|
while (ISSPACE(*cp))
|
||||||
|
cp++;
|
||||||
|
if (strncmp(cp, id, CAPABLEN) == 0) {
|
||||||
|
while (*cp && *cp != ':' && *cp != '#')
|
||||||
|
cp++;
|
||||||
|
if (*cp != '#')
|
||||||
|
return(-1);
|
||||||
|
for (ret = 0, cp++ ; *cp && ISDIGIT(*cp) ; cp++)
|
||||||
|
ret = ret * 10 + *cp - '0';
|
||||||
|
return(ret);
|
||||||
|
}
|
||||||
|
while (*cp && *cp != ':')
|
||||||
|
cp++;
|
||||||
|
}
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* tgetflag - get the boolean flag corresponding to id. Returns -1
|
||||||
|
* if invalid, 0 if the flag is not in termcap entry, or 1 if it is
|
||||||
|
* present.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
tgetflag(id)
|
||||||
|
char *id;
|
||||||
|
{
|
||||||
|
char *cp;
|
||||||
|
|
||||||
|
if ((cp = capab) == NULL || id == NULL)
|
||||||
|
return(-1);
|
||||||
|
while (*++cp != ':')
|
||||||
|
;
|
||||||
|
while (*cp) {
|
||||||
|
cp++;
|
||||||
|
while (ISSPACE(*cp))
|
||||||
|
cp++;
|
||||||
|
if (strncmp(cp, id, CAPABLEN) == 0)
|
||||||
|
return(1);
|
||||||
|
while (*cp && *cp != ':')
|
||||||
|
cp++;
|
||||||
|
}
|
||||||
|
return(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* tgetstr - get the string capability corresponding to id and place
|
||||||
|
* it in area (advancing area at same time). Expand escape sequences
|
||||||
|
* etc. Returns the string, or NULL if it can't do it.
|
||||||
|
*/
|
||||||
|
char *
|
||||||
|
tgetstr(id, area)
|
||||||
|
char *id;
|
||||||
|
char **area;
|
||||||
|
{
|
||||||
|
char *cp;
|
||||||
|
char *ret;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if ((cp = capab) == NULL || id == NULL)
|
||||||
|
return(NULL);
|
||||||
|
while (*++cp != ':')
|
||||||
|
;
|
||||||
|
while (*cp) {
|
||||||
|
cp++;
|
||||||
|
while (ISSPACE(*cp))
|
||||||
|
cp++;
|
||||||
|
if (strncmp(cp, id, CAPABLEN) == 0) {
|
||||||
|
while (*cp && *cp != ':' && *cp != '=')
|
||||||
|
cp++;
|
||||||
|
if (*cp != '=')
|
||||||
|
return(NULL);
|
||||||
|
for (ret = *area, cp++; *cp && *cp != ':' ; (*area)++, cp++)
|
||||||
|
switch(*cp) {
|
||||||
|
case '^' :
|
||||||
|
**area = *++cp - 'A' + 1;
|
||||||
|
break;
|
||||||
|
case '\\' :
|
||||||
|
switch(*++cp) {
|
||||||
|
case 'E' :
|
||||||
|
**area = '\033';
|
||||||
|
break;
|
||||||
|
case 'n' :
|
||||||
|
**area = '\n';
|
||||||
|
break;
|
||||||
|
case 'r' :
|
||||||
|
**area = '\r';
|
||||||
|
break;
|
||||||
|
case 't' :
|
||||||
|
**area = '\t';
|
||||||
|
break;
|
||||||
|
case 'b' :
|
||||||
|
**area = '\b';
|
||||||
|
break;
|
||||||
|
case 'f' :
|
||||||
|
**area = '\f';
|
||||||
|
break;
|
||||||
|
case '0' :
|
||||||
|
case '1' :
|
||||||
|
case '2' :
|
||||||
|
case '3' :
|
||||||
|
for (i=0 ; *cp && ISDIGIT(*cp) ; cp++)
|
||||||
|
i = i * 8 + *cp - '0';
|
||||||
|
**area = i;
|
||||||
|
cp--;
|
||||||
|
break;
|
||||||
|
case '^' :
|
||||||
|
case '\\' :
|
||||||
|
**area = *cp;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default :
|
||||||
|
**area = *cp;
|
||||||
|
}
|
||||||
|
*(*area)++ = '\0';
|
||||||
|
return(ret);
|
||||||
|
}
|
||||||
|
while (*cp && *cp != ':')
|
||||||
|
cp++;
|
||||||
|
}
|
||||||
|
return(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* tgoto - given the cursor motion string cm, make up the string
|
||||||
|
* for the cursor to go to (destcol, destline), and return the string.
|
||||||
|
* Returns "OOPS" if something's gone wrong, or the string otherwise.
|
||||||
|
*/
|
||||||
|
char *
|
||||||
|
tgoto(cm, destcol, destline)
|
||||||
|
char *cm;
|
||||||
|
int destcol;
|
||||||
|
int destline;
|
||||||
|
{
|
||||||
|
register char *rp;
|
||||||
|
static char ret[32];
|
||||||
|
char added[16];
|
||||||
|
int *dp = &destline;
|
||||||
|
int numval;
|
||||||
|
int swapped = 0;
|
||||||
|
|
||||||
|
added[0] = 0;
|
||||||
|
for (rp = ret ; *cm ; cm++) {
|
||||||
|
if (*cm == '%') {
|
||||||
|
switch(*++cm) {
|
||||||
|
case '>' :
|
||||||
|
if (dp == NULL)
|
||||||
|
return("OOPS");
|
||||||
|
cm++;
|
||||||
|
if (*dp > *cm++) {
|
||||||
|
*dp += *cm;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '+' :
|
||||||
|
case '.' :
|
||||||
|
if (dp == NULL)
|
||||||
|
return("OOPS");
|
||||||
|
if (*cm == '+') *dp = *dp + *++cm;
|
||||||
|
for (;;) {
|
||||||
|
switch(*dp) {
|
||||||
|
case 0:
|
||||||
|
case 04:
|
||||||
|
case '\t':
|
||||||
|
case '\n':
|
||||||
|
/* filter these out */
|
||||||
|
if (dp == &destcol || swapped || UP) {
|
||||||
|
strcat(added, dp == &destcol || swapped ?
|
||||||
|
(BC ? BC : "\b") :
|
||||||
|
UP);
|
||||||
|
(*dp)++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
*rp++ = *dp;
|
||||||
|
dp = (dp == &destline) ? &destcol : NULL;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'r' : {
|
||||||
|
int tmp = destline;
|
||||||
|
|
||||||
|
destline = destcol;
|
||||||
|
destcol = tmp;
|
||||||
|
swapped = 1 - swapped;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'n' :
|
||||||
|
destcol ^= 0140;
|
||||||
|
destline ^= 0140;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '%' :
|
||||||
|
*rp++ = '%';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'i' :
|
||||||
|
destcol++;
|
||||||
|
destline++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'B' :
|
||||||
|
if (dp == NULL)
|
||||||
|
return("OOPS");
|
||||||
|
*dp = 16 * (*dp / 10) + *dp % 10;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'D' :
|
||||||
|
if (dp == NULL)
|
||||||
|
return("OOPS");
|
||||||
|
*dp = *dp - 2 * (*dp % 16);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'd' :
|
||||||
|
case '2' :
|
||||||
|
case '3' :
|
||||||
|
if (dp == NULL)
|
||||||
|
return("OOPS");
|
||||||
|
numval = *dp;
|
||||||
|
dp = (dp == &destline) ? &destcol : NULL;
|
||||||
|
if (numval >= 100) {
|
||||||
|
*rp++ = '0' + numval / 100;
|
||||||
|
}
|
||||||
|
else if (*cm == '3') {
|
||||||
|
*rp++ = ' ';
|
||||||
|
}
|
||||||
|
if (numval >= 10) {
|
||||||
|
*rp++ = '0' + ((numval%100)/10);
|
||||||
|
}
|
||||||
|
else if (*cm == '3' || *cm == '2') {
|
||||||
|
*rp++ = ' ';
|
||||||
|
}
|
||||||
|
*rp++ = '0' + (numval%10);
|
||||||
|
break;
|
||||||
|
default :
|
||||||
|
return("OOPS");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else *rp++ = *cm;
|
||||||
|
}
|
||||||
|
*rp = '\0';
|
||||||
|
strcpy(rp, added);
|
||||||
|
return(ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int tens_of_ms_p_char[] = { /* index as returned by gtty */
|
||||||
|
/* assume 10 bits per char */
|
||||||
|
0, 2000, 1333, 909, 743, 666, 500, 333, 166, 83, 55, 41, 20, 10, 5, 2
|
||||||
|
};
|
||||||
|
/*
|
||||||
|
* tputs - put the string cp out onto the terminal, using the function
|
||||||
|
* outc. Also handle padding.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
tputs(cp, affcnt, outc)
|
||||||
|
register char *cp;
|
||||||
|
int affcnt;
|
||||||
|
int (*outc)();
|
||||||
|
{
|
||||||
|
int delay = 0;
|
||||||
|
if (cp == NULL)
|
||||||
|
return(1);
|
||||||
|
while (ISDIGIT(*cp)) {
|
||||||
|
delay = delay * 10 + (*cp++ - '0');
|
||||||
|
}
|
||||||
|
delay *= 10;
|
||||||
|
if (*cp == '.') {
|
||||||
|
cp++;
|
||||||
|
if (ISDIGIT(*cp)) {
|
||||||
|
delay += *cp++ - '0';
|
||||||
|
}
|
||||||
|
while (ISDIGIT(*cp)) cp++;
|
||||||
|
}
|
||||||
|
if (*cp == '*') {
|
||||||
|
delay *= affcnt;
|
||||||
|
cp++;
|
||||||
|
}
|
||||||
|
while (*cp)
|
||||||
|
(*outc)(*cp++);
|
||||||
|
if (delay != 0 &&
|
||||||
|
ospeed > 0 &&
|
||||||
|
ospeed < (sizeof tens_of_ms_p_char / sizeof tens_of_ms_p_char[0])) {
|
||||||
|
delay = (delay + tens_of_ms_p_char[ospeed] - 1) /
|
||||||
|
tens_of_ms_p_char[ospeed];
|
||||||
|
while (delay--) (*outc)(PC);
|
||||||
|
}
|
||||||
|
return(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* That's all, folks...
|
||||||
|
*/
|
47
lang/m2/libm2/XXTermcap.def
Normal file
47
lang/m2/libm2/XXTermcap.def
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
(*
|
||||||
|
(c) copyright 1988 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
||||||
|
See the copyright notice in the ACK home directory, in the file "Copyright".
|
||||||
|
*)
|
||||||
|
|
||||||
|
(*
|
||||||
|
Module: Interface to termcap database
|
||||||
|
From: Unix manual chapter 3
|
||||||
|
Version: $Header$
|
||||||
|
*)
|
||||||
|
|
||||||
|
(*$Foreign*)
|
||||||
|
|
||||||
|
DEFINITION MODULE XXTermcap;
|
||||||
|
(* See the Unix termcap manual to see what this does.
|
||||||
|
Interfaces directly to C routines. Not pretty.
|
||||||
|
Use Termcap.def instead for a nicer interface.
|
||||||
|
*)
|
||||||
|
|
||||||
|
FROM SYSTEM IMPORT ADDRESS;
|
||||||
|
|
||||||
|
TYPE PUTPROC = PROCEDURE(CHAR);
|
||||||
|
|
||||||
|
VAR PC: CHAR;
|
||||||
|
UP, BC: ADDRESS;
|
||||||
|
ospeed: INTEGER[0..32767];
|
||||||
|
|
||||||
|
|
||||||
|
PROCEDURE tgetent(bp, name: ADDRESS): INTEGER;
|
||||||
|
(* name must be null-terminated *)
|
||||||
|
|
||||||
|
PROCEDURE tgetnum(id: ADDRESS): INTEGER;
|
||||||
|
(* id must be null-terminated *)
|
||||||
|
|
||||||
|
PROCEDURE tgetflag(id: ADDRESS): INTEGER;
|
||||||
|
(* id must be null-terminated *)
|
||||||
|
|
||||||
|
PROCEDURE tgetstr(id: ADDRESS; area: ADDRESS): ADDRESS;
|
||||||
|
(* id must be null-terminated *)
|
||||||
|
|
||||||
|
PROCEDURE tgoto(cm: ADDRESS; col, line: INTEGER) : ADDRESS;
|
||||||
|
(* cm must be null-terminated *)
|
||||||
|
|
||||||
|
PROCEDURE tputs(cp: ADDRESS; affcnt: INTEGER; p: PUTPROC);
|
||||||
|
(* cp must be null-terminated *)
|
||||||
|
|
||||||
|
END XXTermcap.
|
Loading…
Add table
Reference in a new issue