/*
* dvi2tty
* Copyright (C) 2003 Marcel J.E. Mol <
[email protected]>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
*/
/*
* Include files
*/
#include "dvi2tty.h"
#if defined(VMS)
# include types.h
# include stat
#else
# if defined(THINK_C)
# include <unix.h>
# else
# include <sys/types.h>
# include <sys/stat.h>
# endif
#endif
#if defined(MSDOS) || defined(THINK_C)
# include <math.h>
#endif
#include "commands.h"
#include "tex2ucs.h"
/*
* Constant definitions
*/
#if defined(VMS)
#define mseek vmsseek
#define ROUND(a) (a>=0.0 ? (int) (a + 0.5) : (int) (a - 0.5) )
#else
#define mseek fseek
#endif
#define VERSIONID 2 /* dvi version number that pgm handles */
#define VERTICALEPSILON 450000L /* crlf when increasing v more than this */
#define rightmargin MAXTERMWIDTH+20
/* nr of columns allowed to the right of h=0*/
#define leftmargin -50 /* give some room for negative h-coordinate */
#define LINELEN rightmargin - leftmargin + 1
#define MOVE TRUE /* if advancing h when outputing a rule */
#define STAY FALSE /* if not advancing h when outputing a rule */
#define absolute 0 /* for seeking in files */
#define relative 1
#define FORM 12 /* formfeed */
#define SPACE 32 /* space */
#define DEL 127 /* delete */
#define LASTCHAR 127 /* max dvi character, above are commands */
#define LASTCHAR8B 255
#define IS_UNICODE 0x400000 /* flag for unicode */
#define MAX_UNICODE 0x10FFFF /* max unicode */
#define IMIN(a, b) (a<b ? a : b)
#define IMAX(a, b) (a>b ? a : b)
#define get1() num(1)
#define get2() num(2)
#define get3() num(3)
#define get4() num(4)
#define sget1() snum(1)
#define sget2() snum(2)
#define sget3() snum(3)
#define sget4() snum(4)
/*
* Structure and variable definitions
*/
const char *dvistuff = "@(#) dvistuff.c " VERSION " 20101027 M.J.E. Mol (c) 1989-2010";
typedef struct {
long hh;
long vv;
long ww;
long xx;
long yy;
long zz;
} stackitem;
typedef struct lineptr { /* the lines of text to be output to outfile */
long vv; /* vertical position of the line */
int charactercount; /* pos of last char on line */
struct lineptr *prev; /* preceding line */
struct lineptr *next; /* succeeding line */
long text[LINELEN+1]; /* leftmargin...rightmargin */
} linetype;
typedef struct _font {
long num;
struct _font * next;
char * name;
unsigned char flags; /* to store font encoding types */
int fontnum; /* helper for japanese fonts */
bool is8bit; /* 8bit fonts */
} font;
#define TTFONT 0x01
#define SYMFONT 0x02
#define MIFONT 0x03
#define T1FONT 0x04
#define TS1FONT 0x05
#define OT2FONT 0x10
#define T2AFONT 0x11
#define T2BFONT 0x12
#define T2CFONT 0x13
#define X2FONT 0x14
#define JPFONT 0x80
bool pageswitchon; /* true if user-set pages to print */
bool sequenceon; /* false if pagesw-nrs refers to TeX-nrs */
bool scascii; /* if true make Scand. nat. chars right */
bool latin1; /* if true make latin1 chars right */
bool utf8; /* if true print by utf8 encoding */
bool noligaturefi; /* if true do not use ligature for ff,fi,fl,ffi,ffl */
bool accent; /* if true output accents etc: \'{e} etc. */
bool ttfont = FALSE; /* if true we assumed ttfonts, not cmr */
bool symbolfont = FALSE; /* true if font is a symbol font */
bool nttj = FALSE; /* switch to NTT japanese fonts ... */
bool asciip = FALSE; /* switch to ASCII japanese fonts ... */
bool uptex = FALSE; /* switch to upTeX CJK fonts ... */
bool japan = FALSE; /* switch to NTT/ASCII/.. japanese fonts ... */
bool jautodetect = FALSE; /* switch if do auto detection of Japanese TeX */
bool jdetect = FALSE; /* switch if Japanese TeX detection is done */
bool mifont = FALSE; /* ASCII japanese font ??? */
bool is8bit = FALSE; /* true if 8bit encoding font */
bool noffd; /* if true output ^L instead of formfeed */
const char *delim; /* -bdelim for font switch printing */
bool printfont; /* true if user wants font switches printed */
bool compose; /* if true try to compose a combining character sequence */
bool allchar; /* true if user sets all characters */
/* overrides sscasci, accent */
int opcode; /* dvi-opcodes */
long h, v; /* coordinates, horizontal and vertical */
long w, x, y, z; /* horizontal and vertical amounts */
long pagecounter; /* sequence page number counter */
long backpointer; /* pointer for offset to previous page */
long pagenr; /* TeX page number */
int stackmax; /* stacksize required */
long maxpagewidth; /* width of widest page in file */
long charwidth; /* aprox width of character */
long lineheight = VERTICALEPSILON;
/* aprox height of a line */
linetype * currentline; /* pointer to current line on current page */
linetype * firstline; /* pointer to first line on current page */
linetype * lastline; /* pointer to last line on current page */
int firstcolumn; /* 1st column with something to print */
stackitem * stack; /* stack for dvi-pushes */
int sptr; /* stack pointer */
font * fonts = NULL; /* List of fontnames defined */
font * fnt = NULL; /* Current font */
int kanji1 = 0; /* number of rest of trailer bytes in kanji character */
/*
* Function definitions
*/
#if defined(MSDOS)
void postamble (void);
void preamble (void);
void walkpages (void);
void initpage (void);
void dopage (void);
void skippage (void);
void printpage (void);
bool inlist (long);
void rule (bool, long, long);
void ruleaux (long, long, char);
long horizontalmove (long);
int skipnops (void);
linetype * my_getline (void);
linetype * findline (void);
unsigned long num (int);
long snum (int);
void dochar (unsigned char);
void symchar (unsigned char);
void michar (unsigned char);
void normchar (char, unsigned char);
void t1char (unsigned char);
void ts1char (unsigned char);
void ot2char (unsigned char);
void t2char (char, unsigned char);
void outchar (long);
void putcharacter (long);
void setchar (long);
void fontdef (int);
void setfont (long);
void jischar (unsigned long);
int compute_jis (int, unsigned int, unsigned int *, unsigned int *);
void dounichar (long);
void dokanji (long);
int getjsubfont (char *);
#else
void postamble (void);
void preamble (void);
void walkpages (void);
void initpage (void);
void dopage (void);
void skippage (void);
void printpage (void);
bool inlist (long pagenr);
void rule (bool moving, long rulewt, long ruleht);
void ruleaux (long rulewt, long ruleht, char ch);
long horizontalmove (long amount);
int skipnops (void);
linetype * my_getline (void);
linetype * findline (void);
unsigned long num (int size);
long snum (int size);
void dochar (unsigned char ch);
void symchar (unsigned char ch);
void michar (unsigned char ch);
void normchar (char flag, unsigned char ch);
void t1char (unsigned char ch);
void ts1char (unsigned char ch);
void ot2char (unsigned char ch);
void t2char (char flag, unsigned char ch);
void outchar (long ch);
void putcharacter (long charnr);
void setchar (long charnr);
void fontdef (int x);
void setfont (long fntnum);
void jischar (unsigned long ch);
void compute_jis (int f, unsigned int c, unsigned int * ku, unsigned int * ten);
void dounichar (long ch);
void dokanji (long ch);
int getjsubfont (char * s);
#if defined(VMS)
long vmsseek ();
long vms_ftell ();
long vms_ungetc ();
#endif
#endif
/*
* DVIMAIN -- The main function for processing the dvi file.
* Here we assume there are to file pointers: DVIfile and output.
* Also we have a list of pages pointed to by 'currentpage',
* which is only used (in 'inlist()') when a page list is given.
*/
void dvimain(void)
{
postamble(); /* seek and process the postamble */
preamble(); /* process preamble */
/* note that walkpages *must* immediately follow preamble */
walkpages(); /* time to do the actual work! */
return;
} /* dvimain */
/*
* POSTAMBLE -- Find and process postamble, use random access
*/
void postamble(void)
{
register long size;
register int count;
#if !defined (THINK_C) && defined(VMS)
struct stat st;
#endif
#if defined (THINK_C)
size = DVIfile->len;
#elif defined(VMS)
fstat (fileno(DVIfile), &st);
size = (long) st.st_size; /* get size of file */
#else
fseek (DVIfile, 0L, SEEK_END);
size = ftell (DVIfile); /* get size of file */
#endif
count = -1;
do { /* back file up past signature bytes (223), to id-byte */
if (size-- == 0)
errorexit(nopst);
mseek(DVIfile, size, absolute);
opcode = (int) get1();
count++;
} while (opcode == TRAILER);
if (count < 4) { /* must have 4 trailer bytes */
foo = count;
errorexit(fwsgn);
}
if (opcode != VERSIONID)
errorexit(badid);
mseek(DVIfile, size-4, absolute); /* back up to back-pointer */
mseek(DVIfile, sget4(), absolute); /* and to start of postamble */
if (get1() != POST)
errorexit(nopst);
mseek(DVIfile, 20L, relative); /* lastpageoffset, numerator, denominator */
/* magnification, maxpageheight */
maxpagewidth = sget4();
charwidth = maxpagewidth / (ttywidth + espace);
stackmax = (int) get2();
if ((stack = (stackitem *) malloc(stackmax * sizeof(stackitem))) == NULL)
errorexit(stkrq);
/* get2() -- totalpages */
/* fontdefs do fontdefs in flight ... */
return;
} /* postamble */
/*
* PREAMBLE --process preamble, use random access
*/
void preamble(void)
{
mseek(DVIfile, 0L, absolute); /* read the dvifile from the start */
if ((opcode = skipnops()) != PRE)
errorexit(nopre);
opcode = (int) get1(); /* check id in preamble, ignore rest of it */
if (opcode != VERSIONID)
errorexit(badid);
mseek(DVIfile, 12L, relative); /* numerator, denominator, magnification */
mseek(DVIfile, get1(), relative); /* skip job identification */
return;
} /* preamble */
/*
* WALKPAGES -- process the pages in the DVI-file
*/
void walkpages(void)
{
register bool wantpage;
pagecounter = 0L;
while ((opcode = skipnops()) != POST) {
if (opcode != BOP) /* should be at start of page now */
errorexit(nobop);
pagecounter++;
pagenr = sget4(); /* get TeX page number */
mseek(DVIfile, 36L, relative); /* skip page header */
backpointer = sget4(); /* get previous page offset */
if (pageswitchon)
wantpage = inlist(sequenceon ? pagecounter : pagenr);
else
wantpage = TRUE;
if (wantpage) {
initpage();
dopage();
printpage();
}
else
skippage();
}
return;
} /* walkpages */
/*
* INITPAGE -- Setup a new, empty page.
*/
void initpage(void)
{
h = 0L; v = 0L; /* initialize coordinates */
x = 0L; w = 0L; y = 0L; z = 0L; /* initialize amounts */
sptr = 0; /* initialize stack */
currentline = my_getline(); /* initialize list of lines */
currentline->vv = 0L;
firstline = currentline;
lastline = currentline;
firstcolumn = rightmargin;
if (pageswitchon) {
if ((sequenceon ? pagecounter : pagenr) != firstpage->pag) {
if (noffd)
fprintf(output, "^L\n");
else
putc(FORM, output);
}
}
else
if (backpointer != -1) { /* not FORM at first page */
if (noffd)
fprintf(output, "^L\n");
else
putc(FORM, output);
}
return;
} /* initpage */
/*
* DOPAGE -- Process the dvi file until an end-off-page.
* Build up a page image.
*/
void dopage(void)
{
while ((opcode = (int) get1()) != EOP) { /* process page until eop */
if (opcode <= LASTCHAR)
dochar((unsigned char) opcode);
else if ((opcode >= FONT_00) && (opcode <= FONT_63))
setfont((long) opcode - FONT_00);
else if (opcode > POST_POST)
errorexit(illop);
else
switch (opcode) {
case SET1 : nttj ? jischar(get1()) : setchar(get1());break;
case SET2 : (asciip || uptex) ? dokanji(get2()) : setchar(get2()); break;
case SET3 : uptex ? dokanji(get3()) : setchar(get3()); break;
case SET4 : setchar(get4()); break;
case SET_RULE : { long height = sget4();
rule(MOVE, sget4(), height); break;
}
case PUT1 : putcharacter(get1()); break;
case PUT2 : putcharacter(get2()); break;
case PUT3 : putcharacter(get3()); break;
case PUT4 : putcharacter(get4()); break;
case PUT_RULE : { long height = sget4();
rule(STAY, sget4(), height); break;
}
case NOP : break; /* no-op */
case BOP : errorexit(bdbop); break;
/* case EOP : break; strange place to have EOP */
case PUSH : if (sptr >= stackmax) /* push */
errorexit(stkof);
stack[sptr].hh = h;
stack[sptr].vv = v;
stack[sptr].ww = w;
stack[sptr].xx = x;
stack[sptr].yy = y;
stack[sptr].zz = z;
sptr++;
break;
case POP : if (sptr-- == 0) /* pop */
errorexit(stkuf);
h = stack[sptr].hh;
v = stack[sptr].vv;
w = stack[sptr].ww;
x = stack[sptr].xx;
y = stack[sptr].yy;
z = stack[sptr].zz;
break;
case RIGHT1 : (void) horizontalmove(sget1()); break;
case RIGHT2 : (void) horizontalmove(sget2()); break;
case RIGHT3 : (void) horizontalmove(sget3()); break;
case RIGHT4 : (void) horizontalmove(sget4()); break;
case W0 : h += w; break;
case W1 : w = horizontalmove(sget1()); break;
case W2 : w = horizontalmove(sget2()); break;
case W3 : w = horizontalmove(sget3()); break;
case W4 : w = horizontalmove(sget4()); break;
case X0 : h += x; break;
case X1 : x = horizontalmove(sget1()); break;
case X2 : x = horizontalmove(sget2()); break;
case X3 : x = horizontalmove(sget3()); break;
case X4 : x = horizontalmove(sget4()); break;
case DOWN1 : v += sget1(); break;
case DOWN2 : v += sget2(); break;
case DOWN3 : v += sget3(); break;
case DOWN4 : v += sget4(); break;
case Y0 : v += y; break;
case Y1 : y = sget1(); v += y; break;
case Y2 : y = sget2(); v += y; break;
case Y3 : y = sget3(); v += y; break;
case Y4 : y = sget4(); v += y; break;
case Z0 : v += z; break;
case Z1 : z = sget1(); v += z; break;
case Z2 : z = sget2(); v += z; break;
case Z3 : z = sget3(); v += z; break;
case Z4 : z = sget4(); v += z; break;
case FNT1 :
case FNT2 :
case FNT3 :
case FNT4 : setfont(num(opcode - FNT1 + 1));
break;
case XXX1 : mseek(DVIfile, get1(), relative); break;
case XXX2 : mseek(DVIfile, get2(), relative); break;
case XXX3 : mseek(DVIfile, get3(), relative); break;
case XXX4 : mseek(DVIfile, get4(), relative); break;
case FNT_DEF1 :
case FNT_DEF2 :
case FNT_DEF3 :
case FNT_DEF4 : fontdef(opcode - FNT_DEF1 + 1);
break;
case PRE : errorexit(bdpre); break;
case POST : errorexit(bdpst); break;
case POST_POST: errorexit(bdpp); break;
}
}
return;
} /* dopage */
/*
* SKIPPAGE -- Scan the dvi file until an end-off-page.
* Skip this page.
*/
void skippage(void)
{
register int opcode;
while ((opcode = (int) get1()) != EOP) {
if (opcode > POST_POST)
errorexit(illop);
else
switch (opcode) {
case SET1 :
case PUT1 :
case RIGHT1 :
case W1 :
case X1 :
case DOWN1 :
case Y1 :
case Z1 : /* assume FNT change can also be skipped */
case FNT1 : mseek(DVIfile, 1L, relative); break;
case SET2 :
case PUT2 :
case RIGHT2 :
case W2 :
case X2 :
case DOWN2 :
case Y2 :
case Z2 :
case FNT2 : mseek(DVIfile, 2L, relative); break;
case SET3 :
case PUT3 :
case RIGHT3 :
case W3 :
case X3 :
case DOWN3 :
case Y3 :
case Z3 :
case FNT3 : mseek(DVIfile, 3L, relative); break;
case SET4 :
case PUT4 :
case RIGHT4 :
case W4 :
case X4 :
case DOWN4 :
case Y4 :
case Z4 :
case FNT4 : mseek(DVIfile, 4L, relative); break;
case SET_RULE :
case PUT_RULE : mseek(DVIfile, 8L, relative); break;
case BOP : errorexit(bdbop); break;
case XXX1 : mseek(DVIfile, get1(), relative); break;
case XXX2 : mseek(DVIfile, get2(), relative); break;
case XXX3 : mseek(DVIfile, get3(), relative); break;
case XXX4 : mseek(DVIfile, get4(), relative); break;
case FNT_DEF1 :
case FNT_DEF2 :
case FNT_DEF3 :
case FNT_DEF4 : fontdef(opcode - FNT_DEF1 + 1); break;
case PRE : errorexit(bdpre); break;
case POST : errorexit(bdpst); break;
case POST_POST: errorexit(bdpp); break;
}
}
return;
} /* skippage */
/*
* PRINTPAGE -- 'end of page', writes lines of page to output file
*/
void printpage(void)
{
register int i, j, k;
register long ch, mbch;
unsigned char buff[4];
if (sptr != 0)
fprintf(stderr, "dvi2tty: warning - stack not empty at eop.\n");
for (currentline = firstline; currentline != nil;
currentline = currentline->next) {
if (currentline != firstline) {
foo = ((currentline->vv - currentline->prev->vv)/lineheight)-1;
if (foo > 3)
foo = 3; /* linespacings not too large */
for (i = 1; i <= (int) foo; i++)
putc('\n', output);
}
if (currentline->charactercount >= leftmargin) {
foo = ttywidth - 2;
for (i = firstcolumn, j = 1; i <= currentline->charactercount;
i++, j++) {
ch = currentline->text[i - leftmargin];
if (japan && !(ch & IS_UNICODE)) {
if (ch > 127) {
for (k = 0; k < 4; k++) {
if (i - leftmargin + k < LINELEN+1)
buff[k] = currentline->text[i - leftmargin + k];
else buff[k] = '\0';
}
kanji1 = multistrlen(buff, 4, 0) - 1;
}
else kanji1 = 0;
if (kanji1 && (j + kanji1 > (int) foo) &&
(currentline->charactercount > i+1)) {
putc2('*', output);
putc2('\n', output); /* if line to large */
putc2(' ', output);
putc2('*', output); /* mark output */
j = 2;
}
}
if (ch >= SPACE || allchar) {
if (utf8 && (ch & IS_UNICODE)) {
mbch = UCStoUTF8(ch & MAX_UNICODE);
if (BYTE1(mbch) != 0) putc((unsigned char)BYTE1(mbch), output);
if (BYTE2(mbch) != 0) putc((unsigned char)BYTE2(mbch), output);
if (BYTE3(mbch) != 0) putc((unsigned char)BYTE3(mbch), output);
/* always */ putc((unsigned char)BYTE4(mbch), output);
}
else if (japan) {
for (k = 0; k < kanji1; k++) {
putc2(ch, output);
i++; j++;
ch = currentline->text[i - leftmargin];
}
putc2(ch, output);
}
else
putc(ch, output);
}
if ((j > (int) foo) && (currentline->charactercount > i+1)) {
if (japan) {
putc2('*', output);
putc2('\n', output); /* if line to large */
putc2(' ', output);
putc2('*', output); /* mark output */
}
else {
fprintf(output, "*\n"); /* if line to large */
fprintf(output, " *"); /* mark output */
}
j = 2;
}
}
}
if (japan)
putc2('\n', output);
else
putc('\n', output);
}
currentline = firstline;
while (currentline->next != nil) {
currentline = currentline->next;
free(currentline->prev);
}
free(currentline); /* free last line */
currentline = nil;
return;
} /* printpage */
/*
* INLIST -- return true if pagenr is in the list of pages to be printed.
*/
bool inlist(long pagenr)
{
while ((currentpage->pag < 0) && (currentpage->pag != pagenr) &&
!currentpage->all && (currentpage->nxt != nil))
currentpage = currentpage->nxt;
if ((currentpage->all && (pagenr < currentpage->pag)) ||
(currentpage->pag == pagenr))
return TRUE;
else if (pagenr > 0) {
while ((currentpage->pag < pagenr) && (currentpage->nxt != nil))
currentpage = currentpage->nxt;
if (currentpage->pag == pagenr)
return TRUE;
}
return FALSE;
} /* inlist */
/*
* RULE -- Output a rule (vertical or horizontal).
* Increment h if moving is true.
*/
void rule(bool moving, long rulewt, long ruleht)
{
register char ch; /* character to set rule with */
register long saveh = 0, savev;
/* rule -- starts up the recursive routine */
if (!moving)
saveh = h;
if ((ruleht <= 0) || (rulewt <= 0))
h += rulewt;
else {
savev = v;
if ((ruleht / rulewt) > 0) /* value < 1 truncates to 0 */
ch = '|';
else if (ruleht > (lineheight / 2))
ch = '=';
else
ch = '_';
ruleaux(rulewt, ruleht, ch);
v = savev;
}
if (!moving)
h = saveh;
return;
} /* rule */
/*
* RULEAUX -- do the actual output for the rule recursively.
*/
void ruleaux(long rulewt, long ruleht, char ch)
{
register long wt, lmh, rmh;
wt = rulewt;
lmh = h; /* save left margin */
if (h < 0) { /* let rules that start at negative h */
wt -= h; /* start at coordinate 0, but let it */
h = 0; /* have the right length */
}
while (wt > 0) { /* output the part of the rule that */
rmh = h; /* goes on this line */
outchar(ch);
wt -= (h-rmh); /* decrease the width left on line */
}
ruleht -= lineheight; /* decrease the height */
if (ruleht > lineheight) { /* still more vertical? */
rmh = h; /* save current h (right margin) */
h = lmh; /* restore left margin */
v -= (lineheight + lineheight / 10);
ruleaux(rulewt, ruleht, ch);
h = rmh; /* restore right margin */
}
return;
} /* ruleaux */
/*
* HORIZONTALMOVE -- Move the h pointer by amount.
*/
long horizontalmove(long amount)
{
#if defined(MSDOS) || defined(THINK_C)
if (labs(amount) > charwidth / 4L) { /* } to make vi happy */
#else
if (abs(amount) > charwidth / 4L) {
#endif
foo = 3*charwidth / 4;
if (amount > 0)
amount = ((amount+foo) / charwidth) * charwidth;
else
#if defined(VMS)
amount = (ROUND( (float) (amount-foo) / charwidth) + 1)* charwidth;
#else
amount = ((amount-foo) / charwidth) * charwidth;
#endif
h += amount;
return amount;
}
else
return 0;
} /* horizontalmove */
/*
* SKIPNOPS -- Return first non NOP opcode.
*/
int skipnops(void)
{
register int opcode;
while ((opcode = (int) num(1)) == NOP);
return opcode;
} /* skipnops */
/*
* GETLINE -- Returns an initialized line-object
*/
linetype *my_getline(void)
{
register int i;
register linetype *temp;
if ((temp = (linetype *) malloc(sizeof(linetype))) == NULL)
errorexit(lnerq);
temp->charactercount = leftmargin - 1;
temp->prev = nil;
temp->next = nil;
for (i = 0; i < LINELEN; i++)
temp->text[i] = ' ';
temp->text[i] = '\0';
return temp;
} /* my_getline */
/*
* FINDLINE -- Find best fit line were text should go
* and generate new line if needed.
*/
linetype *findline(void)
{
register linetype *temp;
register long topd, botd;
if (v <= firstline->vv) { /* above first line */
if (firstline->vv - v > lineheight) {
temp = my_getline();
temp->next = firstline;
firstline->prev = temp;
temp->vv = v;
firstline = temp;
}
return firstline;
}
if (v >= lastline->vv) { /* below last line */
if (v - lastline->vv > lineheight) {
temp = my_getline();
temp->prev = lastline;
lastline->next = temp;
temp->vv = v;
lastline = temp;
}
return lastline;
}
temp = lastline; /* in between two lines */
while ((temp->vv > v) && (temp != firstline))
temp = temp->prev;
/* temp->vv < v < temp->next->vv --- temp is above, temp->next is below */
topd = v - temp->vv;
botd = temp->next->vv - v;
if ((topd < lineheight) || (botd < lineheight)) {
if (topd < botd) /* take best fit */
return temp;
else
return temp->next;
}
/* no line fits suitable, generate a new one */
currentline = my_getline();
currentline->next = temp->next;
currentline->prev = temp;
temp->next->prev = currentline;
temp->next = currentline;
currentline->vv = v;
return currentline;
} /* findline */
/*
* NUM --
*/
unsigned long num(int size)
{
register int i;
register unsigned long x = 0;
for (i = size; i > 0; i--)
x = (x << 8) + (unsigned) getc(DVIfile);
return x;
} /* num */
/*
* SNUM --
*/
long snum(int size)
{
register int i;
register long x;
x = getc(DVIfile);
if (x & 0x80)
x -= 0x100;
for (i = size - 1; i > 0; i--)
x = (x << 8) + (unsigned) getc(DVIfile);
return x;
} /* snum */
/*
* DOUNICHAR -- Process a Unicode character
*/
void dounichar(long ch)
{
unsigned char c[4] = {}, *cc;
if (noligaturefi && 0xFB00<=ch && ch<=0xFB04) {
switch (ch) {
case 0xFB00: strcpy(c,"ff"); break;
case 0xFB01: strcpy(c,"fi"); break;
case 0xFB02: strcpy(c,"fl"); break;
case 0xFB03: strcpy(c,"ffi"); break;
case 0xFB04: strcpy(c,"ffl"); break;
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
}
if (ch>0x7F)
outchar((long)(ch | IS_UNICODE));
else {
outchar((long)ch);
}
return;
} /* dounichar */
/*
* DOKANJI -- Process a kanji character opcode.
*/
void dokanji(long ch)
{
long i;
i = toBUFF(fromDVI(ch));
kanji1 = 3;
if (BYTE1(i) != 0) outchar((long)BYTE1(i));
kanji1 = 2;
if (BYTE2(i) != 0) outchar((long)BYTE2(i));
kanji1 = 1;
if (BYTE3(i) != 0) outchar((long)BYTE3(i));
kanji1 = 0;
/* always */ outchar((long)BYTE4(i));
return;
} /* dokanji */
/*
* DOCHAR -- Process a character opcode.
*/
void dochar(unsigned char ch)
{
char flag;
flag = fnt->flags;
if (nttj && fnt->fontnum)
jischar((long) ch);
else if (symbolfont)
symchar(ch);
else if (mifont)
michar(ch);
else if (flag == T1FONT)
t1char(ch);
else if (flag == TS1FONT)
ts1char(ch);
else if (flag == OT2FONT)
ot2char(ch);
else if (flag == T2AFONT || flag == T2BFONT ||
flag == T2CFONT || flag == X2FONT)
t2char(flag, ch);
else
normchar(flag, ch);
return;
} /* dochar */
/*
* SYMCHAR -- Process a character opcode for a symbol font.
*/
void symchar(unsigned char ch)
{
unsigned char c[4] = {}, *cc;
long ucs;
ucs = oms_to_ucs[ch];
if (utf8) {
dounichar(ucs);
return;
}
else if ((latin1 && ucs<0x100) || ucs<0x80) {
outchar(ucs);
return;
}
switch (ch) { /* can do a lot more on MSDOS/latin1/unicode machines ... */
case 0: c[0] = '-'; break;
case 1: c[0] = '.'; break;
case 2: c[0] = 'x'; break;
case 3: c[0] = '*'; break;
case 4: c[0] = '/'; break;
case 6: c[0] = '+'; c[1] = '-'; break;
case 7: c[0] = '-'; c[1] = '+'; break;
case 13: c[0] = 'O'; break;
case 14: c[0] = 'O'; break;
case 15: c[0] = 'o'; break;
case 24: c[0] = '~'; break;
case 28: c[0] = '<'; c[1] = '<'; break;
case 29: c[0] = '>'; c[1] = '>'; break;
case 32: c[0] = '<'; c[1] = '-'; break;
case 33: c[0] = '-'; c[1] = '>'; break;
case 34: c[0] = '^'; break;
case 35: c[0] = 'v'; break;
case 36: c[0] = '<'; c[1] = '-'; c[2] = '>'; break;
case 40: c[0] = '<'; c[1] = '='; break;
case 41: c[0] = '='; c[1] = '>'; break;
case 42: c[0] = '^'; break;
case 43: c[0] = 'v'; break;
case 44: c[0] = '<'; c[1] = '='; c[2] = '>'; break;
case 60: c[0] = 'R'; c[1] = 'e'; break;
case 61: c[0] = 'I'; c[1] = 'm'; break;
case 102: c[0] = '{'; break;
case 103: c[0] = '}'; break;
case 104: c[0] = '<'; break;
case 105: c[0] = '>'; break;
case 106: c[0] = '|'; break;
case 107: c[0] = '|'; c[1] = '|'; break;
case 110: c[0] = '\\'; break;
case 120: c[0] = 'S'; break;
case 121: c[0] = '*'; break;
case 122: c[0] = '*'; c[1] = '*'; break;
case 123: c[0] = 'P'; break;
default: c[0] = '#';
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
} /* symchar */
/*
* MICHAR -- Process a character opcode for OML font.
*/
void michar(unsigned char ch)
{
unsigned char c[4] = {}, *cc;
long ucs;
if (allchar) {
outchar(ch);
return;
}
ucs = oml_to_ucs[ch];
if (utf8) {
dounichar(ucs);
return;
}
else if ((latin1 && ucs<0x100) || ucs<0x80) {
outchar(ucs);
return;
}
switch (ch) {
case 0x3a: c[0] = '.'; break; /* . */
case 0x3b: c[0] = ','; break; /* , */
case 0x3d: c[0] = '/'; break; /* / */
case 0x3e: c[0] = '*'; break; /* \star */
case 0x40: c[0] = 'd'; break; /* \partial */
case 0x60: c[0] = 'l'; break; /* \ell */
case 0x7b: c[0] = 'i'; break; /* dotless i */
case 0x7c: c[0] = 'j'; break; /* dotless j */
case 0x7d: c[0] = 'P'; break; /* \wp */
default : c[0] = '#';
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
} /* michar */
/*
* NORMCHAR -- Process a character opcode for a normal font.
*/
void normchar(char flag, unsigned char ch)
{
unsigned char c[4] = {}, *cc;
const unsigned short *tex_to_ucs;
long ucs;
if (allchar) {
outchar(ch);
return;
}
if (!accent) {
switch (ch) {
case 18 : /* grave from \` */
case 19 : /* acute from \' */
case 20 : /* caron from \v */
case 21 : /* breve from \u */
case 22 : /* macron from \= */
case 23 : /* ring above from \r */
case 24 : /* cedilla from \c */
case 32 : /* stroke i.e. \L,\l */
case 94 : /* circumflex from \^ */
case 126 : /* tilde from \~ */
case 127 : /* diaeresis from \" */
return;
case 125 : /* double acute from \H */
case 95 : /* dot from \. */
if (!ttfont) return;
}
}
switch (flag) {
case TTFONT : tex_to_ucs=tt_to_ucs; break;
default : tex_to_ucs=ot1_to_ucs;
}
ucs = tex_to_ucs[ch];
if (utf8) {
dounichar(ucs);
return;
}
else if ((latin1 && ucs<0x100) || ucs<0x80) {
outchar(ucs);
return;
}
switch (ch) {
case 11 : if (ttfont)
c[0] = '^'; /* up symbol */
else {
c[0] = 'f'; c[1] = 'f'; /* ligature */
}
break;
case 12 : if (ttfont)
c[0] = 'v'; /* low symbol */
else {
c[0] = 'f'; c[1] = 'i'; /* ligature */
}
break;
case 13 : if (ttfont)
c[0] = '`';
else {
c[0] = 'f'; c[1] = 'l'; /* ligature */
}
break;
case 14 : if (ttfont)
c[0] = 'i'; /* spanish ! */
else {
c[0] = 'f'; c[1] = 'f';
c[2] = 'i'; /* ligature */
}
break;
case 15 : if (ttfont)
c[0] = '.'; /* spanish ? */
else {
c[0] = 'f'; c[1] = 'f';
c[2] = 'l'; /* ligature */
}
break;
case 16 : c[0] = 'i'; break;
case 17 : c[0] = 'j'; break;
case 25 : if (latin1)
c[0] = 0xdf;
else {
c[0] = 's'; c[1] = 's';
}
break; /* German double s */
case 26 : if (latin1)
c[0] = 0xe6;
else {
c[0] = 'a'; c[1] = 'e';
}
break; /* Dane/Norw ae */
case 27 : c[0] = 'o'; c[1] = 'e';
break; /* Dane/Norw oe */
case 28 : if (scascii)
c[0] = '|';
else if (latin1)
c[0] = 0xf8;
else
c[0] = 'o';
break; /* Dane/Norw /o */
case 29 : if (latin1)
c[0] = 0xc6;
else {
c[0] = 'A'; c[1] = 'E';
}
break; /* Dane/Norw AE */
case 30 : c[0] = 'O'; c[1] = 'E';
break; /* Dane/Norw OE */
case 31 : if (scascii)
c[0] = '\\';
else if (latin1)
c[0] = 0xd8;
else
c[0] = 'O';
break; /* Dane/Norw /O */
case 60 : if (ttfont)
c[0] = ch; /* '>' */
else if (latin1)
c[0] = 0xa1;
else
c[0] = '!';
break; /* inverted ! */
case 62 : if (ttfont)
c[0] = ch; /* '<' */
else if (latin1)
c[0] = 0xbf;
else
c[0] = '?';
break; /* inverted ? */
case 32 : c[0] = ttfont ? ch : '_'; break; /* underlined blank */
case 92 : c[0] = ttfont ? ch : '"'; break; /* \ from `` */
case 123 : if (ttfont)
c[0] = ch; /* { */
else {
c[0] = '-'; c[1] = '-'; /* -- */
}
break;
case 124 : if (ttfont)
c[0] = ch; /* | */
else {
c[0] = '-'; c[1] = '-'; /* --- */
c[2] = '-';
}
break;
case 125 : if (ttfont)
c[0] = ch; /* } */
else
c[0] = '"'; /* double acute from \H */
break;
case 34 : /* " */
case 39 : /* ' */
case 96 : c[0] = ch; break; /* ` */
/* diacritical marks */
case 18 : c[0] = '`' ; break; /* grave from \` */
case 19 : c[0] = latin1 ? 0xb4 : '\''; break;
/* acute from \' */
case 20 : c[0] = '~' ; break; /* caron from \v */
case 21 : c[0] = '~' ; break; /* breve from \u */
case 22 : c[0] = '~' ; break; /* macron from \= */
case 23 : c[0] = latin1 ? 0xb0 : '~'; break;
/* ring above from \r */
case 24 : c[0] = latin1 ? 0xb8 : ','; break;
/* cedilla from \c */
case 94 : c[0] = '^' ; break; /* circumflex from \^ */
case 95 : c[0] = !ttfont ? '.' : ch; break;
/* dot from \. */
case 126 : c[0] = '~' ; break; /* tilde from \~ */
case 127 : c[0] = '"' ; break; /* diaeresis from \" */
default : c[0] = '#';
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
} /* normchar */
/*
* T1CHAR -- Process a character opcode for a T1 encoding font.
*/
void t1char(unsigned char ch)
{
unsigned char c[4] = {}, *cc;
long ucs;
if (allchar) {
outchar(ch);
return;
}
if (!accent) {
switch (ch) {
case 0x00: /* grave from \` */
case 0x01: /* acute from \' */
case 0x02: /* circumflex from \^ */
case 0x03: /* tilde from \~ */
case 0x04: /* diaeresis from \" */
case 0x05: /* double acute from \H */
case 0x06: /* ring above from \r */
case 0x07: /* caron from \v */
case 0x08: /* breve from \u */
case 0x09: /* macron from \= */
case 0x0a: /* dot from \. */
case 0x0b: /* cedilla from \c */
case 0x0c: /* ogonek from \k */
return;
}
}
if (ch==0xdf) {
outchar('S'); outchar('S'); /* SS */
return;
}
ucs = t1_to_ucs[ch];
if (utf8) {
dounichar(ucs);
return;
}
else if ((latin1 && ucs<0x100) || ucs<0x80) {
outchar(ucs);
return;
}
switch (ch) {
case 0x17: return; /* \textcompwordmark */
case 0x0d: /* \quotesinglbase */
case 0x27: /* \textquoteright */
case 0x60: c[0] = '\''; break; /* \textquoteleft */
case 0x10: /* \textquotedblleft */
case 0x11: /* \textquotedblright */
case 0x12: c[0] = '"'; break; /* \quotedblbase */
case 0x0e: c[0] = '<'; break; /* \guilsinglleft */
case 0x0f: c[0] = '>'; break; /* \guilsinglright */
case 0x13: c[0] = '<'; c[1] = '<'; /* \guillemotleft */
break;
case 0x14: c[0] = '>'; c[1] = '>'; /* \guillemotright */
break;
case 0x15: c[0] = '-'; c[1] = '-'; /* \textendash */
break;
case 0x16: c[0] = '-'; c[1] = '-'; /* \textemdash */
c[2] = '-'; break;
case 0x20: c[0] = '_'; break; /* \textvisiblespace */
case 0x7f: c[0] = '-'; break; /* - */
case 0x19: c[0] = 'i'; break; /* dotless i */
case 0x1a: c[0] = 'j'; break; /* dotless j */
case 0x1b: c[0] = 'f'; c[1] = 'f'; /* ligature */
break;
case 0x1c: c[0] = 'f'; c[1] = 'i'; /* ligature */
break;
case 0x1d: c[0] = 'f'; c[1] = 'l'; /* ligature */
break;
case 0x1e: c[0] = 'f'; c[1] = 'f';
c[2] = 'i'; /* ligature */
break;
case 0x1f: c[0] = 'f'; c[1] = 'f';
c[2] = 'l'; /* ligature */
break;
case 0xff: c[0] = 's'; c[1] = 's';
break; /* German double s */
case 0xe6: c[0] = 'a'; c[1] = 'e';
break; /* Dane/Norw ae */
case 0xf7: c[0] = 'o'; c[1] = 'e';
break; /* Dane/Norw oe */
case 0xf8: c[0] = '/'; c[1] = 'o';
break; /* Dane/Norw /o */
case 0xc6: c[0] = 'A'; c[1] = 'E';
break; /* Dane/Norw AE */
case 0xd7: c[0] = 'O'; c[1] = 'E';
break; /* Dane/Norw OE */
case 0xd8: c[0] = '/'; c[1] = 'O';
break; /* Dane/Norw /O */
case 0x9c: c[0] = 'I'; c[1] = 'J';
break; /* IJ */
case 0xbc: c[0] = 'i'; c[1] = 'j';
break; /* ij */
case 0x8d: c[0] = 'N'; c[1] = 'G';
break; /* ENG */
case 0xad: c[0] = 'n'; c[1] = 'g';
break; /* eng */
case 0xde: c[0] = 'T'; c[1] = 'H';
break; /* THORN */
case 0xfe: c[0] = 't'; c[1] = 'h';
break; /* thorn */
case 0x80: c[0] = '~'; c[1] ='A'; break; /* uA */
case 0x81: c[0] = ','; c[1] ='A'; break; /* ,A */
case 0x82: c[0] = '\''; c[1] ='C'; break; /* 'C */
case 0x83: c[0] = '~'; c[1] ='C'; break; /* vC */
case 0x84: c[0] = '~'; c[1] ='D'; break; /* vD */
case 0x85: c[0] = '~'; c[1] ='E'; break; /* vE */
case 0x86: c[0] = ','; c[1] ='E'; break; /* ,E */
case 0x87: c[0] = '~'; c[1] ='G'; break; /* uG */
case 0x88: c[0] = '\''; c[1] ='L'; break; /* 'L */
case 0x89: c[0] = '\''; c[1] ='L'; break; /* 'L */
case 0x8a: c[0] = '-'; c[1] ='L'; break; /* -L */
case 0x8b: c[0] = '\''; c[1] ='N'; break; /* 'N */
case 0x8c: c[0] = '~'; c[1] ='N'; break; /* vN */
case 0x8e: c[0] = '"'; c[1] ='O'; break; /* "O */
case 0x8f: c[0] = '\''; c[1] ='R'; break; /* 'R */
case 0x90: c[0] = '~'; c[1] ='R'; break; /* vR */
case 0x91: c[0] = '\''; c[1] ='S'; break; /* 'S */
case 0x92: c[0] = '~'; c[1] ='S'; break; /* vS */
case 0x93: c[0] = ','; c[1] ='S'; break; /* ,S */
case 0x94: c[0] = '~'; c[1] ='T'; break; /* vT */
case 0x95: c[0] = ','; c[1] ='T'; break; /* ,T */
case 0x96: c[0] = '"'; c[1] ='U'; break; /* "U */
case 0x97: c[0] = '\''; c[1] ='U'; break; /* oU */
case 0x98: c[0] = '"'; c[1] ='Y'; break; /* "Y */
case 0x99: c[0] = '\''; c[1] ='Z'; break; /* 'Z */
case 0x9a: c[0] = '~'; c[1] ='Z'; break; /* vZ */
case 0x9b: c[0] = '\''; c[1] ='Z'; break; /* .Z */
case 0x9d: c[0] = '\''; c[1] ='I'; break; /* .I */
case 0x9e: c[0] = '-'; c[1] ='d'; break; /* -d */
case 0x9f: c[0] = 'S'; break; /* section sign */
case 0xa0: c[0] = '~'; c[1] ='a'; break; /* ua */
case 0xa1: c[0] = ','; c[1] ='a'; break; /* ,a */
case 0xa2: c[0] = '\''; c[1] ='c'; break; /* 'c */
case 0xa3: c[0] = '~'; c[1] ='c'; break; /* vc */
case 0xa4: c[0] = '\''; c[1] ='d'; break; /* 'd */
case 0xa5: c[0] = '~'; c[1] ='e'; break; /* ve */
case 0xa6: c[0] = ','; c[1] ='e'; break; /* ,e */
case 0xa7: c[0] = '~'; c[1] ='g'; break; /* ug */
case 0xa8: c[0] = '\''; c[1] ='l'; break; /* 'l */
case 0xa9: c[0] = '\''; c[1] ='l'; break; /* 'l */
case 0xaa: c[0] = '-'; c[1] ='l'; break; /* -l */
case 0xab: c[0] = '\''; c[1] ='n'; break; /* 'n */
case 0xac: c[0] = '~'; c[1] ='n'; break; /* vn */
case 0xae: c[0] = '"'; c[1] ='o'; break; /* "o */
case 0xaf: c[0] = '\''; c[1] ='r'; break; /* 'r */
case 0xb0: c[0] = '~'; c[1] ='r'; break; /* vr */
case 0xb1: c[0] = '\''; c[1] ='s'; break; /* 's */
case 0xb2: c[0] = '~'; c[1] ='s'; break; /* vs */
case 0xb3: c[0] = ','; c[1] ='s'; break; /* ,s */
case 0xb4: c[0] = '\''; c[1] ='t'; break; /* 't */
case 0xb5: c[0] = ','; c[1] ='t'; break; /* ,t */
case 0xb6: c[0] = '"'; c[1] ='u'; break; /* "u */
case 0xb7: c[0] = '\''; c[1] ='u'; break; /* ou */
case 0xb8: c[0] = '"'; c[1] ='y'; break; /* "y */
case 0xb9: c[0] = '\''; c[1] ='z'; break; /* 'z */
case 0xba: c[0] = '~'; c[1] ='z'; break; /* vz */
case 0xbb: c[0] = '\''; c[1] ='z'; break; /* .z */
case 0xbd: c[0] = '!'; break; /* inversed ! */
case 0xbe: c[0] = '?'; break; /* inversed ? */
case 0xbf: c[0] = 'L'; break; /* pound sign */
case 0xc0: c[0] = '`'; c[1] ='A'; break; /* `A */
case 0xc1: c[0] = '\''; c[1] ='A'; break; /* 'A */
case 0xc2: c[0] = '^'; c[1] ='A'; break; /* ^A */
case 0xc3: c[0] = '~'; c[1] ='A'; break; /* ~A */
case 0xc4: c[0] = '"'; c[1] ='A'; break; /* "A */
case 0xc5: c[0] = 'A'; c[1] ='A'; break; /* oA */
case 0xc7: c[0] = ','; c[1] ='C'; break; /* ,C */
case 0xc8: c[0] = '`'; c[1] ='E'; break; /* `E */
case 0xc9: c[0] = '\''; c[1] ='E'; break; /* 'E */
case 0xca: c[0] = '^'; c[1] ='E'; break; /* ^E */
case 0xcb: c[0] = '^'; c[1] ='E'; break; /* "E */
case 0xcc: c[0] = '`'; c[1] ='I'; break; /* `I */
case 0xcd: c[0] = '\''; c[1] ='I'; break; /* 'I */
case 0xce: c[0] = '^'; c[1] ='I'; break; /* ^I */
case 0xcf: c[0] = '"'; c[1] ='I'; break; /* "I */
case 0xd0: c[0] = '-'; c[1] ='D'; break; /* -D */
case 0xd1: c[0] = '~'; c[1] ='n'; break; /* ~n */
case 0xd2: c[0] = '`'; c[1] ='O'; break; /* `O */
case 0xd3: c[0] = '\''; c[1] ='O'; break; /* 'O */
case 0xd4: c[0] = '^'; c[1] ='O'; break; /* ^O */
case 0xd5: c[0] = '~'; c[1] ='O'; break; /* ~O */
case 0xd6: c[0] = '"'; c[1] ='O'; break; /* "O */
case 0xd9: c[0] = '`'; c[1] ='U'; break; /* `U */
case 0xda: c[0] = '\''; c[1] ='U'; break; /* 'U */
case 0xdb: c[0] = '^'; c[1] ='U'; break; /* ^U */
case 0xdc: c[0] = '"'; c[1] ='U'; break; /* "U */
case 0xdd: c[0] = '\''; c[1] ='Y'; break; /* 'Y */
case 0xe0: c[0] = '`'; c[1] ='a'; break; /* `a */
case 0xe1: c[0] = '\''; c[1] ='a'; break; /* 'a */
case 0xe2: c[0] = '^'; c[1] ='a'; break; /* ^a */
case 0xe3: c[0] = '~'; c[1] ='a'; break; /* ~a */
case 0xe4: c[0] = '"'; c[1] ='a'; break; /* "a */
case 0xe5: c[0] = 'a'; c[1] ='a'; break; /* oa */
case 0xe7: c[0] = ','; c[1] ='c'; break; /* ,c */
case 0xe8: c[0] = '`'; c[1] ='e'; break; /* `e */
case 0xe9: c[0] = '\''; c[1] ='e'; break; /* 'e */
case 0xea: c[0] = '^'; c[1] ='e'; break; /* ^e */
case 0xeb: c[0] = '^'; c[1] ='e'; break; /* "e */
case 0xec: c[0] = '`'; c[1] ='i'; break; /* `i */
case 0xed: c[0] = '\''; c[1] ='i'; break; /* 'i */
case 0xee: c[0] = '^'; c[1] ='i'; break; /* ^i */
case 0xef: c[0] = '"'; c[1] ='i'; break; /* "i */
case 0xf0: c[0] = '-'; c[1] ='d'; break; /* -d */
case 0xf1: c[0] = '~'; c[1] ='n'; break; /* ~n */
case 0xf2: c[0] = '`'; c[1] ='o'; break; /* `o */
case 0xf3: c[0] = '\''; c[1] ='o'; break; /* 'o */
case 0xf4: c[0] = '^'; c[1] ='o'; break; /* ^o */
case 0xf5: c[0] = '~'; c[1] ='o'; break; /* ~o */
case 0xf6: c[0] = '"'; c[1] ='o'; break; /* "o */
case 0xf9: c[0] = '`'; c[1] ='u'; break; /* `u */
case 0xfa: c[0] = '\''; c[1] ='u'; break; /* 'u */
case 0xfb: c[0] = '^'; c[1] ='u'; break; /* ^u */
case 0xfc: c[0] = '"'; c[1] ='u'; break; /* "u */
case 0xfd: c[0] = '\''; c[1] ='y'; break; /* 'y */
/* diacritical marks */
case 0x00: c[0] = '`' ; break; /* grave from \` */
case 0x01: c[0] = latin1 ? 0xb4 : '\''; break;
/* acute from \' */
case 0x02: c[0] = '^' ; break; /* circumflex from \^ */
case 0x03: c[0] = '~' ; break; /* tilde from \~ */
case 0x04: c[0] = '"' ; break; /* diaeresis from \" */
case 0x05: c[0] = '"' ; break; /* double acute from \H */
case 0x06: c[0] = latin1 ? 0xb0 : '~'; break;
/* ring above from \r */
case 0x07: c[0] = '~' ; break; /* caron from \v */
case 0x08: c[0] = '~' ; break; /* breve from \u */
case 0x09: c[0] = '~' ; break; /* macron from \= */
case 0x0a: c[0] = '.' ; break; /* dot from \. */
case 0x0b: c[0] = latin1 ? 0xb8 : ','; break;
/* cedilla from \c */
case 0x0c: c[0] = ',' ; break; /* ogonek from \k */
default : c[0] = '#';
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
} /* t1char */
/*
* TS1CHAR -- Process a character opcode for a TS1 encoding font.
*/
void ts1char(unsigned char ch)
{
unsigned char c[4] = {}, *cc;
long ucs;
if (allchar) {
outchar(ch);
return;
}
ucs = ts1_to_ucs[ch];
if (utf8) {
dounichar(ucs);
return;
}
else if ((latin1 && ucs<0x100) || ucs<0x80) {
outchar(ucs);
return;
}
switch (ch) {
case 0x17: /* \capitalcompwordmark */
case 0x1F: return; /* \textascendercompwordmark */
case 0x0D: /* \textquotestraightbase */
case 0x27: c[0] = '\''; break; /* \textquotesingle */
case 0x12: c[0] = '"'; break; /* \textquotestraghtdblbase */
case 0x15: c[0] = '-'; break; /* \texttwelveudash */
case 0x16: c[0] = '-'; c[1] = '-'; /* \textthreequartersemdash */
break;
case 0x18: c[0] = '<'; c[1] = '-'; /* \textleftarrow */
break;
case 0x19: c[0] = '-'; c[1] = '>'; /* \textrightarrow */
break;
case 0x2A: c[0] = '*'; break; /* \textasteriskcentered */
case 0x2D: c[0] = '='; break; /* \textdblhyphen */
case 0x2F: c[0] = '/'; break; /* \textfractionsolidus */
case 0x3C: c[0] = '<'; break; /* \textlangle */
case 0x3D: c[0] = '-'; break; /* \textminus */
case 0x3E: c[0] = '>'; break; /* \textrangle */
case 0x5B: c[0] = '['; break; /* \textlbrackdbl */
case 0x5D: c[0] = ']'; break; /* \textrbrackdbl */
case 0x5E: c[0] = '^'; break; /* \textuparrow */
case 0x5F: c[0] = 'v'; break; /* \textdownarrow */
case 0x7E: c[0] = '~'; break; /* \texttildelow */
case 0x7F: c[0] = '='; break; /* \textdblhyphenchar */
case 0x84: c[0] = '*'; break; /* \textdagger */
case 0x85: c[0] = '*'; c[1] = '*'; /* \textdaggerdbl */
break;
case 0x86: c[0] = '|'; c[1] = '|'; /* \textbardbl */
break;
case 0x89: if (latin1) {
c[0] = 0xb0; c[1] = 'C';
}
else
c[0] = 'C';
break; /* \textcelsius */
case 0x8B: c[0] = 'c'; break; /* \textcent */
case 0x8C: c[0] = 'f'; break; /* \textflorin */
case 0x8D: c[0] = 'C'; break; /* \textcentoldstyle */
case 0x8E: c[0] = 'W'; break; /* \textwon */
case 0x8F: c[0] = 'N'; break; /* \textnaira */
case 0x90: c[0] = 'G'; break; /* \textguarani */
case 0x91: c[0] = 'P'; break; /* \textpeso */
case 0x92: c[0] = 'L'; break; /* \textlira */
case 0x93: c[0] = 'R'; break; /* \textrecipe */
case 0x94: /* \textinterrobang */
case 0x95: c[0] = '!'; c[1] = '?'; /* \textinterrobangdown */
break;
case 0x97: c[0] = 'T'; c[1] = 'M'; /* \texttrademark */
break;
case 0x99: c[0] = 'P'; break; /* \textpilcrow */
case 0x9B: c[0] = 'N'; c[1] = 'o'; /* \textnumero */
break;
case 0x9F: c[0] = 'S'; c[1] = 'M'; /* \textservicemark */
break;
case 0xA0: c[0] = '{'; break; /* \textlquill */
case 0xA1: c[0] = '}'; break; /* \textrquill */
case 0xA2: c[0] = 'c'; break; /* \textcent */
case 0xA3: c[0] = 'L'; break; /* \textsterling */
case 0xA5: c[0] = 'Y'; break; /* \textyen */
case 0xA6: c[0] = '|'; break; /* \textbrokenbar */
case 0xA7: c[0] = 'S'; break; /* \textsection */
case 0xA9: c[0] = 'C'; break; /* \textcopyright */
case 0xAD: c[0] = 'P'; break; /* \textcircledP */
case 0xAE: c[0] = 'R'; break; /* \textregistered */
case 0xB6: c[0] = 'P'; break; /* \textparagraph */
case 0xB1: c[0] = '+'; c[1] = '-'; /* \textpm */
break;
case 0xBC: c[0] = '1'; c[1] = '/'; /* \textonequarter */
c[2] = '4'; break;
case 0xBD: c[0] = '1'; c[1] = '/'; /* \textonehalf */
c[2] = '2'; break;
case 0xBE: c[0] = '3'; c[1] = '/'; /* \textthreequarters */
c[2] = '4'; break;
case 0xBF: c[0] = 'E'; break; /* \texteuro */
case 0xD6: c[0] = 'x'; break; /* \texttimes */
case 0xF6: c[0] = '/'; break; /* \textdiv */
case 0x30: case 0x31: case 0x32: case 0x33:
case 0x34: case 0x35: case 0x36: case 0x37:
case 0x38: case 0x39: case 0x3A: case 0x3B:
c[0] = ch; break;
/* diacritical marks */
case 0x00: c[0] = '`' ; break; /* grave from \` */
case 0x01: c[0] = latin1 ? 0xb4 : '\''; break;
/* acute from \' */
case 0x02: c[0] = '^' ; break; /* circumflex from \^ */
case 0x03: c[0] = '~' ; break; /* tilde from \~ */
case 0x04: c[0] = '"' ; break; /* diaeresis from \" */
case 0x05: c[0] = '"' ; break; /* double acute from \H */
case 0x06: c[0] = latin1 ? 0xb0 : '~'; break;
/* ring above from \r */
case 0x07: c[0] = '~' ; break; /* caron from \v */
case 0x08: c[0] = '~' ; break; /* breve from \u */
case 0x09: c[0] = '~' ; break; /* macron from \= */
case 0x0a: c[0] = '.' ; break; /* dot from \. */
case 0x0b: c[0] = latin1 ? 0xb8 : ','; break;
/* cedilla from \c */
case 0x0c: c[0] = ',' ; break; /* ogonek from \k */
default : c[0] = '#';
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
} /* ts1char */
/*
* T2CHAR -- Process a character opcode for a T2A/T2B/T2C/X2 encoding font.
*/
void t2char(char flag, unsigned char ch)
{
unsigned char c[4] = {}, *cc;
const unsigned short *tex_to_ucs;
long ucs;
if (allchar) {
outchar(ch);
return;
}
if (!accent) {
switch (ch) {
case 0x00: /* grave from \` */
case 0x01: /* acute from \' */
case 0x02: /* circumflex from \^ */
case 0x03: /* tilde from \~ */
case 0x04: /* diaeresis from \" */
case 0x05: /* double acute from \H */
case 0x06: /* ring above from \r */
case 0x07: /* caron from \v */
case 0x08: /* breve from \u */
case 0x09: /* macron from \= */
case 0x0a: /* dot from \. */
case 0x0b: /* cedilla from \c */
case 0x0c: /* ogonek from \k */
case 0x12: /* from \f */
case 0x13: /* from \C */
case 0x14: /* breve from \U */
return;
}
}
switch (flag) {
case T2AFONT: tex_to_ucs=t2a_to_ucs; break;
case T2BFONT: tex_to_ucs=t2b_to_ucs; break;
case T2CFONT: tex_to_ucs=t2c_to_ucs; break;
case X2FONT : tex_to_ucs=x2_to_ucs; break;
default : exit; /* not supported */
}
ucs = tex_to_ucs[ch];
if (utf8) {
dounichar(ucs);
return;
}
else if ((latin1 && ucs<0x100) || ucs<0x80) {
outchar(ucs);
return;
}
switch (ch) {
case 0x49: c[0] = 'I'; break; /* \CYRII */
case 0x69: c[0] = 'i'; break; /* \cyrii */
case 0x4A: c[0] = 'J'; break; /* \CYRJE */
case 0x6A: c[0] = 'j'; break; /* \cyrje */
case 0x51: c[0] = 'Q'; break; /* \CYRQ */
case 0x53: c[0] = 'S'; break; /* \CYRDZE */
case 0x57: c[0] = 'W'; break; /* \CYRW */
case 0x71: c[0] = 'q'; break; /* \cyrq */
case 0x73: c[0] = 's'; break; /* \cyrdze */
case 0x77: c[0] = 'w'; break; /* \cyrw */
case 0x0E: c[0] = '<'; break; /* \cyrlangle */
case 0x0F: c[0] = '>'; break; /* \cyrrangle */
case 0x15: c[0] = '-'; c[1] = '-'; /* \textendash */
break;
case 0x16: c[0] = '-'; c[1] = '-'; /* \textemdash */
c[2] = '-'; break;
case 0x27: /* \textquoteright */
case 0x60: c[0] = '\''; break; /* \textquoteleft */
case 0x10: /* \textquotedblleft */
case 0x11: /* \textquotedblright */
case 0xBD: c[0] = '"'; break; /* \quotedblbase */
case 0x20: c[0] = '_'; break; /* \textvisiblespace */
case 0x17: return; /* \textcompwordmark */
case 0x7E: c[0] = '~'; break; /* \textasciitilde */
case 0x9D: c[0] = 'N'; c[1] = 'o'; /* \textnumero */
break;
case 0x9F: c[0] = 'S'; break; /* \textsection */
case 0xBE: c[0] = '<'; c[1] = '<'; /* \guillemotleft */
break;
case 0xBF: c[0] = '>'; c[1] = '>'; /* \guillemotright */
break;
/* diacritical marks */
case 0x00: c[0] = '`' ; break; /* grave from \` */
case 0x01: c[0] = latin1 ? 0xb4 : '\''; break;
/* acute from \' */
case 0x02: c[0] = '^' ; break; /* circumflex from \^ */
case 0x03: c[0] = '~' ; break; /* tilde from \~ */
case 0x04: c[0] = '"' ; break; /* diaeresis from \" */
case 0x05: c[0] = '"' ; break; /* double acute from \H */
case 0x06: c[0] = latin1 ? 0xb0 : '~'; break;
/* ring above from \r */
case 0x07: c[0] = '~' ; break; /* caron from \v */
case 0x08: c[0] = '~' ; break; /* breve from \u */
case 0x09: c[0] = '~' ; break; /* macron from \= */
case 0x0a: c[0] = '.' ; break; /* dot from \. */
case 0x0b: c[0] = latin1 ? 0xb8 : ','; break;
/* cedilla from \c */
case 0x0c: c[0] = ',' ; break; /* ogonek from \k */
case 0x14: c[0] = '~' ; break; /* breve from \U */
default : c[0] = '#';
}
if (flag != X2FONT) {
switch (ch) {
case 0x19: c[0] = 'i'; break; /* dotless i */
case 0x1A: c[0] = 'j'; break; /* dotless j */
case 0x1B: c[0] = 'f'; c[1] = 'f'; /* ligature */
break;
case 0x1C: c[0] = 'f'; c[1] = 'i'; /* ligature */
break;
case 0x1D: c[0] = 'f'; c[1] = 'l'; /* ligature */
break;
case 0x1E: c[0] = 'f'; c[1] = 'f'; /* ligature */
c[2] = 'i'; break;
case 0x1F: c[0] = 'f'; c[1] = 'f'; /* ligature */
c[2] = 'l'; break;
}
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
} /* t2char */
/*
* OT2CHAR -- Process a character opcode for a OT2 encoding font.
*/
void ot2char(unsigned char ch)
{
unsigned char c[4] = {}, *cc;
long ucs;
if (allchar) {
outchar(ch);
return;
}
if (!accent) {
switch (ch) {
case 0x20: /* diaeresis from \" */
case 0x24: /* breve from \U */
case 0x26: /* acute from \' */
case 0x40: /* breve from \u */
return;
}
}
ucs = ot2_to_ucs[ch];
if (utf8) {
dounichar(ucs);
return;
}
else if ((latin1 && ucs<0x100) || ucs<0x80) {
outchar(ucs);
return;
}
switch (ch) {
case 0x04: c[0] = 'I'; break; /* \CYRII */
case 0x0C: c[0] = 'i'; break; /* \cyrii */
case 0x4A: c[0] = 'J'; break; /* \CYRJE */
case 0x6A: c[0] = 'j'; break; /* \cyrje */
case 0x16: c[0] = 'S'; break; /* \CYRDZE */
case 0x1E: c[0] = 's'; break; /* \cyrdze */
case 0x7B: c[0] = '-'; c[1] = '-'; /* \textendash */
break;
case 0x7C: c[0] = '-'; c[1] = '-'; /* \textemdash */
c[2] = '-'; break;
case 0x7D: c[0] = 'N'; c[1] = 'o'; /* \textnumero */
break;
case 0x3C: c[0] = '<'; c[1] = '<'; /* \guillemotleft */
break;
case 0x3D: c[0] = 'i'; break; /* dotless i */
case 0x3E: c[0] = '>'; c[1] = '>'; /* \guillemotright */
break;
case 0x27: /* \textquoteright */
case 0x60: c[0] = '\''; break; /* \textquoteleft */
case 0x22: /* \textquotedblright */
case 0x5C: c[0] = '"'; break; /* \textquotedblleft */
/* diacritical marks */
case 0x20: c[0] = '"' ; break; /* diaeresis from \" */
case 0x24: c[0] = '~' ; break; /* breve from \u */
case 0x26: c[0] = latin1 ? 0xb4 : '\''; break;
/* acute from \' */
case 0x40: c[0] = '~' ; break; /* breve from \U */
default : c[0] = '#';
}
cc=c;
while (*cc) { outchar(*cc); cc++; }
return;
} /* ot2char */
/*
* OUTCHAR -- Here we put the character into the current page.
*
* This function includes some code to handle Latin1/Scandinavian
* characters. I think that code doesn't belong here. IT
* SHOULD BE MOVED OUT.
*/
void outchar(long ch)
{
register int i, j;
register long dia;
/* fprintf(stderr, "hor: %ld, ver: %ld\n", h, v); */
#if defined(MSDOS) || defined(THINK_C)
if (labs(v - currentline->vv) > lineheight / 2L)
#else
if (abs(v - currentline->vv) > lineheight / 2L)
#endif
currentline = findline();
#if 0
j = (int) (((double) h / (double) maxpagewidth) * (ttywidth-1)) + 1;
#else
j = (int) (h / charwidth);
#endif
if (j > rightmargin) /* leftmargin <= j <= rightmargin */
j = rightmargin;
else if (j < leftmargin)
j = leftmargin;
foo = leftmargin - 1;
/*
* This code does not really belong here ...
*
* The following is very specialized code, it handles national *
* Swe/Fin characters. They are respectively: a and o with two *
* dots ("a & "o) and a with a circle (Oa). In Swe/Fin "ASCII" *
* these characters replace {}|[] and \. TeX outputs these by *
* first issuing the dots or circle and then backspace and set *
* the a or o. When dvi2tty finds an a or o it searches in the *
* near vicinity for the character codes that represent circle *
* or dots and if one is found the corresponding national char *
* replaces the special character codes. *
*/
if (!allchar && compose && scascii) {
if (strchr("aAoO", ch) != NULL) {
for (i = IMAX(leftmargin, j-2);
i <= IMIN(rightmargin, j+2);
i++)
if ((currentline->text[i - leftmargin] == 127) || /* DEL */
(currentline->text[i - leftmargin] == 34) || /* " */
(currentline->text[i - leftmargin] == 23))
foo = i;
if (foo >= leftmargin) {
j = (int) foo;
switch (currentline->text[j - leftmargin]) {
case 127 :
case 34 : /* DEL or " */
if (ch == 'a')
ch = '{'; /* } vi */
else if (ch == 'A') /* dots ... */
ch = '[';
else if (ch == 'o')
ch = '|';
else if (ch == 'O')
ch = '\\';
break;
case 23 : if (ch == 'a')
ch = '}'; /* { vi */
else if (ch == 'A') /* circle */
ch = ']';
break;
}
}
}
}
if (!allchar && compose && (latin1 || utf8)) {
if (strchr("aAeEiIoOuUnCcNYy", ch) != NULL || (ch & MAX_UNICODE) == 0x131) {
for (i = IMAX(leftmargin, j-2);
i <= IMIN(rightmargin, j+2);
i++) {
dia = currentline->text[i - leftmargin] & MAX_UNICODE;
if ((dia == 0x60) || /* grave */
(dia == 0xB0) || /* ring above */
(dia == 0x2DA) || /* ring above */
(dia == 0xB4) || /* acute */
(dia == 0x5E) || /* circumflex */
(dia == 0xA8) || /* diaeresis */
(dia == 0xB8) || /* cedilla */
(dia == 0x7E) || /* tilde */
(dia == 0x2DC)) /* tilde */
foo = i;
}
if (foo >= leftmargin) {
j = (int) foo;
dia = currentline->text[j - leftmargin] & MAX_UNICODE;
switch (dia) {
case 0x60: /* grave */
if (ch == 'a') ch = 0xe0;
else if (ch == 'A') ch = 0xc0;
else if (ch == 'e') ch = 0xe8;
else if (ch == 'E') ch = 0xc8;
else if (ch == 'i') ch = 0xec;
else if ((ch & MAX_UNICODE) == 0x131) ch = 0xec;
else if (ch == 'I') ch = 0xcc;
else if (ch == 'o') ch = 0xf2;
else if (ch == 'O') ch = 0xd2;
else if (ch == 'u') ch = 0xf9;
else if (ch == 'U') ch = 0xd9;
break;
case 0xB0: /* ring above */
case 0x2DA:
if (ch == 'a') ch = 0xe5;
else if (ch == 'A') ch = 0xc5;
break;
case 0xB4: /* acute */
if (ch == 'a') ch = 0xe1;
else if (ch == 'A') ch = 0xc1;
else if (ch == 'e') ch = 0xe9;
else if (ch == 'E') ch = 0xc9;
else if (ch == 'i') ch = 0xed;
else if ((ch & MAX_UNICODE) == 0x131) ch = 0xed;
else if (ch == 'I') ch = 0xcd;
else if (ch == 'o') ch = 0xf3;
else if (ch == 'O') ch = 0xd3;
else if (ch == 'u') ch = 0xfa;
else if (ch == 'U') ch = 0xda;
else if (ch == 'y') ch = 0xfd;
else if (ch == 'Y') ch = 0xdd;
break;
case 0x5E: /* circumflex */
if (ch == 'a') ch = 0xe2;
else if (ch == 'A') ch = 0xc2;
else if (ch == 'e') ch = 0xea;
else if (ch == 'E') ch = 0xca;
else if (ch == 'i') ch = 0xee;
else if ((ch & MAX_UNICODE) == 0x131) ch = 0xee;
else if (ch == 'I') ch = 0xce;
else if (ch == 'o') ch = 0xf4;
else if (ch == 'O') ch = 0xd4;
else if (ch == 'u') ch = 0xfb;
else if (ch == 'U') ch = 0xdb;
break;
case 0xA8: /* diaeresis */
if (ch == 'a') ch = 0xe4;
else if (ch == 'A') ch = 0xc4;
else if (ch == 'e') ch = 0xeb;
else if (ch == 'E') ch = 0xcb;
else if (ch == 'i') ch = 0xef;
else if ((ch & MAX_UNICODE) == 0x131) ch = 0xef;
else if (ch == 'I') ch = 0xcf;
else if (ch == 'o') ch = 0xf6;
else if (ch == 'O') ch = 0xd6;
else if (ch == 'u') ch = 0xfc;
else if (ch == 'U') ch = 0xdc;
else if (ch == 'y') ch = 0xff;
else if (ch == 'Y' && utf8) ch = 0x178;
break;
case 0xB8: /* cedilla */
if (ch == 'c') ch = 0xe7;
else if (ch == 'C') ch = 0xc7; /* It does not seem to work */
break;
case 0x7E: /* tilde */
case 0x2DC:
if (ch == 'a') ch = 0xe3;
else if (ch == 'A') ch = 0xc3;
else if (ch == 'o') ch = 0xf5;
else if (ch == 'O') ch = 0xd5;
else if (ch == 'n') ch = 0xf1;
else if (ch == 'N') ch = 0xd1;
break;
}
if (utf8 && ch>0x7f) ch |= IS_UNICODE;
}
}
}
/*----------------- end of 'latin1 / Scandinavian code' ----------------*/
if (foo == leftmargin-1) {
if (japan) {
while (((currentline->text[j - leftmargin] != SPACE) ||
(kanji1 && (currentline->text[j+kanji1 - leftmargin] != SPACE)))
&& (j < rightmargin)) {
j++;
h += charwidth;
}
} else {
while (j < rightmargin &&
(currentline->text[j - leftmargin] != SPACE)) {
j++;
h += charwidth;
}
}
}
if ( allchar || ((ch >= SPACE) && (ch != DEL)) ||
((latin1 || scascii) && (ch == 23)) ) {
/* ((latin1 || scascii) && (ch == DEL)) ) if VMS ??? */
if (j < rightmargin)
currentline->text[j - leftmargin] = ch;
else
currentline->text[rightmargin - leftmargin] = '@';
if (j > currentline->charactercount)
currentline->charactercount = j;
if (j < firstcolumn)
firstcolumn = j;
}
h += charwidth;
return;
} /* outchar */
/*
* PUTCHARACTER -- Output character, don't change h
*/
void putcharacter(long charnr)
{
register long saveh;
saveh = h;
if (nttj || is8bit)
dochar((unsigned char) charnr);
else if (allchar || ((charnr >= 0) && (charnr <= LASTCHAR)))
outchar((unsigned char) charnr);
else
setchar(charnr);
h = saveh;
return;
} /* putcharacter */
/*
* SETCHAR -- Should print characters with character code>127 from
* current font. Note that the parameter is a dummy, since
* ascii-chars are<=127.
*/
void setchar(long charnr)
{
if (is8bit)
dochar((unsigned char) charnr);
else
outchar((unsigned char)(allchar ? charnr : '#'));
return;
} /* setchar */
static const char *ptex_fontchk[] = {
"min", "goth", "jis",
"hmin", "hgoth", "hmgoth", /* japanese-otf package */
"nmlmin", "nmlgoth", "nmlmgoth",
"hiramin", "hirakaku", "hiramaru",
NULL /* end */
};
static const char *uptex_fontchk[] = {
"umin", "ugoth", "ujis",
"upjis", "upjpn", "upsch", "uptch", "upkor",
"uphmin", "uphgoth", "uphmgoth", /* japanese-otf package */
"upnmlmin", "upnmlgoth", "upnmlmgoth",
"uphiramin", "uphirakaku", "uphiramaru",
NULL /* end */
};
static const char *jtex_fontchk[] = {
"dmj", "dgj",
NULL /* end */
};
static int checkjfont(const char **jfontlist, const char *name)
{
int i, len;
const char *tmpfont;
i=0;
while ( (tmpfont=jfontlist[i]) != NULL ) {
len=strlen(tmpfont);
if ( !strncmp(tmpfont, name, len) ) return 1;
i++;
}
return 0;
} /* checkjfont */
/*
* FONTDEF -- Process a font definition.
*/
void fontdef(int x)
{
register int i;
char * name;
font * fnt;
int namelen;
long fntnum;
int new = 0;
fntnum = num(x);
(void) get4(); /* checksum */
(void) get4(); /* scale */
(void) get4(); /* design */
namelen = (int) get1() + (int) get1();
fnt = fonts;
while (fnt != NULL && fnt->num != fntnum) /* does fontnum exist */
fnt = fnt->next;
if (fnt == NULL) {
if ((fnt = (font *) malloc(sizeof(font))) == NULL) {
perror("fontdef");
exit(1);
}
fnt->num = fntnum;
new = 1;
}
else
free(fnt->name); /* free old name */
if ((name = (char *) malloc((namelen+1) * sizeof(char))) == NULL) {
perror("fontdef");
exit(1);
}
for (i = 0; i < namelen; i++)
name[i] = get1();
name[i] = '\0'; /* properly end string */
fnt->name = name;
if (new) {
fnt->next = fonts;
fonts = fnt;
}
/*
* some magic to learn about font types...
*/
fonts->flags = 0;
fonts->is8bit = FALSE;
if ((asciip == FALSE && nttj == FALSE && uptex == FALSE)
&& (!jdetect) && jautodetect) {
if ( checkjfont(ptex_fontchk, name) ) {
/* Detect as ASCII TeX */
asciip = TRUE;
nttj = uptex = FALSE;
japan = jdetect = TRUE;
fonts->flags |= MIFONT;
set_enc_string (NULL, PTEX_INTERNAL_ENC);
} else if ( checkjfont(uptex_fontchk, name) ) {
/* Detect as upTeX */
uptex = TRUE;
nttj = asciip = FALSE;
japan = jdetect = TRUE;
fonts->flags |= MIFONT;
enable_UPTEX(true);
set_enc_string (NULL, UPTEX_INTERNAL_ENC);
} else if ( checkjfont(jtex_fontchk, name) ) {
/* Detect as NTT JTeX */
nttj = TRUE;
asciip = uptex = FALSE;
japan = jdetect = TRUE;
fonts->flags |= JPFONT;
set_enc_string (NULL, JTEX_INTERNAL_ENC);
}
}
if (nttj)
fonts->fontnum = getjsubfont(name);
else
fonts->fontnum = 0;
if ((strncmp(name, "ec", 2)) == 0) {
fonts->flags = T1FONT;
fonts->is8bit = TRUE;
return;
}
else if ((strncmp(name, "tc", 2)) == 0 ||
(strncmp(name, "ts1", 3)) == 0) {
fonts->flags = TS1FONT;
fonts->is8bit = TRUE;
return;
}
else if ((strncmp(name, "wn", 2)) == 0) {
fonts->flags = OT2FONT;
return;
}
else if ((strncmp(name, "la", 2)) == 0) {
fonts->flags = T2AFONT;
fonts->is8bit = TRUE;
return;
}
else if ((strncmp(name, "lb", 2)) == 0) {
fonts->flags = T2BFONT;
fonts->is8bit = TRUE;
return;
}
else if ((strncmp(name, "lc", 2)) == 0) {
fonts->flags = T2CFONT;
fonts->is8bit = TRUE;
return;
}
else if ((strncmp(name, "rx", 2)) == 0) {
fonts->flags = X2FONT;
fonts->is8bit = TRUE;
return;
}
if ((strstr(name, "sy")) != NULL)
fonts->flags = SYMFONT;
if ((strstr(name, "tt")) != NULL)
fonts->flags = TTFONT;
if ((strstr(name, "mi")) != NULL)
fonts->flags = MIFONT;
return;
} /* fontdef */
#define NJSUBS 33
const char *jsf_names[]={
"sy", "roma", "hira", "kata", "greek", "russian", "keisen",
"ka", "kb", "kc", "kd", "ke", "kf", "kg", "kh", "ki", "kj",
"kk", "kl", "km", "kn", "ko", "kp", "kq", "kr", "ks", "kt",
"ku", "kv", "kw", "kx", "ky", "kz"
};
int getjsubfont(char *s)
{
int jf;
if (strlen(s) > 3 && s[0] == 'd' && (s[1] == 'm' || s[1] == 'g') && s[2] == 'j') {
for (jf = 0; jf < NJSUBS; jf++) {
if (strncmp(&s[3], jsf_names[jf], strlen(jsf_names[jf])) == 0)
return jf+1;
}
}
return 0;
} /* getjsubfont */
/*
* SETFONT -- Switch to specific font. Try to find out if it is a symbol
* font.
* Option -c allchar does not pertain to this portion, so symbols
* are still translated.
*/
void setfont(long fntnum)
{
char * s;
const char * d;
symbolfont = FALSE;
ttfont = FALSE;
mifont = FALSE;
fnt = fonts;
while (fnt != NULL && fnt->num != fntnum)
fnt = fnt->next;
if (fnt == NULL) {
/* error : font not found */
return;
}
if (fnt->fontnum == 0) {
symbolfont = fnt->flags == SYMFONT;
ttfont = fnt->flags == TTFONT;
mifont = fnt->flags == MIFONT;
is8bit = fnt->is8bit;
}
s = fnt->name;
if (printfont) {
d = delim; /* print delim and font name if -b was chosen */
while (*d) {putcharacter(*d); d++;}
while (*s) {putcharacter(*s); s++;}
while (d-- > delim) {putcharacter(*d);}
}
return;
} /* setfont */
void jischar(unsigned long ch)
{
unsigned int Ku, Ten;
compute_jis(fnt->fontnum, (unsigned int) ch, &Ku, &Ten);
kanji1 = 1;
outchar((unsigned char)(Ku+128));
kanji1 = 0;
outchar((unsigned char)(Ten+128));
return;
} /* jischar */
#define kushift(c) c+0x20
#define tenshift(c) c+0x20
void compute_jis(int f, unsigned int c, unsigned int *ku, unsigned int *ten)
{
int n;
if (f <= 7) {
if (f == 1) {
if (c >= 100) {
*ku = kushift(2);
*ten = tenshift(c-100);
}
else {
*ku = kushift(1);
*ten = tenshift(c);
}
}
else if (f == 2) {
*ku = kushift(3);
*ten = tenshift(c-32);
}
else {
*ku = kushift(f+1);
*ten = tenshift(c);
}
}
else if (f <= 19) { /* Daiichi Suijun */
n = (f-8)*256+c;
*ku = kushift((n/94)+16);
*ten = tenshift((n%94)+1);
}
else { /* Daini Suijun */
n = (f-20)*256+c;
*ku = kushift((n/94)+48);
*ten = tenshift((n%94)+1);
}
return;
} /* compute_jis */
/*
* VMS CODE
*/
#if defined(VMS)
long vmsseek(fp,n,dir)
FILE *fp;
long n;
long dir;
{
long k,m,pos,val,oldpos;
struct stat buffer;
for (;;) { /* loops only once or twice */
switch (dir) {
case 0: /* from BOF */
oldpos = vms_ftell(fp);
k = n & 511;
m = n >> 9;
if (((*fp)->_cnt) && ((oldpos >> 9) == m)) {
val = 0; /* still in */
(*fp)->_ptr = ((*fp)->_base) + k;
(*fp)->_cnt = 512 - k;
}
else {
val = fseek(fp, m << 9, 0);
if (val == 0) {
(*fp)->_cnt = 0;
(void) fgetc(fp);
(*fp)->_ptr = ((*fp)->_base) + k;
(*fp)->_cnt = 512 - k;
}
}
return(val);
case 1: pos = vms_ftell(fp);
if (pos == EOF)
return (EOF);
n += pos;
dir = 0;
break;
case 2: val = fstat(fileno(fp), &buffer);
if (val == EOF)
return (EOF);
n += buffer.st_size - 1;
dir = 0;
break;
default : return (EOF);
}
}
/* NOTREACHED */
} /* vmsseek */
long vms_ftell(fp)
FILE *fp;
{
char c;
long pos;
long val;
if ((*fp)->_cnt == 0) {
c = fgetc(fp);
val = vms_ungetc(c, fp);
if (val != c)
return (EOF);
}
pos = ftell(fp);
if (pos >= 0)
pos += ((*fp)->_ptr) - ((*fp)->_base);
return (pos);
} /* vms_ftell */
long vms_ungetc(c,fp)
char c;
FILE *fp;
{
if ((c == EOF) && feof(fp))
return (EOF);
if ((*fp)->_cnt >= 512)
return (EOF);
(*fp)->_cnt++;
(*fp)->_ptr--;
*((*fp)->_ptr) = c;
return (c);
} /*vms_ungetc */
#endif