xref: /aosp_15_r20/external/toybox/toys/other/factor.c (revision cf5a6c84e2b8763fc1a7db14496fd4742913b199)
1*cf5a6c84SAndroid Build Coastguard Worker /* factor.c - Factor integers
2*cf5a6c84SAndroid Build Coastguard Worker  *
3*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2014 Rob Landley <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker  *
5*cf5a6c84SAndroid Build Coastguard Worker  * See https://man7.org/linux/man-pages/man1/factor.1.html
6*cf5a6c84SAndroid Build Coastguard Worker  *
7*cf5a6c84SAndroid Build Coastguard Worker  * -h and -x options come from https://man.netbsd.org/factor.6
8*cf5a6c84SAndroid Build Coastguard Worker 
9*cf5a6c84SAndroid Build Coastguard Worker USE_FACTOR(NEWTOY(factor, "?hx", TOYFLAG_USR|TOYFLAG_BIN))
10*cf5a6c84SAndroid Build Coastguard Worker 
11*cf5a6c84SAndroid Build Coastguard Worker config FACTOR
12*cf5a6c84SAndroid Build Coastguard Worker   bool "factor"
13*cf5a6c84SAndroid Build Coastguard Worker   default y
14*cf5a6c84SAndroid Build Coastguard Worker   help
15*cf5a6c84SAndroid Build Coastguard Worker     usage: factor [-hx] NUMBER...
16*cf5a6c84SAndroid Build Coastguard Worker 
17*cf5a6c84SAndroid Build Coastguard Worker     Factor integers.
18*cf5a6c84SAndroid Build Coastguard Worker 
19*cf5a6c84SAndroid Build Coastguard Worker     -h	Human readable: show repeated factors as x^n
20*cf5a6c84SAndroid Build Coastguard Worker     -x	Hexadecimal output
21*cf5a6c84SAndroid Build Coastguard Worker */
22*cf5a6c84SAndroid Build Coastguard Worker 
23*cf5a6c84SAndroid Build Coastguard Worker #define FOR_factor
24*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
25*cf5a6c84SAndroid Build Coastguard Worker 
factor(char * s)26*cf5a6c84SAndroid Build Coastguard Worker static void factor(char *s)
27*cf5a6c84SAndroid Build Coastguard Worker {
28*cf5a6c84SAndroid Build Coastguard Worker   unsigned long long l, ll, lll;
29*cf5a6c84SAndroid Build Coastguard Worker   char *pat1 = FLAG(x) ? " %llx" : " %llu", *pat2 = FLAG(x) ? "^%x" : "^%u";
30*cf5a6c84SAndroid Build Coastguard Worker 
31*cf5a6c84SAndroid Build Coastguard Worker   for (;;) {
32*cf5a6c84SAndroid Build Coastguard Worker     char *err = s;
33*cf5a6c84SAndroid Build Coastguard Worker     int dash = 0, ii;
34*cf5a6c84SAndroid Build Coastguard Worker 
35*cf5a6c84SAndroid Build Coastguard Worker     while(isspace(*s)) s++;
36*cf5a6c84SAndroid Build Coastguard Worker     if (*s=='-') dash = *s++;
37*cf5a6c84SAndroid Build Coastguard Worker     if (!*s) return;
38*cf5a6c84SAndroid Build Coastguard Worker 
39*cf5a6c84SAndroid Build Coastguard Worker     errno = 0;
40*cf5a6c84SAndroid Build Coastguard Worker     l = strtoull(s, &s, 0);
41*cf5a6c84SAndroid Build Coastguard Worker     if (errno || (*s && !isspace(*s))) {
42*cf5a6c84SAndroid Build Coastguard Worker       error_msg("%s: not integer", err);
43*cf5a6c84SAndroid Build Coastguard Worker       while (*s && !isspace(*s)) s++;
44*cf5a6c84SAndroid Build Coastguard Worker       continue;
45*cf5a6c84SAndroid Build Coastguard Worker     }
46*cf5a6c84SAndroid Build Coastguard Worker 
47*cf5a6c84SAndroid Build Coastguard Worker     if (dash) xputc('-');
48*cf5a6c84SAndroid Build Coastguard Worker     printf(pat1+1, l);
49*cf5a6c84SAndroid Build Coastguard Worker     xputc(':');
50*cf5a6c84SAndroid Build Coastguard Worker 
51*cf5a6c84SAndroid Build Coastguard Worker     // Negative numbers have -1 as a factor
52*cf5a6c84SAndroid Build Coastguard Worker     if (dash) printf(" -1");
53*cf5a6c84SAndroid Build Coastguard Worker 
54*cf5a6c84SAndroid Build Coastguard Worker     // test 2 and odd numbers until square is > remainder or integer wrap.
55*cf5a6c84SAndroid Build Coastguard Worker     for (ll = 2;; ll += 1+(ll!=2)) {
56*cf5a6c84SAndroid Build Coastguard Worker       lll = ll*ll;
57*cf5a6c84SAndroid Build Coastguard Worker       if (lll>l || lll<ll) {
58*cf5a6c84SAndroid Build Coastguard Worker         if (l>1 || ll==2) printf(pat1, l);
59*cf5a6c84SAndroid Build Coastguard Worker         break;
60*cf5a6c84SAndroid Build Coastguard Worker       }
61*cf5a6c84SAndroid Build Coastguard Worker       for (ii = 0; !(l%ll); ii++) {
62*cf5a6c84SAndroid Build Coastguard Worker         if (!ii || !FLAG(h)) printf(pat1, ll);
63*cf5a6c84SAndroid Build Coastguard Worker         l /= ll;
64*cf5a6c84SAndroid Build Coastguard Worker       }
65*cf5a6c84SAndroid Build Coastguard Worker       if (ii>1 && FLAG(h)) printf(pat2, ii);
66*cf5a6c84SAndroid Build Coastguard Worker     }
67*cf5a6c84SAndroid Build Coastguard Worker     xputc('\n');
68*cf5a6c84SAndroid Build Coastguard Worker   }
69*cf5a6c84SAndroid Build Coastguard Worker }
70*cf5a6c84SAndroid Build Coastguard Worker 
factor_main(void)71*cf5a6c84SAndroid Build Coastguard Worker void factor_main(void)
72*cf5a6c84SAndroid Build Coastguard Worker {
73*cf5a6c84SAndroid Build Coastguard Worker   char *s = 0, **ss;
74*cf5a6c84SAndroid Build Coastguard Worker   size_t len = 0;
75*cf5a6c84SAndroid Build Coastguard Worker 
76*cf5a6c84SAndroid Build Coastguard Worker   if (toys.optc) for (ss = toys.optargs; *ss; ss++) factor(*ss);
77*cf5a6c84SAndroid Build Coastguard Worker   else for (;;) {
78*cf5a6c84SAndroid Build Coastguard Worker     if (-1 == getline(&s, &len, stdin)) break;
79*cf5a6c84SAndroid Build Coastguard Worker     factor(s);
80*cf5a6c84SAndroid Build Coastguard Worker   }
81*cf5a6c84SAndroid Build Coastguard Worker }
82