summaryrefslogtreecommitdiff
path: root/src/columnt.c
blob: fc76c1aa06f67dad32ca5e9ece092745d18e5235 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
#include <unistd.h>

#include "alloc.h"
#include "buffer.h"
#include "exit.h"
#include "logmsg.h"
#include "readclose.h"
#include "stralloc.h"

#define WHO "columnt"

#define BSIZE 4096

char outbuf[BSIZE];
buffer bo = BUFFER_INIT(write, 1, outbuf, sizeof(outbuf));

static void nomem()
{
  logmsg(WHO, 111, FATAL, "out of memory");
}

static void die_read()
{
  logmsg(WHO, 110, ERROR, "unable to read input: ");
}

static void die_write()
{
  logmsg(WHO, 110, ERROR, "unable to write output: ");
}

stralloc file = {0};
int *width;
int maxfield = 0;

static void nothing() {}

static void printline()
{
  if (buffer_put(&bo, "\n", 1) == -1) die_write();
}

static void maxfield_check(int fieldnum, char *buf, int len)
{
  if (fieldnum > maxfield) maxfield = fieldnum;
}

static void width_check(int fieldnum, char *buf, int len)
{
  if (len > width[fieldnum]) width[fieldnum] = len;
}

static void width_init()
{
  int i;

  width = (int *)alloc((maxfield + 1) * sizeof(int));
  if (!width) nomem();
  for (i = 0; i <= maxfield; ++i) width[i] = 0;
}

static void printfield(int fieldnum, char *buf, int len)
{
  int i;

  if (fieldnum < maxfield)
    for (i = len; i < width[fieldnum]; ++i)
      if (buffer_put(&bo, " ", 1) == -1) die_write();

  if (buffer_put(&bo, buf, len) == -1) die_write();

  if (fieldnum < maxfield)
    if (buffer_put(&bo, "  ", 2) == -1) die_write();
}

static void split(void (*dofield)(int, char *, int), void (*doline)(void))
{
  int i;
  int j;
  int fieldpos;
  int fieldnum;

  for (j = i = 0; j < file.len; ++j)
    if (file.s[j] == '\n') {
      fieldnum = 0;
      for (;;) {
        while ((file.s[i] == ' ') || (file.s[i] == '\t')) ++i;
        if (i == j) break;
        fieldpos = i;
        while ((file.s[i] != ' ') && (file.s[i] != '\t') && (file.s[i] != '\n')) ++i;
        dofield(fieldnum++, file.s + fieldpos, i - fieldpos);
      }
      doline();
      i = j + 1;
    }
}

int main()
{
  if (readclose_append(0, &file, BSIZE) == -1) die_read();
  if (!file.len) _exit(0);
  if (file.s[file.len - 1] != '\n')
    if (!stralloc_append(&file, "\n")) nomem();

  split(maxfield_check, nothing);
  width_init();
  split(width_check, nothing);
  split(printfield, printline);

  if (buffer_flush(&bo) == -1) die_write();
  _exit(0);
}