#include <stdio.h>
#include <stdarg.h>
#include <signal.h>
#include <ctype.h>
#include "dev.h"
#define NFONT 10
int output = 0;
int nolist = 0;
int olist[20];
int erase = 1;
float aspect = 1.5;
int wflag = 0;
void (*sigint)(int);
void (*sigquit)(int);
void done(void);
int error(int, char *, ...);
struct dev dev;
struct font *fontbase[NFONT];
short psizes[] ={ 11, 16, 22, 36, 0};
short *pstab = psizes;
int nsizes = 1;
int nfonts;
int smnt;
int nchtab;
char *chname;
short *chtab;
char *fitab[NFONT];
char *widthtab[NFONT];
char *codetab[NFONT];
#define FATAL 1
#define BMASK 0377
int dbg = 0;
int res = 972;
FILE *tf = stdout;
char *fontdir = "/usr/lib/font";
extern char devname[];
FILE *fp = stdin;
int nowait = 0;
int
main(int argc, char **argv)
{
char buf[BUFSIZ];
setbuf(stdout, buf);
while (argc > 1 && argv[1][0] == '-') {
switch (argv[1][1]) {
case 'a':
aspect = atof(&argv[1][2]);
break;
case 'e':
erase = 0;
break;
case 'o':
outlist(&argv[1][2]);
break;
case 'd':
dbg = atoi(&argv[1][2]);
if (dbg == 0) dbg = 1;
break;
case 'w':
nowait = 1;
break;
}
argc--;
argv++;
}
if (argc <= 1)
conv(stdin);
else
while (--argc > 0) {
if (strcmp(*++argv, "-") == 0)
fp = stdin;
else if ((fp = fopen(*argv, "r")) == NULL)
error(FATAL, "can't open %s", *argv);
conv(fp);
fclose(fp);
}
done();
return (0);
}
int
outlist(char *s)
{
int n1, n2, i;
nolist = 0;
while (*s) {
n1 = 0;
if (isdigit((unsigned char)*s))
do
n1 = 10 * n1 + *s++ - '0';
while (isdigit((unsigned char)*s));
else
n1 = -9999;
n2 = n1;
if (*s == '-') {
s++;
n2 = 0;
if (isdigit((unsigned char)*s))
do
n2 = 10 * n2 + *s++ - '0';
while (isdigit((unsigned char)*s));
else
n2 = 9999;
}
olist[nolist++] = n1;
olist[nolist++] = n2;
if (*s != '\0')
s++;
}
olist[nolist] = 0;
if (dbg)
for (i=0; i<nolist; i += 2)
printf("%3d %3d\n", olist[i], olist[i+1]);
return (0);
}
int
in_olist(int n)
{
int i;
if (nolist == 0)
return(1);
for (i = 0; i < nolist; i += 2)
if (n >= olist[i] && n <= olist[i+1])
return(1);
return(0);
}
int
conv(FILE *fp)
{
int c, k;
int m, n, i, n1, m1;
char str[100], buf[300];
while ((c = getc(fp)) != EOF) {
switch (c) {
case '\n':
case ' ':
case 0:
break;
case '{':
t_push();
break;
case '}':
t_pop();
break;
case '0': case '1': case '2': case '3': case '4':
case '5': case '6': case '7': case '8': case '9':
hmot((c-'0')*10 + getc(fp)-'0');
put1(getc(fp));
break;
case 'c':
put1(getc(fp));
break;
case 'C':
fscanf(fp, "%s", str);
put1s(str);
break;
case 't':
fgets(buf, sizeof(buf), fp);
t_text(buf);
break;
case 'D':
fgets(buf, sizeof(buf), fp);
switch (buf[0]) {
case 'l':
sscanf(buf+1, "%d %d", &n, &m);
drawline(n, m, ".");
break;
case 'c':
sscanf(buf+1, "%d", &n);
drawcirc(n);
break;
case 'e':
sscanf(buf+1, "%d %d", &m, &n);
drawellip(m, n);
break;
case 'a':
sscanf(buf+1, "%d %d %d %d", &n, &m, &n1, &m1);
drawarc(n, m, n1, m1);
break;
case '~':
drawwig(buf+1);
break;
default:
error(FATAL, "unknown drawing function %s\n", buf);
break;
}
break;
case 's':
fscanf(fp, "%d", &n);
setsize(t_size(n));
break;
case 'f':
fscanf(fp, "%s", str);
setfont(t_font(str));
break;
case 'H':
while ((c = getc(fp)) == ' ')
;
k = 0;
do {
k = 10 * k + c - '0';
} while (isdigit(c = getc(fp)));
ungetc(c, fp);
hgoto(k);
break;
case 'h':
while ((c = getc(fp)) == ' ')
;
k = 0;
do {
k = 10 * k + c - '0';
} while (isdigit(c = getc(fp)));
ungetc(c, fp);
hmot(k);
break;
case 'w':
putc(' ', stdout);
break;
case 'V':
fscanf(fp, "%d", &n);
vgoto(n);
break;
case 'v':
fscanf(fp, "%d", &n);
vmot(n);
break;
case 'p':
fscanf(fp, "%d", &n);
t_page(n);
break;
case 'n':
while (getc(fp) != '\n')
;
t_newline();
break;
case '#':
while (getc(fp) != '\n')
;
break;
case 'x':
devcntrl(fp);
break;
default:
error(!FATAL, "unknown input character %o %c\n", c, c);
done();
}
}
return (0);
}
int
devcntrl(FILE *fp)
{
char str[20];
int c, n;
fscanf(fp, "%s", str);
switch (str[0]) {
case 'i':
fileinit();
t_init(0);
break;
case 'T':
fscanf(fp, "%s", devname);
break;
case 't':
t_trailer();
break;
case 'p':
t_reset('p');
break;
case 's':
t_reset('s');
break;
case 'r':
fscanf(fp, "%d", &res);
break;
case 'f':
fscanf(fp, "%d %s", &n, str);
loadfont(n, str);
break;
}
while (getc(fp) != '\n')
;
return (0);
}
int
fileinit(void)
{
return (0);
}
int
fontprint(int i)
{
return (0);
}
int
loadcode(int n, int nw)
{
return (0);
}
int
loadfont(int n, char *s)
{
return (0);
}
int
error(int f, char *fmt, ...)
{
va_list ap;
fprintf(stderr, "ta: ");
va_start(ap, fmt);
(void) vfprintf(stderr, fmt, ap);
va_end(ap);
fprintf(stderr, "\n");
if (f)
exit(1);
return (0);
}
char devname[20] = "hp2621";
#define ESC 033
#define HOME 'H'
#define CLEAR 'J'
#define FF 014
int size = 1;
int font = 1;
int hpos;
int vpos;
int horig;
int vorig;
int DX = 10;
int DY = 10;
int drawdot = '.';
int drawsize = 1;
int
t_init(int reinit)
{
int i, j;
fflush(stdout);
hpos = vpos = 0;
return (0);
}
#define MAXSTATE 5
struct state {
int ssize;
int sfont;
int shpos;
int svpos;
int shorig;
int svorig;
};
struct state state[MAXSTATE];
struct state *statep = state;
int
t_push(void)
{
hflush();
statep->ssize = size;
statep->sfont = font;
statep->shorig = horig;
statep->svorig = vorig;
statep->shpos = hpos;
statep->svpos = vpos;
horig = hpos;
vorig = vpos;
hpos = vpos = 0;
if (statep++ >= state+MAXSTATE)
error(FATAL, "{ nested too deep");
hpos = vpos = 0;
return (0);
}
int
t_pop(void)
{
if (--statep < state)
error(FATAL, "extra }");
size = statep->ssize;
font = statep->sfont;
hpos = statep->shpos;
vpos = statep->svpos;
horig = statep->shorig;
vorig = statep->svorig;
return (0);
}
int np;
int npmax;
int pgnum[40];
long pgadr[40];
int
t_page(int n)
{
long ftell();
int c, m, i;
char buf[100], *bp;
pgnum[np++] = n;
pgadr[np] = ftell(fp);
if (np > npmax)
npmax = np;
if (output == 0) {
output = in_olist(n);
t_init(1);
return (0);
}
putpage();
fflush(stdout);
if (nowait)
return (0);
next:
for (bp = buf; (*bp = readch()); )
if (*bp++ == '\n')
break;
*bp = 0;
switch (buf[0]) {
case 0:
done();
break;
case '\n':
output = in_olist(n);
t_init(1);
return (0);
case '!':
callunix(&buf[1]);
fputs("!\n", stderr);
break;
case 'e':
erase = 1 - erase;
break;
case 'w':
wflag = 1 - wflag;
break;
case 'a':
aspect = atof(&buf[1]);
break;
case '-':
case 'p':
m = atoi(&buf[1]) + 1;
if (fp == stdin) {
fputs("you can't; it's not a file\n", stderr);
break;
}
if (np - m <= 0) {
fputs("too far back\n", stderr);
break;
}
np -= m;
fseek(fp, pgadr[np], 0);
output = 1;
t_init(1);
return (0);
case '0': case '1': case '2': case '3': case '4':
case '5': case '6': case '7': case '8': case '9':
m = atoi(&buf[0]);
for (i = 0; i < npmax; i++)
if (m == pgnum[i])
break;
if (i >= npmax || fp == stdin) {
fputs("you can't\n", stderr);
break;
}
np = i + 1;
fseek(fp, pgadr[np], 0);
output = 1;
t_init(1);
return (0);
case 'o':
outlist(&buf[1]);
output = 0;
t_init(1);
return (0);
case '?':
fputs("!cmd unix cmd\n", stderr);
fputs("p print this page again\n", stderr);
fputs("-n go back n pages\n", stderr);
fputs("n print page n (previously printed)\n", stderr);
fputs("o... set the -o output list to ...\n", stderr);
fputs("en n=0 -> don't erase; n=1 -> erase\n", stderr);
fputs("an sets aspect ratio to n\n", stderr);
break;
default:
fputs("?\n", stderr);
break;
}
goto next;
}
int
putpage(void)
{
int i, j, k;
fflush(stdout);
return (0);
}
int
t_newline(void)
{
printf("\n");
hpos = 0;
return (0);
}
int
t_size(int n)
{
return (0);
}
int
t_font(char *s)
{
return (0);
}
int
t_text(char *s)
{
int c, w=0;
char str[100];
if (!output)
return (0);
while ((c = *s++) != '\n') {
if (c == '\\') {
switch (c = *s++) {
case '\\':
case 'e':
put1('\\');
break;
case '(':
str[0] = *s++;
str[1] = *s++;
str[2] = '\0';
put1s(str);
break;
}
} else {
put1(c);
}
hmot(w);
}
return (0);
}
int
t_reset(int c)
{
int n;
output = 1;
fflush(stdout);
if (c == 's')
t_page(9999);
return (0);
}
int
t_trailer(void)
{
return (0);
}
int
hgoto(int n)
{
hpos = n;
return (0);
}
int
hmot(int n)
{
hgoto(hpos + n);
return (0);
}
int
hflush(void)
{
return (0);
}
int
vgoto(int n)
{
vpos = n;
return (0);
}
int
vmot(int n)
{
vgoto(vpos + n);
return (0);
}
int
put1s(char *s)
{
int i;
char *p;
extern char *spectab[];
static char prev[10] = "";
static int previ;
if (!output)
return (0);
if (strcmp(s, prev) != 0) {
previ = -1;
for (i = 0; spectab[i] != 0; i += 2)
if (strcmp(spectab[i], s) == 0) {
strcpy(prev, s);
previ = i;
break;
}
}
if (previ >= 0) {
for (p = spectab[previ+1]; *p; p++)
putc(*p, stdout);
} else
prev[0] = 0;
return (0);
}
int
put1(int c)
{
if (!output)
return (0);
putc(c, stdout);
return (0);
}
int
setsize(int n)
{
return (0);
}
int
t_fp(int n, char *s)
{
return (0);
}
int
setfont(int n)
{
return (0);
}
void done(void)
{
output = 1;
putpage();
fflush(stdout);
exit(0);
}
int
callunix(char line[])
{
int rc, status, unixpid;
if( (unixpid=fork())==0 ) {
signal(SIGINT,sigint); signal(SIGQUIT,sigquit);
close(0); dup(2);
execl("/bin/sh", "-sh", "-c", line, 0);
exit(255);
}
else if(unixpid == -1)
return (0);
else{ signal(SIGINT, SIG_IGN); signal(SIGQUIT, SIG_IGN);
while( (rc = wait(&status)) != unixpid && rc != -1 ) ;
signal(SIGINT,(void(*)())done); signal(SIGQUIT,sigquit);
}
return (0);
}
int
readch(void)
{
char c;
if (read(2,&c,1)<1) c=0;
return(c);
}
char *spectab[] ={
"em", "-",
"hy", "-",
"en", "-",
"ru", "_",
"l.", ".",
"br", "|",
"vr", "|",
"fm", "'",
"or", "|",
0, 0,
};