/* paste.c - Replace newlines
*
* Copyright 2012 Felix Janda <felix.janda@posteo.de>
*
* http://pubs.opengroup.org/onlinepubs/9699919799/utilities/paste.html
*
USE_PASTE(NEWTOY(paste, "d:s", TOYFLAG_BIN))
config PASTE
bool "paste"
default y
help
usage: paste [-s] [-d list] [file...]
Replace newlines in files.
-d list list of delimiters to separate lines
-s process files sequentially instead of in parallel
By default print corresponding lines separated by <tab>.
*/
#define FOR_paste
#include "toys.h"
GLOBALS(
char *delim;
)
void paste_main(void)
{
char *p, *buf = toybuf, **args = toys.optargs;
size_t ndelim = 0;
int i, j, c;
// Process delimiter list
// TODO: Handle multibyte characters
if (!(toys.optflags & FLAG_d)) TT.delim = "\t";
for (p = TT.delim; *p; p++, buf++, ndelim++) {
if (*p == '\\') {
p++;
if (-1 == (i = stridx("nt\\0", *p)))
error_exit("bad delimiter: \\%c", *p);
*buf = "\n\t\\\0"[i];
} else *buf = *p;
}
*buf = 0;
if (toys.optflags & FLAG_s) { // Sequential
FILE *f;
for (; *args; args++) {
if ((*args)[0] == '-' && !(*args)[1]) f = stdin;
else if (!(f = fopen(*args, "r"))) perror_exit_raw(*args);
for (i = 0, c = 0; c != EOF;) {
switch(c = getc(f)) {
case '\n':
putchar(toybuf[i++ % ndelim]);
case EOF:
break;
default:
putchar(c);
}
}
if (f != stdin) fclose(f);
putchar('\n');
}
} else { // Parallel
// Need to be careful not to print an extra line at the end
FILE **files;
int anyopen = 1;
files = (FILE**)(buf + 1);
for (; *args; args++, files++) {
if ((*args)[0] == '-' && !(*args)[1]) *files = stdin;
else if (!(*files = fopen(*args, "r"))) perror_exit_raw(*args);
}
while (anyopen) {
anyopen = 0;
for (i = 0; i < toys.optc; i++) {
FILE **f = (FILE**)(buf + 1) + i;
if (*f) for (;;) {
c = getc(*f);
if (c != EOF) {
if (!anyopen++) for (j = 0; j < i; j++) putchar(toybuf[j % ndelim]);
if (c != '\n') putchar(c);
else break;
}
else {
if (*f != stdin) fclose(*f);
*f = 0;
break;
}
}
if (anyopen) putchar((i + 1 == toys.optc) ? toybuf[i % ndelim] : '\n');
}
}
}
}