xref: /aosp_15_r20/external/toybox/toys/posix/fold.c (revision cf5a6c84e2b8763fc1a7db14496fd4742913b199)
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