1*cf5a6c84SAndroid Build Coastguard Worker /* shuf.c - Output lines in random order.
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 https://man7.org/linux/man-pages/man1/shuf.1.html
6*cf5a6c84SAndroid Build Coastguard Worker
7*cf5a6c84SAndroid Build Coastguard Worker USE_SHUF(NEWTOY(shuf, "zen#<0", TOYFLAG_USR|TOYFLAG_BIN))
8*cf5a6c84SAndroid Build Coastguard Worker
9*cf5a6c84SAndroid Build Coastguard Worker config SHUF
10*cf5a6c84SAndroid Build Coastguard Worker bool "shuf"
11*cf5a6c84SAndroid Build Coastguard Worker default y
12*cf5a6c84SAndroid Build Coastguard Worker help
13*cf5a6c84SAndroid Build Coastguard Worker usage: shuf [-ze] [-n COUNT] [FILE...]
14*cf5a6c84SAndroid Build Coastguard Worker
15*cf5a6c84SAndroid Build Coastguard Worker Write lines of input to output in random order.
16*cf5a6c84SAndroid Build Coastguard Worker
17*cf5a6c84SAndroid Build Coastguard Worker -z Input/output lines are NUL terminated.
18*cf5a6c84SAndroid Build Coastguard Worker -n Stop after COUNT many output lines.
19*cf5a6c84SAndroid Build Coastguard Worker -e Echo mode: arguments are inputs to shuffle, not files to read.
20*cf5a6c84SAndroid Build Coastguard Worker */
21*cf5a6c84SAndroid Build Coastguard Worker
22*cf5a6c84SAndroid Build Coastguard Worker #define FOR_shuf
23*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
24*cf5a6c84SAndroid Build Coastguard Worker
GLOBALS(long n;char ** lines;long count;)25*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
26*cf5a6c84SAndroid Build Coastguard Worker long n;
27*cf5a6c84SAndroid Build Coastguard Worker
28*cf5a6c84SAndroid Build Coastguard Worker char **lines;
29*cf5a6c84SAndroid Build Coastguard Worker long count;
30*cf5a6c84SAndroid Build Coastguard Worker )
31*cf5a6c84SAndroid Build Coastguard Worker
32*cf5a6c84SAndroid Build Coastguard Worker static void do_shuf_line(char **pline, long len)
33*cf5a6c84SAndroid Build Coastguard Worker {
34*cf5a6c84SAndroid Build Coastguard Worker if (!pline) return;
35*cf5a6c84SAndroid Build Coastguard Worker if (!(TT.count&255))
36*cf5a6c84SAndroid Build Coastguard Worker TT.lines = xrealloc(TT.lines, sizeof(void *)*(TT.count+256));
37*cf5a6c84SAndroid Build Coastguard Worker TT.lines[TT.count++] = *pline; // TODO: repack?
38*cf5a6c84SAndroid Build Coastguard Worker *pline = 0;
39*cf5a6c84SAndroid Build Coastguard Worker }
40*cf5a6c84SAndroid Build Coastguard Worker
do_shuf(int fd,char * name)41*cf5a6c84SAndroid Build Coastguard Worker static void do_shuf(int fd, char *name)
42*cf5a6c84SAndroid Build Coastguard Worker {
43*cf5a6c84SAndroid Build Coastguard Worker do_lines(fd, '\n'*!FLAG(z), do_shuf_line);
44*cf5a6c84SAndroid Build Coastguard Worker }
45*cf5a6c84SAndroid Build Coastguard Worker
shuf_main(void)46*cf5a6c84SAndroid Build Coastguard Worker void shuf_main(void)
47*cf5a6c84SAndroid Build Coastguard Worker {
48*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(e)) {
49*cf5a6c84SAndroid Build Coastguard Worker TT.lines = toys.optargs;
50*cf5a6c84SAndroid Build Coastguard Worker TT.count = toys.optc;
51*cf5a6c84SAndroid Build Coastguard Worker } else loopfiles(toys.optargs, do_shuf);
52*cf5a6c84SAndroid Build Coastguard Worker
53*cf5a6c84SAndroid Build Coastguard Worker if (!FLAG(n) || TT.n>TT.count) TT.n = TT.count;
54*cf5a6c84SAndroid Build Coastguard Worker
55*cf5a6c84SAndroid Build Coastguard Worker srandom(millitime());
56*cf5a6c84SAndroid Build Coastguard Worker while (TT.n--) {
57*cf5a6c84SAndroid Build Coastguard Worker long ll = random()%TT.count;
58*cf5a6c84SAndroid Build Coastguard Worker writeall(1, TT.lines[ll], strlen(TT.lines[ll])+FLAG(z));
59*cf5a6c84SAndroid Build Coastguard Worker if (!FLAG(e)) free(TT.lines[ll]);
60*cf5a6c84SAndroid Build Coastguard Worker else if (!FLAG(z)) writeall(1, "\n", 1);
61*cf5a6c84SAndroid Build Coastguard Worker TT.lines[ll] = TT.lines[--TT.count];
62*cf5a6c84SAndroid Build Coastguard Worker }
63*cf5a6c84SAndroid Build Coastguard Worker }
64