/* expand.c - expands tabs to space
*
* Copyright 2012 Jonathan Clairembault <jonathan at clairembault dot fr>
*
* See http://pubs.opengroup.org/onlinepubs/9699919799/utilities/expand.html
USE_EXPAND(NEWTOY(expand, "t*", TOYFLAG_USR|TOYFLAG_BIN|TOYFLAG_LOCALE))
config EXPAND
bool "expand"
default y
help
usage: expand [-t TABLIST] [FILE...]
Expand tabs to spaces according to tabstops.
-t TABLIST
Specify tab stops, either a single number instead of the default 8,
or a comma separated list of increasing numbers representing tabstop
positions (absolute, not increments) with each additional tab beyound
that becoming one space.
*/
#define FOR_expand
#include "toys.h"
GLOBALS(
struct arg_list *tabs;
unsigned tabcount, *tab;
)
static void do_expand(int fd, char *name)
{
int i, len, x=0, stop = 0;
for (;;) {
len = readall(fd, toybuf, sizeof(toybuf));
if (len<0) {
perror_msg_raw(name);
return;
}
if (!len) break;
for (i=0; i<len; i++) {
int width = 1;
char c;
if (CFG_TOYBOX_I18N) {
wchar_t blah;
width = mbrtowc(&blah, toybuf+i, len-i, 0);
if (width > 1) {
if (width != fwrite(toybuf+i, width, 1, stdout))
perror_exit("stdout");
i += width-1;
x++;
continue;
} else if (width == -2) break;
else if (width == -1) continue;
}
c = toybuf[i];
if (c != '\t') {
if (EOF == putc(c, stdout)) perror_exit(0);
if (c == '\b' && x) width = -1;
if (c == '\n') {
x = stop = 0;
continue;
}
} else {
if (TT.tabcount < 2) {
width = TT.tabcount ? *TT.tab : 8;
width -= x%width;
} else while (stop < TT.tabcount) {
if (TT.tab[stop] > x) {
width = TT.tab[stop] - x;
break;
} else stop++;
}
xprintf("%*c", width, ' ');
}
x += width;
}
}
}
// Parse -t options to fill out unsigned array in tablist (if not NULL)
// return number of entries in tablist
static int parse_tablist(unsigned *tablist)
{
struct arg_list *tabs;
int tabcount = 0;
for (tabs = TT.tabs; tabs; tabs = tabs->next) {
char *s = tabs->arg;
while (*s) {
int count;
unsigned x, *t = tablist ? tablist+tabcount : &x;
if (tabcount >= sizeof(toybuf)/sizeof(unsigned)) break;
if (sscanf(s, "%u%n", t, &count) != 1) break;
if (tabcount++ && tablist && *(t-1) >= *t) break;
s += count;
if (*s==' ' || *s==',') s++;
else break;
}
if (*s) error_exit("bad tablist");
}
return tabcount;
}
void expand_main(void)
{
TT.tabcount = parse_tablist(NULL);
// Determine size of tablist, allocate memory, fill out tablist
if (TT.tabcount) {
TT.tab = xmalloc(sizeof(unsigned)*TT.tabcount);
parse_tablist(TT.tab);
}
loopfiles(toys.optargs, do_expand);
if (CFG_TOYBOX_FREE) free(TT.tab);
}