// -*- C++ -*- /* Copyright (C) 1989, 1990, 1991, 1992 Free Software Foundation, Inc. Written by James Clark (jjc@jclark.com) This file is part of groff. groff 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, or (at your option) any later version. groff 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 groff; see the file COPYING. If not, write to the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "driver.h" #include "stringclass.h" #include "cset.h" #include "ps.h" #include static int landscape_flag = 0; static int ncopies = 1; static int linewidth = -1; // Non-zero means generate PostScript code that guesses the paper // length using the imageable area. static int guess_flag = 0; // Non-zero if -b was specified on the command line. static int bflag = 0; unsigned broken_flags = 0; #define DEFAULT_LINEWIDTH 40 /* in ems/1000 */ #define MAX_LINE_LENGTH 72 #define FILL_MAX 1000 const char *const dict_name = "grops"; const char *const defs_dict_name = "DEFS"; const int DEFS_DICT_SPARE = 50; double degrees(double r) { return r*180.0/PI; } double radians(double d) { return d*PI/180.0; } inline double transform_fill(int fill) { return 1 - fill/double(FILL_MAX); } // This is used for testing whether a character should be output in the // PostScript file using \nnn, so we really want the character to be // less than 0200. inline int is_ascii(char c) { return (unsigned char)c < 0200; } ps_output::ps_output(FILE *f, int n) : fp(f), max_line_length(n), col(0), need_space(0), fixed_point(0) { } ps_output &ps_output::set_file(FILE *f) { fp = f; col = 0; return *this; } ps_output &ps_output::copy_file(FILE *infp) { int c; while ((c = getc(infp)) != EOF) putc(c, fp); return *this; } ps_output &ps_output::end_line() { if (col != 0) { putc('\n', fp); col = 0; need_space = 0; } return *this; } ps_output &ps_output::special(const char *s) { if (s == 0 || *s == '\0') return *this; if (col != 0) { putc('\n', fp); col = 0; } fputs(s, fp); if (strchr(s, '\0')[-1] != '\n') putc('\n', fp); need_space = 0; return *this; } ps_output &ps_output::simple_comment(const char *s) { if (col != 0) putc('\n', fp); putc('%', fp); putc('%', fp); fputs(s, fp); putc('\n', fp); col = 0; need_space = 0; return *this; } ps_output &ps_output::begin_comment(const char *s) { if (col != 0) putc('\n', fp); putc('%', fp); putc('%', fp); fputs(s, fp); col = 2 + strlen(s); return *this; } ps_output &ps_output::end_comment() { if (col != 0) { putc('\n', fp); col = 0; } need_space = 0; return *this; } ps_output &ps_output::comment_arg(const char *s) { int len = strlen(s); if (col + len + 1 > max_line_length) { putc('\n', fp); fputs("%%+", fp); col = 3; } putc(' ', fp); fputs(s, fp); col += len + 1; return *this; } ps_output &ps_output::set_fixed_point(int n) { assert(n >= 0 && n <= 10); fixed_point = n; return *this; } ps_output &ps_output::put_delimiter(char c) { if (col + 1 > max_line_length) { putc('\n', fp); col = 0; } putc(c, fp); col++; need_space = 0; return *this; } ps_output &ps_output::put_string(const char *s, int n) { int len = 0; for (int i = 0; i < n; i++) { char c = s[i]; if (is_ascii(c) && csprint(c)) { if (c == '(' || c == ')' || c == '\\') len += 2; else len += 1; } else len += 4; } if (len > n*2) { if (col + n*2 + 2 > max_line_length && n*2 + 2 <= max_line_length) { putc('\n', fp); col = 0; } if (col + 1 > max_line_length) { putc('\n', fp); col = 0; } putc('<', fp); col++; for (i = 0; i < n; i++) { if (col + 2 > max_line_length) { putc('\n', fp); col = 0; } fprintf(fp, "%02x", s[i] & 0377); col += 2; } putc('>', fp); col++; } else { if (col + len + 2 > max_line_length && len + 2 <= max_line_length) { putc('\n', fp); col = 0; } if (col + 2 > max_line_length) { putc('\n', fp); col = 0; } putc('(', fp); col++; for (i = 0; i < n; i++) { char c = s[i]; if (is_ascii(c) && csprint(c)) { if (c == '(' || c == ')' || c == '\\') len = 2; else len = 1; } else len = 4; if (col + len + 1 > max_line_length) { putc('\\', fp); putc('\n', fp); col = 0; } switch (len) { case 1: putc(c, fp); break; case 2: putc('\\', fp); putc(c, fp); break; case 4: fprintf(fp, "\\%03o", c & 0377); break; default: assert(0); } col += len; } putc(')', fp); col++; } need_space = 0; return *this; } ps_output &ps_output::put_number(int n) { char buf[1 + INT_DIGITS + 1]; sprintf(buf, "%d", n); int len = strlen(buf); if (col > 0 && col + len + need_space > max_line_length) { putc('\n', fp); col = 0; need_space = 0; } if (need_space) { putc(' ', fp); col++; } fputs(buf, fp); col += len; need_space = 1; return *this; } ps_output &ps_output::put_fix_number(int i) { const char *p = iftoa(i, fixed_point); int len = strlen(p); if (col > 0 && col + len + need_space > max_line_length) { putc('\n', fp); col = 0; need_space = 0; } if (need_space) { putc(' ', fp); col++; } fputs(p, fp); col += len; need_space = 1; return *this; } ps_output &ps_output::put_float(double d) { char buf[128]; sprintf(buf, "%.4f", d); int len = strlen(buf); if (col > 0 && col + len + need_space > max_line_length) { putc('\n', fp); col = 0; need_space = 0; } if (need_space) { putc(' ', fp); col++; } fputs(buf, fp); col += len; need_space = 1; return *this; } ps_output &ps_output::put_symbol(const char *s) { int len = strlen(s); if (col > 0 && col + len + need_space > max_line_length) { putc('\n', fp); col = 0; need_space = 0; } if (need_space) { putc(' ', fp); col++; } fputs(s, fp); col += len; need_space = 1; return *this; } ps_output &ps_output::put_literal_symbol(const char *s) { int len = strlen(s); if (col > 0 && col + len + 1 > max_line_length) { putc('\n', fp); col = 0; } putc('/', fp); fputs(s, fp); col += len + 1; need_space = 1; return *this; } class ps_font : public font { ps_font(const char *); public: int encoding_index; char *encoding; char *reencoded_name; ~ps_font(); void handle_unknown_font_command(const char *command, const char *arg, const char *filename, int lineno); static ps_font *load_ps_font(const char *); }; ps_font *ps_font::load_ps_font(const char *s) { ps_font *f = new ps_font(s); if (!f->load()) { delete f; return 0; } return f; } ps_font::ps_font(const char *nm) : font(nm), encoding(0), reencoded_name(0), encoding_index(-1) { } ps_font::~ps_font() { a_delete encoding; a_delete reencoded_name; } void ps_font::handle_unknown_font_command(const char *command, const char *arg, const char *filename, int lineno) { if (strcmp(command, "encoding") == 0) { if (arg == 0) error_with_file_and_line(filename, lineno, "`encoding' command requires an argument"); else encoding = strsave(arg); } } static void handle_unknown_desc_command(const char *command, const char *arg, const char *filename, int lineno) { if (strcmp(command, "broken") == 0) { if (arg == 0) error_with_file_and_line(filename, lineno, "`broken' command requires an argument"); else if (!bflag) broken_flags = atoi(arg); } } struct style { font *f; int point_size; int height; int slant; style(); style(font *, int, int, int); int operator==(const style &) const; int operator!=(const style &) const; }; style::style() : f(0) { } style::style(font *p, int sz, int h, int sl) : f(p), point_size(sz), height(h), slant(sl) { } int style::operator==(const style &s) const { return (f == s.f && point_size == s.point_size && height == s.height && slant == s.slant); } int style::operator!=(const style &s) const { return !(*this == s); } class ps_printer : public printer { FILE *tempfp; ps_output out; int res; int space_char_index; int pages_output; int paper_length; int equalise_spaces; enum { SBUF_SIZE = 256 }; char sbuf[SBUF_SIZE]; int sbuf_len; int sbuf_start_hpos; int sbuf_vpos; int sbuf_end_hpos; int sbuf_space_width; int sbuf_space_count; int sbuf_space_diff_count; int sbuf_space_code; int sbuf_kern; style sbuf_style; style output_style; int output_hpos; int output_vpos; int output_draw_point_size; int line_thickness; int output_line_thickness; int fill; unsigned char output_space_code; enum { MAX_DEFINED_STYLES = 50 }; style defined_styles[MAX_DEFINED_STYLES]; int ndefined_styles; int next_encoding_index; string defs; int ndefs; resource_manager rm; int invis_count; void flush_sbuf(); void set_style(const style &); void set_space_code(unsigned char c); int set_encoding_index(ps_font *); void do_exec(char *, const environment *); void do_import(char *, const environment *); void do_def(char *, const environment *); void do_mdef(char *, const environment *); void do_file(char *, const environment *); void do_invis(char *, const environment *); void do_endinvis(char *, const environment *); void set_line_thickness(const environment *); void fill_path(); void encode_fonts(); void define_encoding(const char *, int); void reencode_font(ps_font *); public: ps_printer(); ~ps_printer(); void set_char(int i, font *f, const environment *env, int w); void draw(int code, int *p, int np, const environment *env); void begin_page(int); void end_page(int); void special(char *arg, const environment *env); font *make_font(const char *); void end_of_line(); }; ps_printer::ps_printer() : pages_output(0), sbuf_len(0), output_hpos(-1), output_vpos(-1), out(0, MAX_LINE_LENGTH), ndefined_styles(0), next_encoding_index(0), line_thickness(-1), fill(FILL_MAX + 1), ndefs(0), invis_count(0) { tempfp = xtmpfile(); out.set_file(tempfp); if (linewidth < 0) linewidth = DEFAULT_LINEWIDTH; if (font::hor != 1) fatal("horizontal resolution must be 1"); if (font::vert != 1) fatal("vertical resolution must be 1"); if (font::res % (font::sizescale*72) != 0) fatal("res must be a multiple of 72*sizescale"); int r = font::res; int point = 0; while (r % 10 == 0) { r /= 10; point++; } res = r; out.set_fixed_point(point); space_char_index = font::name_to_index("space"); paper_length = font::paperlength; if (paper_length == 0) paper_length = 11*font::res; equalise_spaces = font::res >= 72000; } int ps_printer::set_encoding_index(ps_font *f) { if (f->encoding_index >= 0) return f->encoding_index; for (font_pointer_list *p = font_list; p; p = p->next) if (p->p != f) { char *encoding = ((ps_font *)p->p)->encoding; int encoding_index = ((ps_font *)p->p)->encoding_index; if (encoding != 0 && encoding_index >= 0 && strcmp(f->encoding, encoding) == 0) { return f->encoding_index = encoding_index; } } return f->encoding_index = next_encoding_index++; } void ps_printer::set_char(int i, font *f, const environment *env, int w) { if (i == space_char_index || invis_count > 0) return; unsigned char code = f->get_code(i); style sty(f, env->size, env->height, env->slant); if (sty.slant != 0) { if (sty.slant > 80 || sty.slant < -80) { error("silly slant `%1' degrees", sty.slant); sty.slant = 0; } } if (sbuf_len > 0) { if (sbuf_len < SBUF_SIZE && sty == sbuf_style && sbuf_vpos == env->vpos) { if (sbuf_end_hpos == env->hpos) { sbuf[sbuf_len++] = code; sbuf_end_hpos += w + sbuf_kern; return; } if (sbuf_len == 1 && sbuf_kern == 0) { sbuf_kern = env->hpos - sbuf_end_hpos; sbuf_end_hpos = env->hpos + sbuf_kern + w; sbuf[sbuf_len++] = code; return; } /* If sbuf_end_hpos - sbuf_kern == env->hpos, we are better off starting a new string. */ if (sbuf_len < SBUF_SIZE - 1 && env->hpos >= sbuf_end_hpos && (sbuf_kern == 0 || sbuf_end_hpos - sbuf_kern != env->hpos)) { if (sbuf_space_code < 0) { if (f->contains(space_char_index)) { sbuf_space_code = f->get_code(space_char_index); sbuf_space_width = env->hpos - sbuf_end_hpos; sbuf_end_hpos = env->hpos + w + sbuf_kern; sbuf[sbuf_len++] = sbuf_space_code; sbuf[sbuf_len++] = code; sbuf_space_count++; return; } } else { int diff = env->hpos - sbuf_end_hpos - sbuf_space_width; if (diff == 0 || (equalise_spaces && (diff == 1 || diff == -1))) { sbuf_end_hpos = env->hpos + w + sbuf_kern; sbuf[sbuf_len++] = sbuf_space_code; sbuf[sbuf_len++] = code; sbuf_space_count++; if (diff == 1) sbuf_space_diff_count++; else if (diff == -1) sbuf_space_diff_count--; return; } } } } flush_sbuf(); } sbuf_len = 1; sbuf[0] = code; sbuf_end_hpos = env->hpos + w; sbuf_start_hpos = env->hpos; sbuf_vpos = env->vpos; sbuf_style = sty; sbuf_space_code = -1; sbuf_space_width = 0; sbuf_space_count = sbuf_space_diff_count = 0; sbuf_kern = 0; } static char *make_encoding_name(int encoding_index) { static char buf[3 + INT_DIGITS + 1]; sprintf(buf, "ENC%d", encoding_index); return buf; } const char *const WS = " \t\n\r"; void ps_printer::define_encoding(const char *encoding, int encoding_index) { char *vec[256]; for (int i = 0; i < 256; i++) vec[i] = 0; char *path; FILE *fp = font::open_file(encoding, &path); if (fp == 0) fatal("can't open encoding file `%1'", encoding); int lineno = 1; char buf[256]; while (fgets(buf, 512, fp) != 0) { char *p = buf; while (csspace(*p)) p++; if (*p != '#' && *p != '\0' && (p = strtok(buf, WS)) != 0) { char *q = strtok(0, WS); int n; if (q == 0 || sscanf(q, "%d", &n) != 1 || n < 0 || n >= 256) fatal_with_file_and_line(path, lineno, "bad second field"); vec[n] = new char[strlen(p) + 1]; strcpy(vec[n], p); } lineno++; } a_delete path; out.put_literal_symbol(make_encoding_name(encoding_index)); out.put_delimiter('['); for (i = 0; i < 256; i++) { if (vec[i] == 0) out.put_literal_symbol(".notdef"); else { out.put_literal_symbol(vec[i]); a_delete vec[i]; } } out.put_delimiter(']').put_symbol("def"); } void ps_printer::reencode_font(ps_font *f) { out.put_literal_symbol(f->reencoded_name) .put_symbol(make_encoding_name(f->encoding_index)) .put_literal_symbol(f->get_internal_name()) .put_symbol("RE"); } void ps_printer::encode_fonts() { if (next_encoding_index == 0) return; char *done_encoding = new char[next_encoding_index]; for (int i = 0; i < next_encoding_index; i++) done_encoding[i] = 0; for (font_pointer_list *f = font_list; f; f = f->next) { int encoding_index = ((ps_font *)f->p)->encoding_index; if (encoding_index >= 0) { assert(encoding_index < next_encoding_index); if (!done_encoding[encoding_index]) { done_encoding[encoding_index] = 1; define_encoding(((ps_font *)f->p)->encoding, encoding_index); } reencode_font((ps_font *)f->p); } } a_delete done_encoding; } void ps_printer::set_style(const style &sty) { char buf[1 + INT_DIGITS + 1]; for (int i = 0; i < ndefined_styles; i++) if (sty == defined_styles[i]) { sprintf(buf, "F%d", i); out.put_symbol(buf); return; } if (ndefined_styles >= MAX_DEFINED_STYLES) ndefined_styles = 0; sprintf(buf, "F%d", ndefined_styles); out.put_literal_symbol(buf); const char *psname = sty.f->get_internal_name(); if (psname == 0) fatal("no internalname specified for font `%1'", sty.f->get_name()); char *encoding = ((ps_font *)sty.f)->encoding; if (encoding != 0) { char *s = ((ps_font *)sty.f)->reencoded_name; if (s == 0) { int ei = set_encoding_index((ps_font *)sty.f); char *tem = new char[strlen(psname) + 1 + INT_DIGITS + 1]; sprintf(tem, "%s@%d", psname, ei); psname = tem; ((ps_font *)sty.f)->reencoded_name = tem; } else psname = s; } out.put_fix_number((font::res/(72*font::sizescale))*sty.point_size); if (sty.height != 0 || sty.slant != 0) { int h = sty.height == 0 ? sty.point_size : sty.height; h *= font::res/(72*font::sizescale); int c = int(h*tan(radians(sty.slant)) + .5); out.put_fix_number(c).put_fix_number(h).put_literal_symbol(psname) .put_symbol("MF"); } else { out.put_literal_symbol(psname).put_symbol("SF"); } defined_styles[ndefined_styles++] = sty; } void ps_printer::set_space_code(unsigned char c) { out.put_literal_symbol("SC").put_number(c).put_symbol("def"); } void ps_printer::end_of_line() { flush_sbuf(); // this ensures that we do an absolute motion to the beginning of a line output_vpos = output_hpos = -1; } void ps_printer::flush_sbuf() { enum { NONE, RELATIVE_H, RELATIVE_V, RELATIVE_HV, ABSOLUTE } motion = NONE; int space_flag = 0; if (sbuf_len == 0) return; if (output_style != sbuf_style) { set_style(sbuf_style); output_style = sbuf_style; } int extra_space = 0; if (output_hpos < 0 || output_vpos < 0) motion = ABSOLUTE; else { if (output_hpos != sbuf_start_hpos) motion = RELATIVE_H; if (output_vpos != sbuf_vpos) { if (motion != NONE) motion = RELATIVE_HV; else motion = RELATIVE_V; } } if (sbuf_space_code >= 0) { int w = sbuf_style.f->get_width(space_char_index, sbuf_style.point_size); if (w + sbuf_kern != sbuf_space_width) { if (sbuf_space_code != output_space_code) { set_space_code(sbuf_space_code); output_space_code = sbuf_space_code; } space_flag = 1; extra_space = sbuf_space_width - w - sbuf_kern; if (sbuf_space_diff_count > sbuf_space_count/2) extra_space++; else if (sbuf_space_diff_count < -(sbuf_space_count/2)) extra_space--; } } if (space_flag) out.put_fix_number(extra_space); if (sbuf_kern != 0) out.put_fix_number(sbuf_kern); out.put_string(sbuf, sbuf_len); char sym[2]; sym[0] = 'A' + motion*4 + space_flag + 2*(sbuf_kern != 0); sym[1] = '\0'; switch (motion) { case NONE: break; case ABSOLUTE: out.put_fix_number(sbuf_start_hpos) .put_fix_number(sbuf_vpos); break; case RELATIVE_H: out.put_fix_number(sbuf_start_hpos - output_hpos); break; case RELATIVE_V: out.put_fix_number(sbuf_vpos - output_vpos); break; case RELATIVE_HV: out.put_fix_number(sbuf_start_hpos - output_hpos) .put_fix_number(sbuf_vpos - output_vpos); break; default: assert(0); } out.put_symbol(sym); output_hpos = sbuf_end_hpos; output_vpos = sbuf_vpos; sbuf_len = 0; } void ps_printer::set_line_thickness(const environment *env) { if (line_thickness < 0) { if (output_draw_point_size != env->size) { // we ought to check for overflow here int lw = ((font::res/(72*font::sizescale))*linewidth*env->size)/1000; out.put_fix_number(lw).put_symbol("LW"); output_draw_point_size = env->size; output_line_thickness = -1; } } else { if (output_line_thickness != line_thickness) { out.put_fix_number(line_thickness).put_symbol("LW"); output_line_thickness = line_thickness; output_draw_point_size = -1; } } } void ps_printer::fill_path() { if (fill > FILL_MAX) out.put_symbol("BL"); else out.put_float(transform_fill(fill)).put_symbol("FL"); } void ps_printer::draw(int code, int *p, int np, const environment *env) { if (invis_count > 0) return; int fill_flag = 0; switch (code) { case 'C': fill_flag = 1; // fall through case 'c': // troff adds an extra argument to C if (np != 1 && !(code == 'C' && np == 2)) { error("1 argument required for circle"); break; } out.put_fix_number(env->hpos + p[0]/2) .put_fix_number(env->vpos) .put_fix_number(p[0]/2) .put_symbol("DC"); if (fill_flag) { fill_path(); } else { set_line_thickness(env); out.put_symbol("ST"); } break; case 'l': if (np != 2) { error("2 arguments required for line"); break; } set_line_thickness(env); out.put_fix_number(p[0] + env->hpos) .put_fix_number(p[1] + env->vpos) .put_fix_number(env->hpos) .put_fix_number(env->vpos) .put_symbol("DL"); break; case 'E': fill_flag = 1; // fall through case 'e': if (np != 2) { error("2 arguments required for ellipse"); break; } out.put_fix_number(p[0]) .put_fix_number(p[1]) .put_fix_number(env->hpos + p[0]/2) .put_fix_number(env->vpos) .put_symbol("DE"); if (fill_flag) { fill_path(); } else { set_line_thickness(env); out.put_symbol("ST"); } break; case 'P': fill_flag = 1; // fall through case 'p': { if (np & 1) { error("even number of arguments required for polygon"); break; } if (np == 0) { error("no arguments for polygon"); break; } out.put_fix_number(env->hpos) .put_fix_number(env->vpos) .put_symbol("MT"); for (int i = 0; i < np; i += 2) out.put_fix_number(p[i]) .put_fix_number(p[i+1]) .put_symbol("RL"); out.put_symbol("CL"); if (fill_flag) { fill_path(); } else { set_line_thickness(env); out.put_symbol("ST"); } break; } case '~': { if (np & 1) { error("even number of arguments required for spline"); break; } if (np == 0) { error("no arguments for spline"); break; } out.put_fix_number(env->hpos) .put_fix_number(env->vpos) .put_symbol("MT"); out.put_fix_number(p[0]/2) .put_fix_number(p[1]/2) .put_symbol("RL"); /* tnum/tden should be between 0 and 1; the closer it is to 1 the tighter the curve will be to the guiding lines; 2/3 is the standard value */ const int tnum = 2; const int tden = 3; for (int i = 0; i < np - 2; i += 2) { out.put_fix_number((p[i]*tnum)/(2*tden)) .put_fix_number((p[i + 1]*tnum)/(2*tden)) .put_fix_number(p[i]/2 + (p[i + 2]*(tden - tnum))/(2*tden)) .put_fix_number(p[i + 1]/2 + (p[i + 3]*(tden - tnum))/(2*tden)) .put_fix_number((p[i] - p[i]/2) + p[i + 2]/2) .put_fix_number((p[i + 1] - p[i + 1]/2) + p[i + 3]/2) .put_symbol("RC"); } out.put_fix_number(p[np - 2] - p[np - 2]/2) .put_fix_number(p[np - 1] - p[np - 1]/2) .put_symbol("RL"); set_line_thickness(env); out.put_symbol("ST"); } break; case 'a': { if (np != 4) { error("4 arguments required for arc"); break; } set_line_thickness(env); double c[2]; if (adjust_arc_center(p, c)) out.put_fix_number(env->hpos + int(c[0])) .put_fix_number(env->vpos + int(c[1])) .put_fix_number(int(sqrt(c[0]*c[0] + c[1]*c[1]))) .put_float(degrees(atan2(-c[1], -c[0]))) .put_float(degrees(atan2(p[1] + p[3] - c[1], p[0] + p[2] - c[0]))) .put_symbol("DA"); else out.put_fix_number(p[0] + p[2] + env->hpos) .put_fix_number(p[1] + p[3] + env->vpos) .put_fix_number(env->hpos) .put_fix_number(env->vpos) .put_symbol("DL"); } break; case 't': { if (np == 0) { line_thickness = -1; } else { // troff gratuitously adds an extra 0 if (np != 1 && np != 2) { error("0 or 1 argument required for thickness"); break; } line_thickness = p[0]; } break; } case 'f': { if (np != 1 && np != 2) { error("1 argument required for fill"); break; } fill = p[0]; if (fill < 0 || fill > FILL_MAX) { // This means fill with the current color. fill = FILL_MAX + 1; } break; } default: error("unrecognised drawing command `%1'", char(code)); break; } output_hpos = output_vpos = -1; } void ps_printer::begin_page(int n) { out.begin_comment("Page:").comment_arg(itoa(n)); out.comment_arg(itoa(++pages_output)).end_comment(); output_style.f = 0; output_space_code = 32; output_draw_point_size = -1; output_line_thickness = -1; output_hpos = output_vpos = -1; ndefined_styles = 0; out.simple_comment("BeginPageSetup"); out.put_symbol("BP"); out.simple_comment("EndPageSetup"); } void ps_printer::end_page(int) { flush_sbuf(); out.put_symbol("EP"); if (invis_count != 0) { error("missing `endinvis' command"); invis_count = 0; } } font *ps_printer::make_font(const char *nm) { return ps_font::load_ps_font(nm); } ps_printer::~ps_printer() { out.simple_comment("Trailer"); out.put_symbol("end"); out.simple_comment("EOF"); if (fseek(tempfp, 0L, 0) < 0) fatal("fseek on temporary file failed"); fputs("%!PS-Adobe-", stdout); fputs((broken_flags & USE_PS_ADOBE_2_0) ? "2.0" : "3.0", stdout); putchar('\n'); out.set_file(stdout); { extern const char *version_string; out.begin_comment("Creator:") .comment_arg("groff") .comment_arg("version") .comment_arg(version_string) .end_comment(); } { fputs("%%CreationDate: ", out.get_file()); #ifdef LONG_FOR_TIME_T long #else time_t #endif t = time(0); fputs(ctime(&t), out.get_file()); } for (font_pointer_list *f = font_list; f; f = f->next) { ps_font *psf = (ps_font *)(f->p); rm.need_font(psf->get_internal_name()); } rm.print_header_comments(out); out.begin_comment("Pages:").comment_arg(itoa(pages_output)).end_comment(); out.begin_comment("PageOrder:").comment_arg("Ascend").end_comment(); #if 0 fprintf(out.get_file(), "%%%%DocumentMedia: () %g %g 0 () ()\n", font::paperwidth*72.0/font::res, paper_length*72.0/font::res); #endif out.begin_comment("Orientation:") .comment_arg(landscape_flag ? "Landscape" : "Portrait") .end_comment(); if (ncopies != 1) { out.end_line(); fprintf(out.get_file(), "%%%%Requirements: numcopies(%d)\n", ncopies); } out.simple_comment("EndComments"); out.simple_comment("BeginProlog"); rm.output_prolog(out); if (!(broken_flags & NO_SETUP_SECTION)) { out.simple_comment("EndProlog"); out.simple_comment("BeginSetup"); } rm.document_setup(out); out.put_symbol(dict_name).put_symbol("begin"); if (ndefs > 0) ndefs += DEFS_DICT_SPARE; out.put_literal_symbol(defs_dict_name) .put_number(ndefs + 1) .put_symbol("dict") .put_symbol("def"); out.put_symbol(defs_dict_name) .put_symbol("begin"); out.put_literal_symbol("u") .put_delimiter('{') .put_fix_number(1) .put_symbol("mul") .put_delimiter('}') .put_symbol("bind") .put_symbol("def"); defs += '\0'; out.special(defs.contents()); out.put_symbol("end"); if (ncopies != 1) out.put_literal_symbol("#copies").put_number(ncopies).put_symbol("def"); out.put_literal_symbol("RES").put_number(res).put_symbol("def"); out.put_literal_symbol("PL"); if (guess_flag) out.put_symbol("PLG"); else out.put_fix_number(paper_length); out.put_symbol("def"); out.put_literal_symbol("LS") .put_symbol(landscape_flag ? "true" : "false") .put_symbol("def"); encode_fonts(); out.simple_comment((broken_flags & NO_SETUP_SECTION) ? "EndProlog" : "EndSetup"); out.end_line(); out.copy_file(tempfp); fclose(tempfp); } void ps_printer::special(char *arg, const environment *env) { typedef void (ps_printer::*SPECIAL_PROCP)(char *, const environment *); static struct { const char *name; SPECIAL_PROCP proc; } proc_table[] = { { "exec", &ps_printer::do_exec }, { "def", &ps_printer::do_def }, { "mdef", &ps_printer::do_mdef }, { "import", &ps_printer::do_import }, { "file", &ps_printer::do_file }, { "invis", &ps_printer::do_invis }, { "endinvis", &ps_printer::do_endinvis }, }; for (char *p = arg; *p == ' ' || *p == '\n'; p++) ; char *tag = p; for (; *p != '\0' && *p != ':' && *p != ' ' && *p != '\n'; p++) ; if (*p == '\0' || strncmp(tag, "ps", p - tag) != 0) { error("X command without `ps:' tag ignored"); return; } p++; for (; *p == ' ' || *p == '\n'; p++) ; char *command = p; for (; *p != '\0' && *p != ' ' && *p != '\n'; p++) ; if (*command == '\0') { error("X command without `ps:' tag ignored"); return; } for (int i = 0; i < sizeof(proc_table)/sizeof(proc_table[0]); i++) if (strncmp(command, proc_table[i].name, p - command) == 0) { (this->*(proc_table[i].proc))(p, env); return; } error("X command `%1' not recognised", command); } // A conforming PostScript document must not have lines longer // than 255 characters (excluding line termination characters). static int check_line_lengths(const char *p) { for (;;) { const char *end = strchr(p, '\n'); if (end == 0) end = strchr(p, '\0'); if (end - p > 255) return 0; if (*end == '\0') break; p = end + 1; } return 1; } void ps_printer::do_exec(char *arg, const environment *env) { flush_sbuf(); while (csspace(*arg)) arg++; if (*arg == '\0') { error("missing argument to X exec command"); return; } if (!check_line_lengths(arg)) { error("lines in X exec command must not be more than 255 characters long"); return; } out.put_fix_number(env->hpos) .put_fix_number(env->vpos) .put_symbol("EBEGIN") .special(arg) .put_symbol("EEND"); output_hpos = output_vpos = -1; output_style.f = 0; output_draw_point_size = -1; output_line_thickness = -1; ndefined_styles = 0; if (!ndefs) ndefs = 1; } void ps_printer::do_file(char *arg, const environment *env) { flush_sbuf(); while (csspace(*arg)) arg++; if (*arg == '\0') { error("missing argument to X file command"); return; } const char *filename = arg; do { ++arg; } while (*arg != '\0' && *arg != ' ' && *arg != '\n'); out.put_fix_number(env->hpos) .put_fix_number(env->vpos) .put_symbol("EBEGIN"); rm.import_file(filename, out); out.put_symbol("EEND"); output_hpos = output_vpos = -1; output_style.f = 0; output_draw_point_size = -1; output_line_thickness = -1; ndefined_styles = 0; if (!ndefs) ndefs = 1; } void ps_printer::do_def(char *arg, const environment *) { flush_sbuf(); while (csspace(*arg)) arg++; if (!check_line_lengths(arg)) { error("lines in X def command must not be more than 255 characters long"); return; } defs += arg; if (*arg != '\0' && strchr(arg, '\0')[-1] != '\n') defs += '\n'; ndefs++; } // Like def, but the first argument says how many definitions it contains. void ps_printer::do_mdef(char *arg, const environment *) { flush_sbuf(); char *p; int n = (int)strtol(arg, &p, 10); if (n == 0 && p == arg) { error("first argument to X mdef must be an integer"); return; } if (n < 0) { error("out of range argument `%1' to X mdef command", int(n)); return; } arg = p; while (csspace(*arg)) arg++; if (!check_line_lengths(arg)) { error("lines in X mdef command must not be more than 255 characters long"); return; } defs += arg; if (*arg != '\0' && strchr(arg, '\0')[-1] != '\n') defs += '\n'; ndefs += n; } void ps_printer::do_import(char *arg, const environment *env) { flush_sbuf(); while (*arg == ' ' || *arg == '\n') arg++; for (char *p = arg; *p != '\0' && *p != ' ' && *p != '\n'; p++) ; if (*p != '\0') *p++ = '\0'; int parms[6]; int nparms = 0; while (nparms < 6) { char *end; long n = strtol(p, &end, 10); if (n == 0 && end == p) break; parms[nparms++] = int(n); p = end; } if (csalpha(*p) && (p[1] == '\0' || p[1] == ' ' || p[1] == '\n')) { error("scaling indicators not allowed in arguments for X import command"); return; } while (*p == ' ' || *p == '\n') p++; if (nparms < 5) { if (*p == '\0') error("too few arguments for X import command"); else error("invalid argument `%1' for X import command", p); return; } if (*p != '\0') { error("superflous argument `%1' for X import command", p); return; } int llx = parms[0]; int lly = parms[1]; int urx = parms[2]; int ury = parms[3]; int desired_width = parms[4]; int desired_height = parms[5]; if (desired_width <= 0) { error("bad width argument `%1' for X import command: must be > 0", desired_width); return; } if (nparms == 6 && desired_height <= 0) { error("bad height argument `%1' for X import command: must be > 0", desired_height); return; } if (llx == urx) { error("llx and urx arguments for X import command must not be equal"); return; } if (lly == ury) { error("lly and ury arguments for X import command must not be equal"); return; } if (nparms == 5) { int old_wid = urx - llx; int old_ht = ury - lly; if (old_wid < 0) old_wid = -old_wid; if (old_ht < 0) old_ht = -old_ht; desired_height = int(desired_width*(double(old_ht)/double(old_wid)) + .5); } if (env->vpos - desired_height < 0) warning("top of imported graphic is above the top of the page"); out.put_number(llx) .put_number(lly) .put_fix_number(desired_width) .put_number(urx - llx) .put_fix_number(-desired_height) .put_number(ury - lly) .put_fix_number(env->hpos) .put_fix_number(env->vpos) .put_symbol("PBEGIN"); rm.import_file(arg, out); // do this here just in case application defines PEND out.put_symbol("end"); out.put_symbol("PEND"); } void ps_printer::do_invis(char *, const environment *) { invis_count++; } void ps_printer::do_endinvis(char *, const environment *) { if (invis_count == 0) error("unbalanced `endinvis' command"); else --invis_count; } printer *make_printer() { return new ps_printer; } static void usage(); int main(int argc, char **argv) { program_name = argv[0]; static char stderr_buf[BUFSIZ]; setbuf(stderr, stderr_buf); int c; while ((c = getopt(argc, argv, "F:glc:w:vb:")) != EOF) switch(c) { case 'v': { extern const char *version_string; fprintf(stderr, "grops version %s\n", version_string); fflush(stderr); break; } case 'c': if (sscanf(optarg, "%d", &ncopies) != 1 || ncopies <= 0) { error("bad number of copies `%s'", optarg); ncopies = 1; } break; case 'g': guess_flag = 1; break; case 'l': landscape_flag = 1; break; case 'F': font::command_line_font_dir(optarg); break; case 'w': if (sscanf(optarg, "%d", &linewidth) != 1 || linewidth < 0) { error("bad linewidth `%s'", optarg); linewidth = -1; } break; case 'b': // XXX check this broken_flags = atoi(optarg); bflag = 1; break; case '?': usage(); break; default: assert(0); } font::set_unknown_desc_command_handler(handle_unknown_desc_command); if (optind >= argc) do_file("-"); else { for (int i = optind; i < argc; i++) do_file(argv[i]); } delete pr; return 0; } static void usage() { fprintf(stderr, "usage: %s [-glv] [-b n] [-c n] [-w n] [-F dir] [files ...]\n", program_name); exit(1); }