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