okular/dvi_draw.c

1242 lines
28 KiB
C
Raw Normal View History

/*
* Copyright (c) 1994 Paul Vojta. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*
* NOTE:
* xdvi is based on prior work as noted in the modification history, below.
*/
/*
* DVI previewer for X.
*
* Eric Cooper, CMU, September 1985.
*
* Code derived from dvi-imagen.c.
*
* Modification history:
* 1/1986 Modified for X.10 --Bob Scheifler, MIT LCS.
* 7/1988 Modified for X.11 --Mark Eichin, MIT
* 12/1988 Added 'R' option, toolkit, magnifying glass
* --Paul Vojta, UC Berkeley.
* 2/1989 Added tpic support --Jeffrey Lee, U of Toronto
* 4/1989 Modified for System V --Donald Richardson, Clarkson Univ.
* 3/1990 Added VMS support --Scott Allendorf, U of Iowa
* 7/1990 Added reflection mode --Michael Pak, Hebrew U of Jerusalem
* 1/1992 Added greyscale code --Till Brychcy, Techn. Univ. Muenchen
* and Lee Hetherington, MIT
* 4/1994 Added DPS support, bounding box
* --Ricardo Telichevesky
* and Luis Miguel Silveira, MIT RLE.
*/
#include "oconfig.h"
#include <kpathsea/c-ctype.h>
#include <kpathsea/c-fopen.h>
#include <kpathsea/c-vararg.h>
#include "dvi.h"
static struct frame frame0; /* dummy head of list */
#ifdef TEXXET
static struct frame *scan_frame; /* head frame for scanning */
#endif
#ifndef DVI_BUFFER_LEN
#define DVI_BUFFER_LEN 512
#endif
static ubyte dvi_buffer[DVI_BUFFER_LEN];
static struct frame *current_frame;
#ifndef TEXXET
#define DIR 1
#else
#define DIR currinf.dir
#endif
/*
* Explanation of the following constant:
* offset_[xy] << 16: margin (defaults to one inch)
* shrink_factor << 16: one pixel page border
* shrink_factor << 15: rounding for pixel_conv
*/
#define OFFSET_X (offset_x << 16) + (shrink_factor * 3 << 15)
#define OFFSET_Y (offset_y << 16) + (shrink_factor * 3 << 15)
#ifndef BMLONG
#ifndef BMSHORT
BMUNIT bit_masks[9] = {
0x0, 0x1, 0x3, 0x7,
0xf, 0x1f, 0x3f, 0x7f,
0xff
};
#else /* BMSHORT */
BMUNIT bit_masks[17] = {
0x0, 0x1, 0x3, 0x7,
0xf, 0x1f, 0x3f, 0x7f,
0xff, 0x1ff, 0x3ff, 0x7ff,
0xfff, 0x1fff, 0x3fff, 0x7fff,
0xffff
};
#endif /* BMSHORT */
#else /* BMLONG */
BMUNIT bit_masks[33] = {
0x0, 0x1, 0x3, 0x7,
0xf, 0x1f, 0x3f, 0x7f,
0xff, 0x1ff, 0x3ff, 0x7ff,
0xfff, 0x1fff, 0x3fff, 0x7fff,
0xffff, 0x1ffff, 0x3ffff, 0x7ffff,
0xfffff, 0x1fffff, 0x3fffff, 0x7fffff,
0xffffff, 0x1ffffff, 0x3ffffff, 0x7ffffff,
0xfffffff, 0x1fffffff, 0x3fffffff, 0x7fffffff,
0xffffffff
};
#endif /* BMLONG */
#ifdef VMS
#define off_t int
#endif
extern off_t lseek();
#ifndef SEEK_SET /* if <unistd.h> is not provided (or for <X11R5) */
#define SEEK_SET 0
#define SEEK_CUR 1
#define SEEK_END 2
#endif
static void draw_part();
#ifdef KDVI
void qtPutRule(int x, int y, int w, int h);
void qtPutBorder(int x, int y, int w, int h);
void qtPutBitmap( int x, int y, int w, int h, unsigned char *bits );
void qt_processEvents();
#endif
/*
* X routines.
*/
/*
* Draw the border of a rectangle on the screen.
*/
/*
* Put a rectangle on the screen.
*/
static void
put_rule(x, y, w, h)
int x, y;
unsigned int w, h;
{
if (x < max_x && x + w >= min_x && y < max_y && y + h >= min_y) {
#ifndef KDVI
if (--event_counter == 0) read_events(False);
XFillRectangle(DISP, currwin.win, ruleGC,
x - currwin.base_x, y - currwin.base_y, w ? w : 1, h ? h : 1);
#else
qtPutRule( x - currwin.base_x, y - currwin.base_y, (int)w, (int)h );
#endif /* KDVI */
}
}
static void
put_bitmap(bitmap, x, y)
register struct bitmap *bitmap;
register int x, y;
{
if (debug & DBG_BITMAP)
Printf("X(%d,%d)\n", x - currwin.base_x, y - currwin.base_y);
if (x < max_x && x + (int) bitmap->w >= min_x &&
y < max_y && y + (int) bitmap->h >= min_y) {
#ifndef KDVI
if (--event_counter == 0) read_events(False);
image->width = bitmap->w;
image->height = bitmap->h;
image->data = bitmap->bits;
image->bytes_per_line = bitmap->bytes_wide;
XPutImage(DISP, currwin.win, foreGC, image,
0, 0,
x - currwin.base_x, y - currwin.base_y,
bitmap->w, bitmap->h);
if (foreGC2)
XPutImage(DISP, currwin.win, foreGC2, image,
0, 0,
x - currwin.base_x, y - currwin.base_y,
bitmap->w, bitmap->h);
#else
qtPutBitmap( x - currwin.base_x, y - currwin.base_y,
bitmap->bytes_wide*8, bitmap->h, bitmap->bits );
#endif /* KDVI */
}
}
#ifdef GREY
static void
put_image(img, x, y)
register XImage *img;
register int x, y;
{
if (x < max_x && x + img->width >= min_x &&
y < max_y && y + img->height >= min_y) {
#ifndef KDVI
if (--event_counter == 0) read_events (False);
#endif /* KDVI */
XPutImage(DISP, currwin.win, foreGC, img,
0, 0,
x - currwin.base_x, y - currwin.base_y,
(unsigned int) img->width, (unsigned int) img->height);
}
}
#endif /* GREY */
void
put_border(x, y, width, height, ourGC)
int x, y;
unsigned int width, height;
GC ourGC;
{
--width;
--height;
#ifndef KDVI
/* top */
XFillRectangle(DISP, currwin.win, ourGC, x, y, width, 1);
/* right */
XFillRectangle(DISP, currwin.win, ourGC, x + (int) width, y, 1, height);
/* bottom */
XFillRectangle(DISP, currwin.win, ourGC, x + 1, y + (int) height,
width, 1);
/* left */
XFillRectangle(DISP, currwin.win, ourGC, x, y + 1, 1, height);
#else
qtPutBorder( x, y, (int)width, (int)height );
#endif /* KDVI */
}
/*
* Byte reading routines for dvi file.
*/
#define xtell(pos) ((long) (lseek(fileno(dvi_file), 0L, SEEK_CUR) - \
(currinf.end - (pos))))
static ubyte
xxone()
{
if (currinf.virtual) {
++currinf.pos;
return EOP;
}
currinf.end = dvi_buffer +
read(fileno(dvi_file), (char *) (currinf.pos = dvi_buffer),
DVI_BUFFER_LEN);
return currinf.end > dvi_buffer ? *(currinf.pos)++ : EOF;
}
#define xone() (currinf.pos < currinf.end ? *(currinf.pos)++ : xxone())
static unsigned long
xnum(size)
register ubyte size;
{
register long x = 0;
while (size--) x = (x << 8) | xone();
return x;
}
static long
xsnum(size)
register ubyte size;
{
register long x;
#ifdef __STDC__
x = (signed char) xone();
#else
x = xone();
if (x & 0x80) x -= 0x100;
#endif
while (--size) x = (x << 8) | xone();
return x;
}
#define xsfour() xsnum(4)
static void
xskip(offset)
long offset;
{
currinf.pos += offset;
if (!currinf.virtual && currinf.pos > currinf.end)
(void) lseek(fileno(dvi_file), (long) (currinf.pos - currinf.end),
SEEK_CUR);
}
#if NeedVarargsPrototypes
static NORETURN void
tell_oops(_Xconst char *message, ...)
#else
/* VARARGS */
static NORETURN void
tell_oops(va_alist)
va_dcl
#endif
{
#if !NeedVarargsPrototypes
_Xconst char *message;
#endif
va_list args;
Fprintf(stderr, "%s: ", prog);
#if NeedVarargsPrototypes
va_start(args, message);
#else
va_start(args);
message = va_arg(args, _Xconst char *);
#endif
(void) vfprintf(stderr, message, args);
va_end(args);
if (currinf.virtual)
Fprintf(stderr, " in virtual font %s\n", currinf.virtual->fontname);
else
Fprintf(stderr, ", offset %ld\n", xtell(currinf.pos - 1));
#ifdef KDVI
dvi_oops_msg = (message), longjmp(dvi_env, 1); /* dvi_oops */
#endif
exit(1);
}
/*
* Code for debugging options.
*/
static void
print_bitmap(bitmap)
register struct bitmap *bitmap;
{
register BMUNIT *ptr = (BMUNIT *) bitmap->bits;
register int x, y, i;
if (ptr == NULL) oops("print_bitmap called with null pointer.");
Printf("w = %d, h = %d, bytes wide = %d\n",
bitmap->w, bitmap->h, bitmap->bytes_wide);
for (y = 0; y < (int) bitmap->h; ++y) {
for (x = bitmap->bytes_wide; x > 0; x -= BYTES_PER_BMUNIT) {
#ifndef MSBITFIRST
for (i = 0; i < BITS_PER_BMUNIT; ++i)
#else
for (i = BITS_PER_BMUNIT - 1; i >= 0; --i)
#endif
Putchar((*ptr & (1 << i)) ? '@' : ' ');
++ptr;
}
Putchar('\n');
}
}
static void
print_char(ch, g)
ubyte ch;
struct glyph *g;
{
Printf("char %d", ch);
if (ISPRINT(ch))
Printf(" (%c)", ch);
Putchar('\n');
Printf("x = %d, y = %d, dvi = %ld\n", g->x, g->y, g->dvi_adv);
print_bitmap(&g->bitmap);
}
static _Xconst char *dvi_table1[] = {
"SET1", NULL, NULL, NULL, "SETRULE", "PUT1", NULL, NULL,
NULL, "PUTRULE", "NOP", "BOP", "EOP", "PUSH", "POP", "RIGHT1",
"RIGHT2", "RIGHT3", "RIGHT4", "W0", "W1", "W2", "W3", "W4",
"X0", "X1", "X2", "X3", "X4", "DOWN1", "DOWN2", "DOWN3",
"DOWN4", "Y0", "Y1", "Y2", "Y3", "Y4", "Z0", "Z1",
"Z2", "Z3", "Z4"};
static _Xconst char *dvi_table2[] = {
"FNT1", "FNT2", "FNT3", "FNT4", "XXX1", "XXX2", "XXX3", "XXX4",
"FNTDEF1", "FNTDEF2", "FNTDEF3", "FNTDEF4", "PRE", "POST", "POSTPOST",
"SREFL", "EREFL", NULL, NULL, NULL, NULL};
static void
print_dvi(ch)
ubyte ch;
{
_Xconst char *s;
Printf("%4d %4d ", PXL_H, PXL_V);
if (ch <= (ubyte) (SETCHAR0 + 127)) {
Printf("SETCHAR%-3d", ch - SETCHAR0);
if (ISPRINT(ch))
Printf(" (%c)", ch);
Putchar('\n');
return;
}
else if (ch < FNTNUM0) s = dvi_table1[ch - 128];
else if (ch <= (ubyte) (FNTNUM0 + 63)) {
Printf("FNTNUM%d\n", ch - FNTNUM0);
return;
}
else s = dvi_table2[ch - (FNTNUM0 + 64)];
if (s) Puts(s);
else
tell_oops("unknown op-code %d", ch);
}
/*
* Count the number of set bits in a given region of the bitmap
*/
char sample_count[] = {0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4};
static int
sample(bits, bytes_wide, bit_skip, w, h)
BMUNIT *bits;
int bytes_wide, bit_skip, w, h;
{
BMUNIT *ptr, *endp;
register BMUNIT *cp;
int bits_left;
register int n, bit_shift, wid;
ptr = bits + bit_skip / BITS_PER_BMUNIT;
endp = ADD(bits, h * bytes_wide);
bits_left = w;
#ifndef MSBITFIRST
bit_shift = bit_skip % BITS_PER_BMUNIT;
#else
bit_shift = BITS_PER_BMUNIT - bit_skip % BITS_PER_BMUNIT;
#endif
n = 0;
while (bits_left) {
#ifndef MSBITFIRST
wid = BITS_PER_BMUNIT - bit_shift;
#else
wid = bit_shift;
#endif
if (wid > bits_left) wid = bits_left;
if (wid > 4) wid = 4;
#ifdef MSBITFIRST
bit_shift -= wid;
#endif
for (cp = ptr; cp < endp; cp = ADD(cp, bytes_wide))
n += sample_count[(*cp >> bit_shift) & bit_masks[wid]];
#ifndef MSBITFIRST
bit_shift += wid;
if (bit_shift == BITS_PER_BMUNIT) {
bit_shift = 0;
++ptr;
}
#else
if (bit_shift == 0) {
bit_shift = BITS_PER_BMUNIT;
++ptr;
}
#endif
bits_left -= wid;
}
return n;
}
static void
shrink_glyph(g)
register struct glyph *g;
{
int shrunk_bytes_wide, shrunk_height;
int rows_left, rows, init_cols;
int cols_left;
register int cols;
BMUNIT *old_ptr, *new_ptr;
register BMUNIT m, *cp;
int min_sample = shrink_factor * shrink_factor * density / 100;
/* These machinations ensure that the character is shrunk according to
its hot point, rather than its upper left-hand corner. */
g->x2 = g->x / shrink_factor;
init_cols = g->x - g->x2 * shrink_factor;
if (init_cols <= 0) init_cols += shrink_factor;
else ++g->x2;
g->bitmap2.w = g->x2 + ROUNDUP((int) g->bitmap.w - g->x, shrink_factor);
/* include row zero with the positively numbered rows */
cols = g->y + 1; /* spare register variable */
g->y2 = cols / shrink_factor;
rows = cols - g->y2 * shrink_factor;
if (rows <= 0) {
rows += shrink_factor;
--g->y2;
}
g->bitmap2.h = shrunk_height = g->y2 +
ROUNDUP((int) g->bitmap.h - cols, shrink_factor) + 1;
alloc_bitmap(&g->bitmap2);
old_ptr = (BMUNIT *) g->bitmap.bits;
new_ptr = (BMUNIT *) g->bitmap2.bits;
shrunk_bytes_wide = g->bitmap2.bytes_wide;
rows_left = g->bitmap.h;
bzero((char *) new_ptr, shrunk_bytes_wide * shrunk_height);
while (rows_left) {
if (rows > rows_left) rows = rows_left;
cols_left = g->bitmap.w;
#ifndef MSBITFIRST
m = (1 << 0);
#else
m = ((BMUNIT) 1 << (BITS_PER_BMUNIT-1));
#endif
cp = new_ptr;
cols = init_cols;
while (cols_left) {
if (cols > cols_left) cols = cols_left;
if (sample(old_ptr, g->bitmap.bytes_wide,
(int) g->bitmap.w - cols_left, cols, rows)
>= min_sample)
*cp |= m;
#ifndef MSBITFIRST
if (m == ((BMUNIT)1 << (BITS_PER_BMUNIT-1))) {
m = (1 << 0);
++cp;
}
else m <<= 1;
#else
if (m == (1 << 0)) {
m = ((BMUNIT) 1 << (BITS_PER_BMUNIT-1));
++cp;
}
else m >>= 1;
#endif
cols_left -= cols;
cols = shrink_factor;
}
*((char **) &new_ptr) += shrunk_bytes_wide;
*((char **) &old_ptr) += rows * g->bitmap.bytes_wide;
rows_left -= rows;
rows = shrink_factor;
}
g->y2 = g->y / shrink_factor;
if (debug & DBG_BITMAP)
print_bitmap(&g->bitmap2);
}
#ifdef GREY
static void
shrink_glyph_grey(g)
register struct glyph *g;
{
int rows_left, rows, init_cols;
int cols_left;
register int cols;
int x, y;
long thesample;
BMUNIT *old_ptr;
unsigned int size;
/* These machinations ensure that the character is shrunk according to
its hot point, rather than its upper left-hand corner. */
g->x2 = g->x / shrink_factor;
init_cols = g->x - g->x2 * shrink_factor;
if (init_cols <= 0) init_cols += shrink_factor;
else ++g->x2;
g->bitmap2.w = g->x2 + ROUNDUP((int) g->bitmap.w - g->x, shrink_factor);
/* include row zero with the positively numbered rows */
cols = g->y + 1; /* spare register variable */
g->y2 = cols / shrink_factor;
rows = cols - g->y2 * shrink_factor;
if (rows <= 0) {
rows += shrink_factor;
--g->y2;
}
g->bitmap2.h = g->y2 + ROUNDUP((int) g->bitmap.h - cols, shrink_factor)
+ 1;
g->image2 = XCreateImage(DISP, DefaultVisualOfScreen(SCRN),
(unsigned int) DefaultDepthOfScreen(SCRN),
ZPixmap, 0, (char *) NULL,
g->bitmap2.w, g->bitmap2.h,
BITS_PER_BMUNIT, 0);
size = g->image2->bytes_per_line * g->bitmap2.h;
g->pixmap2 = g->image2->data = xmalloc(size != 0 ? size : 1,
"character pixmap");
old_ptr = (BMUNIT *) g->bitmap.bits;
rows_left = g->bitmap.h;
y = 0;
while (rows_left) {
x = 0;
if (rows > rows_left) rows = rows_left;
cols_left = g->bitmap.w;
cols = init_cols;
while (cols_left) {
if (cols > cols_left) cols = cols_left;
thesample = sample(old_ptr, g->bitmap.bytes_wide,
(int) g->bitmap.w - cols_left, cols, rows);
XPutPixel(g->image2, x, y, pixeltbl[thesample]);
cols_left -= cols;
cols = shrink_factor;
x++;
}
*((char **) &old_ptr) += rows * g->bitmap.bytes_wide;
rows_left -= rows;
rows = shrink_factor;
y++;
}
while (y < (int) g->bitmap2.h) {
for (x = 0; x < (int) g->bitmap2.w; x++)
XPutPixel(g->image2, x, y, *pixeltbl);
y++;
}
g->y2 = g->y / shrink_factor;
}
#endif /* GREY */
/*
* Find font #n.
*/
static void
change_font(n)
unsigned long n;
{
register struct tn *tnp;
if (n < currinf.tn_table_len) currinf.fontp = currinf.tn_table[n];
else {
currinf.fontp = NULL;
for (tnp = currinf.tn_head; tnp != NULL; tnp = tnp->next)
if (tnp->TeXnumber == n) {
currinf.fontp = tnp->fontp;
break;
}
}
if (currinf.fontp == NULL) tell_oops("non-existent font #%d", n);
maxchar = currinf.fontp->maxchar;
currinf.set_char_p = currinf.fontp->set_char_p;
}
/*
* Open a font file.
*/
static void
open_font_file(fontp)
struct font *fontp;
{
if (fontp->file == NULL) {
fontp->file = xfopen(fontp->filename, OPEN_MODE);
if (fontp->file == NULL)
oops("Font file disappeared: %s", fontp->filename);
}
}
/*
* Routines to print characters.
*/
#ifndef TEXXET
#define ERRVAL 0L
#else
#define ERRVAL
#endif
#ifndef TEXXET
long
set_char(ch)
#else
void
set_char(cmd, ch)
wide_ubyte cmd;
#endif
wide_ubyte ch;
{
register struct glyph *g;
#ifdef TEXXET
long dvi_h_sav;
#endif
if (ch > maxchar) realloc_font(currinf.fontp, WIDENINT ch);
if ((g = &currinf.fontp->glyph[ch])->bitmap.bits == NULL) {
if (g->addr == 0) {
if (!hush_chars)
Fprintf(stderr, "Character %d not defined in font %s\n", ch,
currinf.fontp->fontname);
g->addr = -1;
return ERRVAL;
}
if (g->addr == -1)
return ERRVAL; /* previously flagged missing char */
open_font_file(currinf.fontp);
Fseek(currinf.fontp->file, g->addr, 0);
(*currinf.fontp->read_char)(currinf.fontp, ch);
if (debug & DBG_BITMAP) print_char((ubyte) ch, g);
currinf.fontp->timestamp = ++current_timestamp;
}
#ifdef TEXXET
dvi_h_sav = DVI_H;
if (currinf.dir < 0) DVI_H -= g->dvi_adv;
if (scan_frame == NULL) {
#endif
if (shrink_factor == 1)
put_bitmap(&g->bitmap, PXL_H - g->x, PXL_V - g->y);
else {
#ifdef GREY
if (_use_grey) {
if (g->pixmap2 == NULL) {
shrink_glyph_grey(g);
}
put_image(g->image2, PXL_H - g->x2, PXL_V - g->y2);
} else {
if (g->bitmap2.bits == NULL) {
shrink_glyph(g);
}
put_bitmap(&g->bitmap2, PXL_H - g->x2, PXL_V - g->y2);
}
#else
if (g->bitmap2.bits == NULL) {
shrink_glyph(g);
}
put_bitmap(&g->bitmap2, PXL_H - g->x2, PXL_V - g->y2);
#endif
}
#ifndef TEXXET
return g->dvi_adv;
#else
}
if (cmd == PUT1)
DVI_H = dvi_h_sav;
else
if (currinf.dir > 0) DVI_H += g->dvi_adv;
#endif
}
/* ARGSUSED */
#if NeedFunctionPrototypes
#ifndef TEXXET
static long
set_empty_char(wide_ubyte ch)
#else
static void
set_empty_char(wide_ubyte cmd, wide_ubyte ch)
#endif /* TEXXET */
#else /* !NeedFunctionPrototypes */
#ifndef TEXXET
static long
set_empty_char(ch)
#else
static void
set_empty_char(cmd, ch)
wide_ubyte cmd;
#endif /* TEXXET */
wide_ubyte ch;
#endif /* NeedFunctionPrototypes */
{
#ifndef TEXXET
return 0;
#else
return;
#endif
}
#ifndef TEXXET
long
load_n_set_char(ch)
#else
void
load_n_set_char(cmd, ch)
wide_ubyte cmd;
#endif
wide_ubyte ch;
{
if (load_font(currinf.fontp)) { /* if not found */
Fputs("Character(s) will be left blank.\n", stderr);
currinf.set_char_p = currinf.fontp->set_char_p = set_empty_char;
#ifndef TEXXET
return 0;
#else
return;
#endif
}
maxchar = currinf.fontp->maxchar;
currinf.set_char_p = currinf.fontp->set_char_p;
#ifndef TEXXET
return (*currinf.set_char_p)(ch);
#else
(*currinf.set_char_p)(cmd, ch);
return;
#endif
}
#ifndef TEXXET
long
set_vf_char(ch)
#else
void
set_vf_char(cmd, ch)
wide_ubyte cmd;
#endif
wide_ubyte ch;
{
register struct macro *m;
struct drawinf oldinfo;
ubyte oldmaxchar;
static ubyte c;
#ifdef TEXXET
long dvi_h_sav;
#endif
if (ch > maxchar) realloc_virtual_font(currinf.fontp, ch);
if ((m = &currinf.fontp->macro[ch])->pos == NULL) {
if (!hush_chars)
Fprintf(stderr, "Character %d not defined in font %s\n", ch,
currinf.fontp->fontname);
m->pos = m->end = &c;
return ERRVAL;
}
#ifdef TEXXET
dvi_h_sav = DVI_H;
if (currinf.dir < 0) DVI_H -= m->dvi_adv;
if (scan_frame == NULL) {
#endif
oldinfo = currinf;
oldmaxchar = maxchar;
WW = XX = YY = ZZ = 0;
currinf.tn_table_len = VFTABLELEN;
currinf.tn_table = currinf.fontp->vf_table;
currinf.tn_head = currinf.fontp->vf_chain;
currinf.pos = m->pos;
currinf.end = m->end;
currinf.virtual = currinf.fontp;
draw_part(current_frame, currinf.fontp->dimconv);
if (currinf.pos != currinf.end + 1)
tell_oops("virtual character macro does not end correctly");
currinf = oldinfo;
maxchar = oldmaxchar;
#ifndef TEXXET
return m->dvi_adv;
#else
}
if (cmd == PUT1)
DVI_H = dvi_h_sav;
else
if (currinf.dir > 0) DVI_H += m->dvi_adv;
#endif
}
#if NeedFunctionPrototypes
#ifndef TEXXET
static long
set_no_char(wide_ubyte ch)
#else
static void
set_no_char(wide_ubyte cmd, wide_ubyte ch)
#endif /* TEXXET */
#else /* !NeedFunctionPrototypes */
#ifndef TEXXET
static long
set_no_char(ch)
#else
static void
set_no_char(cmd, ch)
ubyte cmd;
#endif /* TEXXET */
wide_ubyte ch;
#endif /* NeedFunctionPrototypes */
{
if (currinf.virtual) {
currinf.fontp = currinf.virtual->first_font;
if (currinf.fontp != NULL) {
maxchar = currinf.fontp->maxchar;
currinf.set_char_p = currinf.fontp->set_char_p;
#ifndef TEXXET
return (*currinf.set_char_p)(ch);
#else
(*currinf.set_char_p)(cmd, ch);
return;
#endif
}
}
tell_oops("attempt to set character of unknown font");
/* NOTREACHED */
}
/*
* Set rule. Arguments are coordinates of lower left corner.
*/
static void
set_rule(h, w)
int h, w;
{
#ifndef TEXXET
put_rule(PXL_H, PXL_V - h + 1, (unsigned int) w, (unsigned int) h);
#else
put_rule(PXL_H - (currinf.dir < 0 ? w - 1 : 0), PXL_V - h + 1,
(unsigned int) w, (unsigned int) h);
#endif
}
static void
special(nbytes)
long nbytes;
{
static char *cmd = NULL;
static long cmdlen = -1;
char *p;
if (cmdlen < nbytes) {
if (cmd) free(cmd);
cmd = xmalloc((unsigned) nbytes + 1, "special");
cmdlen = nbytes;
}
p = cmd;
for (;;) {
int i = currinf.end - currinf.pos;
if (i > nbytes) i = nbytes;
bcopy((char *) currinf.pos, p, i);
currinf.pos += i;
p += i;
nbytes -= i;
if (nbytes == 0) break;
(void) xxone();
--(currinf.pos);
}
*p = '\0';
applicationDoSpecial(cmd);
}
#define xspell_conv(n) spell_conv0(n, current_dimconv)
static void
draw_part(minframe, current_dimconv)
struct frame *minframe;
double current_dimconv;
{
ubyte ch;
#ifdef TEXXET
struct drawinf oldinfo;
ubyte oldmaxchar;
off_t file_pos;
int refl_count;
#endif
currinf.fontp = NULL;
currinf.set_char_p = set_no_char;
#ifdef TEXXET
currinf.dir = 1;
scan_frame = NULL; /* indicates we're not scanning */
#endif
for (;;) {
ch = xone();
if (debug & DBG_DVI)
print_dvi(ch);
if (ch <= (ubyte) (SETCHAR0 + 127))
#ifndef TEXXET
DVI_H += (*currinf.set_char_p)(ch);
#else
(*currinf.set_char_p)(ch, ch);
#endif
else if (FNTNUM0 <= ch && ch <= (ubyte) (FNTNUM0 + 63))
change_font((unsigned long) (ch - FNTNUM0));
else {
long a, b;
switch (ch) {
case SET1:
case PUT1:
#ifndef TEXXET
a = (*currinf.set_char_p)(xone());
if (ch != PUT1) DVI_H += a;
#else
(*currinf.set_char_p)(ch, xone());
#endif
break;
case SETRULE:
/* Be careful, dvicopy outputs rules with
height = 0x80000000. We don't want any
SIGFPE here. */
a = xsfour();
b = xspell_conv(xsfour());
#ifndef TEXXET
if (a > 0 && b > 0)
#else
if (a > 0 && b > 0 && scan_frame == NULL)
#endif
set_rule(pixel_round(xspell_conv(a)),
pixel_round(b));
DVI_H += DIR * b;
break;
case PUTRULE:
a = xspell_conv(xsfour());
b = xspell_conv(xsfour());
#ifndef TEXXET
if (a > 0 && b > 0)
#else
if (a > 0 && b > 0 && scan_frame == NULL)
#endif
set_rule(pixel_round(a), pixel_round(b));
break;
case NOP:
break;
case BOP:
xskip((long) 11 * 4);
DVI_H = OFFSET_X;
DVI_V = OFFSET_Y;
PXL_V = pixel_conv(DVI_V);
WW = XX = YY = ZZ = 0;
break;
case EOP:
if (current_frame != minframe)
tell_oops("stack not empty at EOP");
#if PS
psp.endpage();
#endif
return;
case PUSH:
if (current_frame->next == NULL) {
struct frame *newp = (struct frame *)
xmalloc(sizeof(struct frame), "stack frame");
current_frame->next = newp;
newp->prev = current_frame;
newp->next = NULL;
}
current_frame = current_frame->next;
current_frame->data = currinf.data;
break;
case POP:
if (current_frame == minframe)
tell_oops("more POPs than PUSHes");
currinf.data = current_frame->data;
current_frame = current_frame->prev;
break;
#ifdef TEXXET
case SREFL:
if (scan_frame == NULL) {
/* we're not scanning: save some info. */
oldinfo = currinf;
oldmaxchar = maxchar;
if (!currinf.virtual)
file_pos = xtell(currinf.pos);
scan_frame = current_frame; /* now we're scanning */
refl_count = 0;
break;
}
/* we are scanning */
if (current_frame == scan_frame) ++refl_count;
break;
case EREFL:
if (scan_frame != NULL) { /* if we're scanning */
if (current_frame == scan_frame && --refl_count < 0)
{
/* we've hit the end of our scan */
scan_frame = NULL;
/* first: push */
if (current_frame->next == NULL) {
struct frame *newp = (struct frame *)
xmalloc(sizeof(struct frame),
"stack frame");
current_frame->next = newp;
newp->prev = current_frame;
newp->next = NULL;
}
current_frame = current_frame->next;
current_frame->data = currinf.data;
/* next: restore old file position, XX, etc. */
if (!currinf.virtual) {
off_t bgn_pos = xtell(dvi_buffer);
if (file_pos >= bgn_pos) {
oldinfo.pos = dvi_buffer
+ (file_pos - bgn_pos);
oldinfo.end = currinf.end;
}
else {
(void) lseek(fileno(dvi_file), file_pos,
SEEK_SET);
oldinfo.pos = oldinfo.end;
}
}
currinf = oldinfo;
maxchar = oldmaxchar;
/* and then: recover position info. */
DVI_H = current_frame->data.dvi_h;
DVI_V = current_frame->data.dvi_v;
PXL_V = current_frame->data.pxl_v;
/* and finally, reverse direction */
currinf.dir = -currinf.dir;
}
break;
}
/* we're not scanning, */
/* so just reverse direction and then pop */
currinf.dir = -currinf.dir;
currinf.data = current_frame->data;
current_frame = current_frame->prev;
break;
#endif /* TEXXET */
case RIGHT1:
case RIGHT2:
case RIGHT3:
case RIGHT4:
DVI_H += DIR * xspell_conv(xsnum(ch - RIGHT1 + 1));
break;
case W1:
case W2:
case W3:
case W4:
WW = xspell_conv(xsnum(ch - W0));
case W0:
DVI_H += DIR * WW;
break;
case X1:
case X2:
case X3:
case X4:
XX = xspell_conv(xsnum(ch - X0));
case X0:
DVI_H += DIR * XX;
break;
case DOWN1:
case DOWN2:
case DOWN3:
case DOWN4:
DVI_V += xspell_conv(xsnum(ch - DOWN1 + 1));
PXL_V = pixel_conv(DVI_V);
break;
case Y1:
case Y2:
case Y3:
case Y4:
YY = xspell_conv(xsnum(ch - Y0));
case Y0:
DVI_V += YY;
PXL_V = pixel_conv(DVI_V);
break;
case Z1:
case Z2:
case Z3:
case Z4:
ZZ = xspell_conv(xsnum(ch - Z0));
case Z0:
DVI_V += ZZ;
PXL_V = pixel_conv(DVI_V);
break;
case FNT1:
case FNT2:
case FNT3:
case FNT4:
change_font(xnum(ch - FNT1 + 1));
break;
case XXX1:
case XXX2:
case XXX3:
case XXX4:
a = xnum(ch - XXX1 + 1);
if (a > 0)
special(a);
break;
case FNTDEF1:
case FNTDEF2:
case FNTDEF3:
case FNTDEF4:
xskip((long) (12 + ch - FNTDEF1 + 1));
xskip((long) xone() + (long) xone());
break;
#ifndef TEXXET
case SREFL:
case EREFL:
#endif
case PRE:
case POST:
case POSTPOST:
tell_oops("shouldn't happen: %s encountered",
dvi_table2[ch - (FNTNUM0 + 64)]);
break;
default:
tell_oops("unknown op-code %d", ch);
} /* end switch*/
} /* end else (ch not a SETCHAR or FNTNUM) */
} /* end for */
}
#undef xspell_conv
void
draw_page()
{
/* Check for changes in dvi file. */
if (!check_dvi_file()) return;
put_border(-currwin.base_x, -currwin.base_y,
ROUNDUP(unshrunk_paper_w, shrink_factor) + 2,
ROUNDUP(unshrunk_paper_h, shrink_factor) + 2, highGC);
(void) lseek(fileno(dvi_file), page_offset[current_page], SEEK_SET);
bzero((char *) &currinf.data, sizeof(currinf.data));
currinf.tn_table_len = TNTABLELEN;
currinf.tn_table = tn_table;
currinf.tn_head = tn_head;
currinf.pos = currinf.end = dvi_buffer;
currinf.virtual = NULL;
draw_part(current_frame = &frame0, dimconv);
}