1 /* factor -- print prime factors of n.
2 Copyright (C) 86, 1995-2004 Free Software Foundation, Inc.
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2, or (at your option)
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software Foundation,
16 Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
18 /* Written by Paul Rubin <phr@ocf.berkeley.edu>.
19 Adapted for GNU, fixed to factor UINT_MAX by Jim Meyering. */
23 #include <sys/types.h>
31 #include "long-options.h"
32 #include "readtokens.h"
35 /* The official name of this program (e.g., no `g' prefix). */
36 #define PROGRAM_NAME "factor"
38 #define AUTHORS "Paul Rubin"
40 /* Token delimiters when reading from a file. */
43 /* FIXME: if this program is ever modified to factor integers larger
44 than 2^128, this constant (and the algorithm :-) will have to change. */
45 #define MAX_N_FACTORS 128
47 /* The trial divisor increment wheel. Use it to skip over divisors that
48 are composites of 2, 3, 5, 7, or 11. The part from WHEEL_START up to
49 WHEEL_END is reused periodically, while the "lead in" is used to test
50 for those primes and to jump onto the wheel. For more information, see
51 http://www.utm.edu/research/primes/glossary/WheelFactorization.html */
53 #include "wheel-size.h" /* For the definition of WHEEL_SIZE. */
54 static const unsigned int wheel_tab
[] =
59 #define WHEEL_START (wheel_tab + WHEEL_SIZE)
60 #define WHEEL_END (wheel_tab + (sizeof wheel_tab / sizeof wheel_tab[0]))
62 /* The name this program was run with. */
68 if (status
!= EXIT_SUCCESS
)
69 fprintf (stderr
, _("Try `%s --help' for more information.\n"),
74 Usage: %s [NUMBER]...\n\
77 program_name
, program_name
);
79 Print the prime factors of each NUMBER.\n\
82 fputs (HELP_OPTION_DESCRIPTION
, stdout
);
83 fputs (VERSION_OPTION_DESCRIPTION
, stdout
);
86 Print the prime factors of all specified integer NUMBERs. If no arguments\n\
87 are specified on the command line, they are read from standard input.\n\
89 printf (_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT
);
97 factor (uintmax_t n0
, int max_n_factors
, uintmax_t *factors
)
99 register uintmax_t n
= n0
, d
, q
;
101 unsigned int const *w
= wheel_tab
;
106 /* The exit condition in the following loop is correct because
107 any time it is tested one of these 3 conditions holds:
110 (3) n is composite but has no factors less than d.
111 If (1) or (2) obviously the right thing happens.
112 If (3), then since n is composite it is >= d^2. */
120 assert (n_factors
< max_n_factors
);
121 factors
[n_factors
++] = d
;
131 if (n
!= 1 || n0
== 1)
133 assert (n_factors
< max_n_factors
);
134 factors
[n_factors
++] = n
;
143 print_factors (const char *s
)
145 uintmax_t factors
[MAX_N_FACTORS
];
149 char buf
[INT_BUFSIZE_BOUND (uintmax_t)];
152 if ((err
= xstrtoumax (s
, NULL
, 10, &n
, "")) != LONGINT_OK
)
154 if (err
== LONGINT_OVERFLOW
)
155 error (0, 0, _("`%s' is too large"), s
);
157 error (0, 0, _("`%s' is not a valid positive integer"), s
);
160 n_factors
= factor (n
, MAX_N_FACTORS
, factors
);
161 printf ("%s:", umaxtostr (n
, buf
));
162 for (i
= 0; i
< n_factors
; i
++)
163 printf (" %s", umaxtostr (factors
[i
], buf
));
172 token_buffer tokenbuffer
;
174 init_tokenbuffer (&tokenbuffer
);
178 long int token_length
;
180 token_length
= readtoken (stdin
, DELIM
, sizeof (DELIM
) - 1,
182 if (token_length
< 0)
184 fail
|= print_factors (tokenbuffer
.buffer
);
186 free (tokenbuffer
.buffer
);
192 main (int argc
, char **argv
)
196 initialize_main (&argc
, &argv
);
197 program_name
= argv
[0];
198 setlocale (LC_ALL
, "");
199 bindtextdomain (PACKAGE
, LOCALEDIR
);
200 textdomain (PACKAGE
);
202 atexit (close_stdout
);
204 parse_long_options (argc
, argv
, PROGRAM_NAME
, GNU_PACKAGE
, VERSION
,
205 usage
, AUTHORS
, (char const *) NULL
);
206 /* The above handles --help and --version.
207 Since there is no other invocation of getopt, handle `--' here. */
208 if (argc
> 1 && STREQ (argv
[1], "--"))
220 for (i
= 1; i
< argc
; i
++)
221 fail
|= print_factors (argv
[i
]);
223 usage (EXIT_FAILURE
);
226 exit (fail
? EXIT_FAILURE
: EXIT_SUCCESS
);