summaryrefslogtreecommitdiff
path: root/src/tak.c
blob: 3aa4d8416eca68cc075cf35a0ea8c19f53b14946 (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
113
114
115
116
117
118
119
120
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#include "util/map.h"
#include "util/versie.h"

static void usage(FILE *f) {
  fprintf(f,
      "Gebruik: tak [-hV] <bestand...>\n"
      "\n"
      "Schakel bestanden aaneen naar standaard uitvoer omgekeerd.\n"
      "\n"
      "  -h    Toon deze hulptekst\n"
      "  -V    Toon versienummer\n");
}

// Returns pointer to argument array containing the file names
static char** parse_options(int argc, char **argv) {
  int opt;
  while ((opt = getopt(argc, argv, "hV")) != -1) {
    switch (opt) {
      case 'h':
        usage(stdout);
        exit(0);

      case 'V':
        drukkedoos_print_versie(stdout, "tak");
        exit(0);

      case '?':
        usage(stderr);
        exit(1);
    }
  }

  return argv + optind;
}

struct state {
  char *buf;
  size_t sz;
};

static void process(struct state state) {
  char *lstart, *lend;
  lend = &state.buf[state.sz - 1];

  while (lend > state.buf) {
    if (*lend == '\n') {
      lend--;
    }

    lstart = lend;
    while (*lstart != '\n' && lstart != state.buf) {
      lstart--;
    }
    if (lstart != state.buf) {
      lstart++;
    }

    fwrite(lstart, 1, lend - lstart + 2, stdout);

    lend = lstart-1;
  }
}

static char *read_stdin(size_t *sz) {
  const size_t CHUNK_SIZE = 4096;

  size_t cap = CHUNK_SIZE;
  char *res = malloc(cap);

  while (!feof(stdin)) {
    if (cap-*sz < CHUNK_SIZE) {
      cap *= 2;
      res = realloc(res, cap);
    }

    const size_t n = fread(res+*sz, 1, CHUNK_SIZE, stdin);
    *sz += n;

    if (n != CHUNK_SIZE && !feof(stdin)) {
      fprintf(stderr, "tak: fout tijdens lezen van standaard invoer.\n");
      exit(1);
    }
  }

  return res;
}

int entry_tak(int argc, char **argv) {
  char **args = parse_options(argc, argv);
  if (*args == NULL) {
    size_t sz = 0;
    char *data = read_stdin(&sz);

    struct state state = { .buf = data, .sz = sz };
    process(state);

    free(data);
  } else {
    while (*args != NULL) {
      struct map *map = open_map(*args);
      if (map == NULL) {
        return 1;
      }

      struct state state = { .buf = map->addr, .sz = map->sb.st_size };
      process(state);

      close_map(map);
      args++;
    }
  }

  return 0;
}