1 /* factor -- print factors of n. lose if n > 2^32.
2 Copyright (C) 86, 1995 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
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
18 /* Written by Paul Rubin <phr@ocf.berkeley.edu>.
19 Adapted for GNU, fixed to factor UINT_MAX by Jim Meyering. */
24 #include <sys/types.h>
31 #endif /* HAVE_LIMITS_H */
34 # define UINT_MAX ((unsigned int) ~(unsigned int) 0)
38 # define INT_MAX ((int) (UINT_MAX >> 1))
43 #include "long-options.h"
46 #include "readtokens.h"
48 /* Token delimiters when reading from a file. */
51 /* FIXME: if this program is ever modified to factor integers larger
52 than 2^128, this constant (and the algorithm :-) will have to change. */
53 #define MAX_N_FACTORS 128
55 /* The name this program was run with. */
62 fprintf (stderr
, _("Try `%s --help' for more information.\n"),
67 Usage: %s [NUMBER...]\n\
70 program_name
, program_name
);
73 --help display this help and exit\n\
74 --version output version information and exit\n\
83 factor (long unsigned int n0
, int max_n_factors
, long unsigned int *factors
)
85 register unsigned long n
= n0
, d
;
94 assert (n_factors
< max_n_factors
);
95 factors
[n_factors
++] = 2;
99 /* The exit condition in the following loop is correct because
100 any time it is tested one of these 3 conditions holds:
103 (3) n is composite but has no factors less than d.
104 If (1) or (2) obviously the right thing happens.
105 If (3), then since n is composite it is >= d^2. */
106 sqrt_n
= (unsigned int) sqrt ((double) n
);
107 for (d
= 3; d
<= sqrt_n
; d
+= 2)
111 assert (n_factors
< max_n_factors
);
112 factors
[n_factors
++] = d
;
116 if (n
!= 1 || n0
== 1)
118 assert (n_factors
< max_n_factors
);
119 factors
[n_factors
++] = n
;
128 print_factors (const char *s
)
130 unsigned long int factors
[MAX_N_FACTORS
];
135 if (xstrtoul (s
, NULL
, 10, &n
, NULL
) != LONGINT_OK
)
137 error (0, 0, _("%s: invalid argument"), s
);
140 n_factors
= factor (n
, MAX_N_FACTORS
, factors
);
142 for (i
= 0; i
< n_factors
; i
++)
143 printf (" %lu", factors
[i
]);
152 token_buffer tokenbuffer
;
154 init_tokenbuffer (&tokenbuffer
);
158 long int token_length
;
160 token_length
= readtoken (stdin
, DELIM
, sizeof (DELIM
) - 1,
162 if (token_length
< 0)
164 fail
|= print_factors (tokenbuffer
.buffer
);
166 free (tokenbuffer
.buffer
);
172 main (int argc
, char **argv
)
176 program_name
= argv
[0];
177 setlocale (LC_ALL
, "");
178 bindtextdomain (PACKAGE
, LOCALEDIR
);
179 textdomain (PACKAGE
);
181 parse_long_options (argc
, argv
, "factor", version_string
, usage
);
189 for (i
= 1; i
< argc
; i
++)
190 fail
|= print_factors (argv
[i
]);