/* * $XFree86: xc/programs/xterm/print.c,v 1.3.2.5 1999/07/28 13:38:04 hohndel Exp $ */ /************************************************************ Copyright 1997,1998 by Thomas E. Dickey All Rights Reserved Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE ABOVE LISTED COPYRIGHT HOLDER(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. Except as contained in this notice, the name(s) of the above copyright holders shall not be used in advertising or otherwise to promote the sale, use or other dealings in this Software without prior written authorization. ********************************************************/ #include #include #include #include #define SHIFT_IN '\017' #define SHIFT_OUT '\016' #define CSET_IN 'A' #define CSET_OUT '0' #define isForm(c) ((c) == '\r' || (c) == '\n' || (c) == '\f') #define Strlen(a) strlen((char *)a) #define Strcmp(a,b) strcmp((char *)a,(char *)b) #define Strncmp(a,b,c) strncmp((char *)a,(char *)b,c) #define SGR_MASK (BOLD|BLINK|UNDERLINE|INVERSE) static void charToPrinter (int chr); static void printLine (int row, int chr); static void send_CharSet (int row); static void send_SGR (unsigned attr, int fg, int bg); static void stringToPrinter (char * str); static FILE *Printer; static int Printer_pid; static int initialized; static void closePrinter(void) { if (Printer != 0) { fclose(Printer); TRACE(("closed printer, waiting...\n")); while (nonblocking_wait() > 0) ; Printer = 0; initialized = 0; TRACE(("closed printer\n")); } } static void printCursorLine(void) { register TScreen *screen = &term->screen; TRACE(("printCursorLine\n")) printLine(screen->cur_row, '\n'); } /* * DEC's manual doesn't document whether trailing blanks are removed, or what * happens with a line that is entirely blank. This function prints the * characters that xterm would allow as a selection (which may include blanks). */ static void printLine(int row, int chr) { register TScreen *screen = &term->screen; Char *c = SCRN_BUF_CHARS(screen, row); Char *a = SCRN_BUF_ATTRS(screen, row); Char attr = *a & SGR_MASK; int last = screen->max_col; int col; #if OPT_ISO_COLORS && OPT_PRINT_COLORS register Char *fb = 0; #endif int fg = -1, last_fg = -1; int bg = -1, last_bg = -1; int cs = CSET_IN,last_cs = CSET_IN; TRACE(("printLine(row=%d, chr=%d)\n", row, chr)) if_OPT_ISO_COLORS(screen,{ fb = SCRN_BUF_COLOR(screen, row); }) while (last > 0) { if ((a[last-1] & CHARDRAWN) == 0) last--; else break; } if (last) { if (screen->print_attributes) { send_CharSet(row); send_SGR(0,-1,-1); } for (col = 0; col < last; col++) { Char ch = c[col]; #if OPT_PRINT_COLORS if_OPT_ISO_COLORS(screen,{ if (screen->print_attributes > 1) { fg = (a[col] & FG_COLOR) ? extract_fg(fb[col], a[col]) : -1; bg = (a[col] & BG_COLOR) ? extract_bg(fb[col]) : -1; } }) #endif if ((((a[col] & SGR_MASK) != attr) #if OPT_PRINT_COLORS || (last_fg != fg) || (last_bg != bg) #endif ) && ch) { attr = (a[col] & SGR_MASK); last_fg = fg; last_bg = bg; if (screen->print_attributes) send_SGR(attr, fg, bg); } if (ch == 0) ch = ' '; cs = (ch >= ' ' && ch != 0x7f) ? CSET_IN : CSET_OUT; if (last_cs != cs) { if (screen->print_attributes) { charToPrinter((cs == CSET_OUT) ? SHIFT_OUT : SHIFT_IN); } last_cs = cs; } /* FIXME: we shouldn't have to map back from the * alternate character set, except that the * corresponding charset information is not encoded * into the CSETS array. */ charToPrinter((cs == CSET_OUT) ? (ch == 0x7f ? 0x5f : (ch + 0x5f)) : ch); } if (screen->print_attributes) { send_SGR(0,-1,-1); if (cs != CSET_IN) charToPrinter(SHIFT_IN); } } if (screen->print_attributes) charToPrinter('\r'); charToPrinter(chr); } void xtermPrintScreen(void) { register TScreen *screen = &term->screen; int top = screen->printer_extent ? 0 : screen->top_marg; int bot = screen->printer_extent ? screen->max_row : screen->bot_marg; int was_open = initialized; TRACE(("xtermPrintScreen, rows %d..%d\n", top, bot)) while (top <= bot) printLine(top++, '\n'); if (screen->printer_formfeed) charToPrinter('\f'); if (!was_open || screen->printer_autoclose) { closePrinter(); } } static void send_CharSet(int row) { #if OPT_DEC_CHRSET register TScreen *screen = &term->screen; char *msg = 0; switch (SCRN_BUF_CSETS(screen, row)[0]) { case CSET_SWL: msg = "\033#5"; break; case CSET_DHL_TOP: msg = "\033#3"; break; case CSET_DHL_BOT: msg = "\033#4"; break; case CSET_DWL: msg = "\033#6"; break; } if (msg != 0) stringToPrinter(msg); #endif /* OPT_DEC_CHRSET */ } static void send_SGR(unsigned attr, int fg, int bg) { char msg[80]; strcpy(msg, "\033[0"); if (attr & BOLD) strcat(msg, ";1"); if (attr & UNDERLINE) strcat(msg, ";2"); if (attr & BLINK) strcat(msg, ";5"); if (attr & INVERSE) /* typo? DEC documents this as invisible */ strcat(msg, ";7"); #if OPT_PRINT_COLORS if (bg >= 0) { sprintf(msg + strlen(msg), ";%d", (bg < 8) ? (40 + bg) : (92 + bg)); } if (fg >= 0) { #if OPT_PC_COLORS if (term->screen.boldColors && fg > 8 && attr & BOLD) fg -= 8; #endif sprintf(msg + strlen(msg), ";%d", (fg < 8) ? (30 + fg) : (82 + fg)); } #endif strcat(msg, "m"); stringToPrinter(msg); } /* * This implementation only knows how to write to a pipe. */ static void charToPrinter(int chr) { if (!initialized) { FILE *input; int my_pipe[2]; int c; register TScreen *screen = &term->screen; if (pipe(my_pipe)) SysError (ERROR_FORK); if ((Printer_pid = fork()) < 0) SysError (ERROR_FORK); if (Printer_pid == 0) { TRACE(((char *)0)) close(my_pipe[1]); /* printer is silent */ close (screen->respond); close(fileno(stdout)); dup2(fileno(stderr), 1); if (fileno(stderr) != 2) { dup2(fileno(stderr), 2); close(fileno(stderr)); } setgid (screen->gid); /* don't want privileges! */ setuid (screen->uid); Printer = popen(screen->printer_command, "w"); input = fdopen(my_pipe[0], "r"); while ((c = fgetc(input)) != EOF) { fputc(c, Printer); if (isForm(c)) fflush(Printer); } pclose(Printer); exit(0); } else { close(my_pipe[0]); /* won't read from printer */ Printer = fdopen(my_pipe[1], "w"); TRACE(("opened printer from pid %d/%d\n", (int)getpid(), Printer_pid)) } initialized++; } if (Printer != 0) { fputc(chr, Printer); if (isForm(chr)) fflush(Printer); } } static void stringToPrinter(char *str) { while (*str) charToPrinter(*str++); } /* * This module implements the MC (Media Copy) and related printing control * sequences for VTxxx emulation. This is based on the description in the * VT330/VT340 Programmer Reference Manual EK-VT3XX-TP-001 (Digital Equipment * Corp., March 1987). */ void xtermMediaControl (int param, int private_seq) { register TScreen *screen = &term->screen; TRACE(("MediaCopy param=%d, private=%d\n", param, private_seq)) if (private_seq) { switch (param) { case 1: printCursorLine(); break; case 4: screen->printer_controlmode = 0; TRACE(("Reset autoprint mode\n")) break; case 5: screen->printer_controlmode = 1; TRACE(("Set autoprint mode\n")) break; } } else { switch (param) { case -1: case 0: xtermPrintScreen(); break; case 4: screen->printer_controlmode = 0; TRACE(("Reset printer controller mode\n")) break; case 5: screen->printer_controlmode = 2; TRACE(("Set printer controller mode\n")) break; } } } /* * When in autoprint mode, the printer prints a line from the screen when you * move the cursor off that line with an LF, FF, or VT character, or an * autowrap occurs. The printed line ends with a CR and the character (LF, FF * or VT) that moved the cursor off the previous line. */ void xtermAutoPrint(int chr) { register TScreen *screen = &term->screen; if (screen->printer_controlmode == 1) { TRACE(("AutoPrint %d\n", chr)) printLine(screen->cursor_row, chr); if (Printer != 0) fflush(Printer); } } /* * When in printer controller mode, the terminal send received characters to * the printer without displaying them on the screen. The terminal sends all * characters and control sequences to the printer, except NUL, XON, XOFF, and * the printer controller sequences. * * This function eats characters, returning 0 as long as it must buffer or * divert to the printer. We're only invoked here when in printer controller * mode, and handle the exit from that mode. */ #define LB '[' int xtermPrinterControl(int chr) { register TScreen *screen = &term->screen; static struct { Char seq[5]; int active; } tbl[] = { { { CSI, '5', 'i' }, 2 }, { { CSI, '4', 'i' }, 0 }, { { ESC, LB, '5', 'i' }, 2 }, { { ESC, LB, '4', 'i' }, 0 }, }; static Char bfr[10]; static size_t length; size_t n; TRACE(("In printer:%d\n", chr)) switch (chr) { case 0: case 'Q' & 0x1f: case 'S' & 0x1f: return 0; /* ignored by application */ case CSI: case ESC: case '[': case '4': case '5': case 'i': bfr[length++] = chr; for (n = 0; n < sizeof(tbl)/sizeof(tbl[0]); n++) { size_t len = Strlen(tbl[n].seq); if (length == len && Strcmp(bfr, tbl[n].seq) == 0) { screen->printer_controlmode = tbl[n].active; TRACE(("Set printer controller mode %sactive\n", tbl[n].active ? "" : "in")) if (screen->printer_autoclose && screen->printer_controlmode == 0) closePrinter(); length = 0; return 0; } else if (len > length && Strncmp(bfr, tbl[n].seq, length) == 0) { return 0; } } length--; /* FALLTHRU */ default: for (n = 0; n < length; n++) charToPrinter(bfr[n]); bfr[0] = chr; length = 1; return 0; } }