/* TODO mark translatable strings add usage function call parse_long_options dcl, set program_name do fclose/error checking */ /* asa.c - interpret ASA carriage control characters Copyright (C) 94, 1996 Thomas Koenig 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. */ /* System Headers */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <errno.h> /* Macros */ #define LINESIZE 135 #define NUMLINES 5 #define MAX(a,b) ((a) > (b) ? (a) : (b)) /* Structures and unions */ struct Str { char *chr; size_t len; }; typedef struct Str str; /* File scope variables */ static str *line_buffer = (str *) 0; static size_t line_num = 0; static size_t linebuf_size; /* Function declarations */ char *xmalloc (); char *xrealloc (); static size_t readline (FILE *fp, char **a); static void add_line (str *); static void flush (void); static void copy_file (FILE *fp); /* Local functions */ static void form_feed () { putchar ('\f'); } static void new_line () { putchar ('\n'); } static void add_line (str *line) { if (line_num >= linebuf_size) { linebuf_size += NUMLINES; line_buffer = (str *) xrealloc (line_buffer, linebuf_size * sizeof (str *)); } line_buffer[line_num] = *line; line_num++; } static void flush () { size_t i, j; size_t max_len; if (line_num == 0) return; if (line_num == 1) { printf ("%s\n", line_buffer[0].chr + 1); line_num = 0; return; } max_len = 0; for (j = 0; j < line_num; j++) max_len = MAX (max_len, line_buffer[j].len); for (i = 1; i <= max_len; i++) { int printed = 0; for (j = 0; j < line_num; j++) { if (line_buffer[j].len > i) { int ch = line_buffer[j].chr[i]; if (ch != ' ') { if (printed) putchar ('\b'); putchar (ch); printed = 1; } } } if (!printed) putchar (' '); } for (j = 0; j < line_num; j++) free (line_buffer[j].chr); line_num = 0; putchar ('\n'); } static size_t readline (FILE *fp, char **ret) { static char *buffer = (char *) 0; char *ret_buff; static int bufsize = LINESIZE; int ch; size_t len = 0; int inc; int i; if (buffer == (char *) 0) buffer = (char *) xmalloc (LINESIZE); while (1) { ch = fgetc (fp); if (ch == EOF) break; if (ch == '\t') { ch = ' '; inc = 8 - (len % 8); } else inc = 1; if (len + inc > bufsize - 2) { bufsize += LINESIZE; buffer = xrealloc (buffer, bufsize); } for (i = 0; i < inc; i++) buffer[len + i] = ch; len += inc; if (ch == '\n') break; } buffer[len] = '\0'; ret_buff = xmalloc (len + 1); strcpy (ret_buff, buffer); *ret = ret_buff; return len; } static void copy_file (FILE *fp) { str line; static first_line = 1; while ((line.len = readline (fp, &(line.chr)))) { if (line.chr[line.len - 1] == '\n') { line.chr[line.len - 1] = '\0'; line.len--; } switch (line.chr[0]) { case '+': add_line (&line); break; case '0': flush (); new_line (); add_line (&line); break; case '1': flush (); if (!first_line) form_feed (); add_line (&line); break; case ' ': flush (); add_line (&line); break; default: flush (); add_line (&line); break; } first_line = 0; } } /* Global functions */ int main (int argc, char **argv) { int err; line_buffer = (str *) xmalloc (NUMLINES * sizeof (str *)); linebuf_size = NUMLINES; err = 0; if (argc == 1) copy_file (stdin); else { FILE *fp; char *fname; while (--argc > 0) { fname = *++argv; if ((fp = fopen (fname, "r")) == NULL) { err = 1; error (0, errno, "%s", fname); } else { copy_file (fp); fclose (fp); } } } flush (); exit (err ? EXIT_FAILURE : EXIT_SUCCESS); }