1*cf5a6c84SAndroid Build Coastguard Worker /* fold.c - Line wrap input.
2*cf5a6c84SAndroid Build Coastguard Worker *
3*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2023 Rob Landley <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker *
5*cf5a6c84SAndroid Build Coastguard Worker * See http://pubs.opengroup.org/onlinepubs/9699919799/utilities/fold.html
6*cf5a6c84SAndroid Build Coastguard Worker
7*cf5a6c84SAndroid Build Coastguard Worker USE_FOLD(NEWTOY(fold, "bsw#<1=80", TOYFLAG_USR|TOYFLAG_BIN))
8*cf5a6c84SAndroid Build Coastguard Worker
9*cf5a6c84SAndroid Build Coastguard Worker config FOLD
10*cf5a6c84SAndroid Build Coastguard Worker bool "fold"
11*cf5a6c84SAndroid Build Coastguard Worker default y
12*cf5a6c84SAndroid Build Coastguard Worker help
13*cf5a6c84SAndroid Build Coastguard Worker usage: fold [-bs] [-w WIDTH] [FILE...]
14*cf5a6c84SAndroid Build Coastguard Worker
15*cf5a6c84SAndroid Build Coastguard Worker Break long lines by inserting newlines.
16*cf5a6c84SAndroid Build Coastguard Worker
17*cf5a6c84SAndroid Build Coastguard Worker -b Count bytes instead of utf-8 unicode columns
18*cf5a6c84SAndroid Build Coastguard Worker -s Wrap at whitespace when possible
19*cf5a6c84SAndroid Build Coastguard Worker -w Break at WIDTH columns (default 80)
20*cf5a6c84SAndroid Build Coastguard Worker */
21*cf5a6c84SAndroid Build Coastguard Worker
22*cf5a6c84SAndroid Build Coastguard Worker #define FOR_fold
23*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
24*cf5a6c84SAndroid Build Coastguard Worker
GLOBALS(long w;)25*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
26*cf5a6c84SAndroid Build Coastguard Worker long w;
27*cf5a6c84SAndroid Build Coastguard Worker )
28*cf5a6c84SAndroid Build Coastguard Worker
29*cf5a6c84SAndroid Build Coastguard Worker // wcwidth utf8towc
30*cf5a6c84SAndroid Build Coastguard Worker void do_fold(int fd, char *name)
31*cf5a6c84SAndroid Build Coastguard Worker {
32*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = fd ? fdopen(fd, "r") : stdin;
33*cf5a6c84SAndroid Build Coastguard Worker char *rr, *ss;
34*cf5a6c84SAndroid Build Coastguard Worker long ii, bb, ww, width, space;
35*cf5a6c84SAndroid Build Coastguard Worker unsigned cc;
36*cf5a6c84SAndroid Build Coastguard Worker
37*cf5a6c84SAndroid Build Coastguard Worker // Note: not bothering to handle embedded NUL bytes, they truncate the line.
38*cf5a6c84SAndroid Build Coastguard Worker
39*cf5a6c84SAndroid Build Coastguard Worker // Loop reading/printing lines
40*cf5a6c84SAndroid Build Coastguard Worker while ((ss = rr = xgetdelim(fp, '\n'))) for (ii = width = space = 0;;) {
41*cf5a6c84SAndroid Build Coastguard Worker // Parse next character's byte length and column width
42*cf5a6c84SAndroid Build Coastguard Worker bb = ww = 1;
43*cf5a6c84SAndroid Build Coastguard Worker if (ss[ii]<32) ww = FLAG(b);
44*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(b)) cc = ss[ii];
45*cf5a6c84SAndroid Build Coastguard Worker else {
46*cf5a6c84SAndroid Build Coastguard Worker if ((bb = utf8towc(&cc, ss+ii, 4))>0 && (ww = wcwidth(cc))<0) ww = 0;
47*cf5a6c84SAndroid Build Coastguard Worker if (cc=='\t') ww = 8-(width&7);
48*cf5a6c84SAndroid Build Coastguard Worker }
49*cf5a6c84SAndroid Build Coastguard Worker
50*cf5a6c84SAndroid Build Coastguard Worker // Did line end?
51*cf5a6c84SAndroid Build Coastguard Worker if (!cc || cc=='\r' || cc=='\n') {
52*cf5a6c84SAndroid Build Coastguard Worker if (cc) ii++;
53*cf5a6c84SAndroid Build Coastguard Worker if (ii) {
54*cf5a6c84SAndroid Build Coastguard Worker xwrite(1, ss, ii);
55*cf5a6c84SAndroid Build Coastguard Worker ss += ii;
56*cf5a6c84SAndroid Build Coastguard Worker ii = width = space = 0;
57*cf5a6c84SAndroid Build Coastguard Worker } else {
58*cf5a6c84SAndroid Build Coastguard Worker free(rr);
59*cf5a6c84SAndroid Build Coastguard Worker
60*cf5a6c84SAndroid Build Coastguard Worker break;
61*cf5a6c84SAndroid Build Coastguard Worker }
62*cf5a6c84SAndroid Build Coastguard Worker
63*cf5a6c84SAndroid Build Coastguard Worker // backspace?
64*cf5a6c84SAndroid Build Coastguard Worker } else if (!FLAG(b) && cc=='\b') {
65*cf5a6c84SAndroid Build Coastguard Worker if (width) width--;
66*cf5a6c84SAndroid Build Coastguard Worker ii++;
67*cf5a6c84SAndroid Build Coastguard Worker
68*cf5a6c84SAndroid Build Coastguard Worker // Is it time to wrap?
69*cf5a6c84SAndroid Build Coastguard Worker
70*cf5a6c84SAndroid Build Coastguard Worker } else if (width+ww>TT.w && ss[ii+bb]!='\b'
71*cf5a6c84SAndroid Build Coastguard Worker && (ii || !strchr("\r\n", ss[ii+bb])))
72*cf5a6c84SAndroid Build Coastguard Worker {
73*cf5a6c84SAndroid Build Coastguard Worker if (!ii) ii += bb;
74*cf5a6c84SAndroid Build Coastguard Worker if (!space) space = ii;
75*cf5a6c84SAndroid Build Coastguard Worker
76*cf5a6c84SAndroid Build Coastguard Worker cc = ss[space];
77*cf5a6c84SAndroid Build Coastguard Worker ss[space] = '\n';
78*cf5a6c84SAndroid Build Coastguard Worker xwrite(1, ss, space+1);
79*cf5a6c84SAndroid Build Coastguard Worker ss += space;
80*cf5a6c84SAndroid Build Coastguard Worker *ss = cc;
81*cf5a6c84SAndroid Build Coastguard Worker ii = width = space = 0;
82*cf5a6c84SAndroid Build Coastguard Worker
83*cf5a6c84SAndroid Build Coastguard Worker // move the cursor
84*cf5a6c84SAndroid Build Coastguard Worker } else {
85*cf5a6c84SAndroid Build Coastguard Worker ii += bb;
86*cf5a6c84SAndroid Build Coastguard Worker width += ww;
87*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(s) && iswspace(cc)) space = ii;
88*cf5a6c84SAndroid Build Coastguard Worker }
89*cf5a6c84SAndroid Build Coastguard Worker }
90*cf5a6c84SAndroid Build Coastguard Worker if (fp != stdin) fclose(fp);
91*cf5a6c84SAndroid Build Coastguard Worker }
92*cf5a6c84SAndroid Build Coastguard Worker
fold_main(void)93*cf5a6c84SAndroid Build Coastguard Worker void fold_main(void)
94*cf5a6c84SAndroid Build Coastguard Worker {
95*cf5a6c84SAndroid Build Coastguard Worker loopfiles(toys.optargs, do_fold);
96*cf5a6c84SAndroid Build Coastguard Worker loopfiles_rw(toys.optargs, O_RDONLY|WARN_ONLY, 0, do_fold);
97*cf5a6c84SAndroid Build Coastguard Worker }
98