rndctl.c revision 1.9 1 1.9 enami /* $NetBSD: rndctl.c,v 1.9 2001/09/08 23:20:37 enami Exp $ */
2 1.3 perry
3 1.1 explorer /*-
4 1.1 explorer * Copyright (c) 1997 Michael Graff.
5 1.1 explorer * All rights reserved.
6 1.1 explorer *
7 1.1 explorer * Redistribution and use in source and binary forms, with or without
8 1.1 explorer * modification, are permitted provided that the following conditions
9 1.1 explorer * are met:
10 1.1 explorer * 1. Redistributions of source code must retain the above copyright
11 1.1 explorer * notice, this list of conditions and the following disclaimer.
12 1.1 explorer * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 explorer * notice, this list of conditions and the following disclaimer in the
14 1.1 explorer * documentation and/or other materials provided with the distribution.
15 1.1 explorer * 3. Neither the name of the author nor the names of other contributors
16 1.1 explorer * may be used to endorse or promote products derived from this software
17 1.1 explorer * without specific prior written permission.
18 1.1 explorer *
19 1.1 explorer * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20 1.1 explorer * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
21 1.1 explorer * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
22 1.1 explorer * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
23 1.1 explorer * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
24 1.1 explorer * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
25 1.1 explorer * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
26 1.1 explorer * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
27 1.1 explorer * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28 1.1 explorer * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 1.1 explorer * SUCH DAMAGE.
30 1.1 explorer */
31 1.1 explorer
32 1.1 explorer #include <stdio.h>
33 1.1 explorer #include <stdlib.h>
34 1.1 explorer #include <unistd.h>
35 1.1 explorer #include <fcntl.h>
36 1.1 explorer #include <errno.h>
37 1.1 explorer #include <err.h>
38 1.2 explorer #include <string.h>
39 1.1 explorer
40 1.1 explorer #include <sys/types.h>
41 1.1 explorer #include <sys/ioctl.h>
42 1.1 explorer #include <sys/rnd.h>
43 1.1 explorer
44 1.1 explorer typedef struct {
45 1.9 enami char *a_name;
46 1.9 enami u_int32_t a_type;
47 1.1 explorer } arg_t;
48 1.1 explorer
49 1.1 explorer arg_t source_types[] = {
50 1.6 sommerfe { "???", RND_TYPE_UNKNOWN },
51 1.1 explorer { "disk", RND_TYPE_DISK },
52 1.6 sommerfe { "net", RND_TYPE_NET },
53 1.1 explorer { "net", RND_TYPE_NET },
54 1.1 explorer { "tape", RND_TYPE_TAPE },
55 1.1 explorer { "tty", RND_TYPE_TTY },
56 1.1 explorer { NULL, 0 }
57 1.1 explorer };
58 1.1 explorer
59 1.2 explorer static void usage(void);
60 1.2 explorer u_int32_t find_type(char *name);
61 1.2 explorer char *find_name(u_int32_t);
62 1.2 explorer void do_ioctl(rndctl_t *);
63 1.2 explorer char * strflags(u_int32_t);
64 1.2 explorer void do_list(int, u_int32_t, char *);
65 1.6 sommerfe void do_stats(void);
66 1.2 explorer
67 1.2 explorer static void
68 1.1 explorer usage(void)
69 1.1 explorer {
70 1.9 enami
71 1.5 mycroft fprintf(stderr, "usage: rndctl -CEce [-t devtype] [-d devname]\n");
72 1.8 joda fprintf(stderr, " rndctl -ls [-t devtype] [-d devname]\n");
73 1.5 mycroft exit(1);
74 1.1 explorer }
75 1.1 explorer
76 1.1 explorer u_int32_t
77 1.1 explorer find_type(char *name)
78 1.1 explorer {
79 1.1 explorer arg_t *a;
80 1.1 explorer
81 1.1 explorer a = source_types;
82 1.9 enami
83 1.9 enami while (a->a_name != NULL) {
84 1.9 enami if (strcmp(a->a_name, name) == 0)
85 1.9 enami return (a->a_type);
86 1.1 explorer a++;
87 1.1 explorer }
88 1.1 explorer
89 1.1 explorer errx(1, "Error: Device type %s unknown", name);
90 1.9 enami return (0);
91 1.1 explorer }
92 1.1 explorer
93 1.1 explorer char *
94 1.1 explorer find_name(u_int32_t type)
95 1.1 explorer {
96 1.1 explorer arg_t *a;
97 1.1 explorer
98 1.1 explorer a = source_types;
99 1.9 enami
100 1.9 enami while (a->a_name != NULL) {
101 1.9 enami if (type == a->a_type)
102 1.9 enami return (a->a_name);
103 1.1 explorer a++;
104 1.1 explorer }
105 1.1 explorer
106 1.1 explorer errx(1, "Error: Device type %u unknown", type);
107 1.9 enami return (0);
108 1.1 explorer }
109 1.1 explorer
110 1.1 explorer void
111 1.1 explorer do_ioctl(rndctl_t *rctl)
112 1.1 explorer {
113 1.1 explorer int fd;
114 1.1 explorer int res;
115 1.1 explorer
116 1.1 explorer fd = open("/dev/urandom", O_RDONLY, 0644);
117 1.1 explorer if (fd < 0)
118 1.1 explorer err(1, "open");
119 1.1 explorer
120 1.1 explorer res = ioctl(fd, RNDCTL, rctl);
121 1.1 explorer if (res < 0)
122 1.1 explorer err(1, "ioctl(RNDCTL)");
123 1.1 explorer
124 1.1 explorer close(fd);
125 1.1 explorer }
126 1.1 explorer
127 1.1 explorer char *
128 1.1 explorer strflags(u_int32_t fl)
129 1.1 explorer {
130 1.1 explorer static char str[512];
131 1.1 explorer
132 1.1 explorer str[0] = 0;
133 1.1 explorer if (fl & RND_FLAG_NO_ESTIMATE)
134 1.6 sommerfe ;
135 1.9 enami else
136 1.6 sommerfe strcat(str, "estimate");
137 1.9 enami
138 1.1 explorer if (fl & RND_FLAG_NO_COLLECT)
139 1.6 sommerfe ;
140 1.6 sommerfe else {
141 1.6 sommerfe if (str[0])
142 1.6 sommerfe strcat(str, ", ");
143 1.6 sommerfe strcat(str, "collect");
144 1.6 sommerfe }
145 1.9 enami
146 1.9 enami return (str);
147 1.1 explorer }
148 1.1 explorer
149 1.6 sommerfe #define HEADER "Source Bits Type Flags\n"
150 1.1 explorer
151 1.1 explorer void
152 1.1 explorer do_list(int all, u_int32_t type, char *name)
153 1.1 explorer {
154 1.9 enami rndstat_t rstat;
155 1.9 enami rndstat_name_t rstat_name;
156 1.9 enami int fd;
157 1.9 enami int res;
158 1.9 enami u_int32_t start;
159 1.1 explorer
160 1.1 explorer fd = open("/dev/urandom", O_RDONLY, 0644);
161 1.1 explorer if (fd < 0)
162 1.1 explorer err(1, "open");
163 1.1 explorer
164 1.1 explorer if (all == 0 && type == 0xff) {
165 1.2 explorer strncpy(rstat_name.name, name, 16);
166 1.1 explorer res = ioctl(fd, RNDGETSRCNAME, &rstat_name);
167 1.1 explorer if (res < 0)
168 1.1 explorer err(1, "ioctl(RNDGETSRCNAME)");
169 1.1 explorer printf(HEADER);
170 1.6 sommerfe printf("%-16s %10u %-4s %s\n",
171 1.9 enami rstat_name.source.name,
172 1.9 enami rstat_name.source.total,
173 1.9 enami find_name(rstat_name.source.type),
174 1.9 enami strflags(rstat_name.source.flags));
175 1.1 explorer close(fd);
176 1.1 explorer return;
177 1.1 explorer }
178 1.1 explorer
179 1.1 explorer /*
180 1.9 enami * Run through all the devices present in the system, and either
181 1.1 explorer * print out ones that match, or print out all of them.
182 1.1 explorer */
183 1.1 explorer printf(HEADER);
184 1.1 explorer start = 0;
185 1.1 explorer for (;;) {
186 1.1 explorer rstat.count = RND_MAXSTATCOUNT;
187 1.1 explorer rstat.start = start;
188 1.1 explorer res = ioctl(fd, RNDGETSRCNUM, &rstat);
189 1.1 explorer if (res < 0)
190 1.1 explorer err(1, "ioctl(RNDGETSRCNUM)");
191 1.9 enami
192 1.1 explorer if (rstat.count == 0)
193 1.1 explorer break;
194 1.9 enami
195 1.9 enami for (res = 0; res < rstat.count; res++) {
196 1.9 enami if (all != 0 ||
197 1.9 enami type == rstat.source[res].type)
198 1.6 sommerfe printf("%-16s %10u %-4s %s\n",
199 1.9 enami rstat.source[res].name,
200 1.9 enami rstat.source[res].total,
201 1.9 enami find_name(rstat.source[res].type),
202 1.9 enami strflags(rstat.source[res].flags));
203 1.1 explorer }
204 1.1 explorer start += rstat.count;
205 1.1 explorer }
206 1.1 explorer
207 1.1 explorer close(fd);
208 1.1 explorer }
209 1.1 explorer
210 1.6 sommerfe void
211 1.6 sommerfe do_stats()
212 1.6 sommerfe {
213 1.6 sommerfe rndpoolstat_t rs;
214 1.6 sommerfe int fd;
215 1.9 enami
216 1.6 sommerfe fd = open("/dev/urandom", O_RDONLY, 0644);
217 1.6 sommerfe if (fd < 0)
218 1.6 sommerfe err(1, "open");
219 1.9 enami
220 1.6 sommerfe if (ioctl(fd, RNDGETPOOLSTAT, &rs) < 0)
221 1.6 sommerfe err(1, "ioctl(RNDGETPOOLSTAT)");
222 1.6 sommerfe
223 1.6 sommerfe printf("\t%9d bits mixed into pool\n", rs.added);
224 1.6 sommerfe printf("\t%9d bits currently stored in pool (max %d)\n",
225 1.6 sommerfe rs.curentropy, rs.maxentropy);
226 1.6 sommerfe printf("\t%9d bits of entropy discarded due to full pool\n",
227 1.6 sommerfe rs.discarded);
228 1.6 sommerfe printf("\t%9d hard-random bits generated\n", rs.removed);
229 1.6 sommerfe printf("\t%9d pseudo-random bits generated\n", rs.generated);
230 1.6 sommerfe
231 1.6 sommerfe close(fd);
232 1.6 sommerfe }
233 1.6 sommerfe
234 1.1 explorer int
235 1.1 explorer main(int argc, char **argv)
236 1.1 explorer {
237 1.9 enami rndctl_t rctl;
238 1.9 enami int ch, cmd, lflag, mflag, sflag;
239 1.1 explorer u_int32_t type;
240 1.9 enami char name[16];
241 1.1 explorer
242 1.1 explorer rctl.mask = 0;
243 1.1 explorer rctl.flags = 0;
244 1.1 explorer
245 1.1 explorer cmd = 0;
246 1.1 explorer lflag = 0;
247 1.1 explorer mflag = 0;
248 1.7 joda sflag = 0;
249 1.2 explorer type = 0xff;
250 1.1 explorer
251 1.6 sommerfe while ((ch = getopt(argc, argv, "CEcelt:d:s")) != -1)
252 1.9 enami switch (ch) {
253 1.1 explorer case 'C':
254 1.1 explorer rctl.flags |= RND_FLAG_NO_COLLECT;
255 1.1 explorer rctl.mask |= RND_FLAG_NO_COLLECT;
256 1.1 explorer mflag++;
257 1.1 explorer break;
258 1.1 explorer case 'E':
259 1.1 explorer rctl.flags |= RND_FLAG_NO_ESTIMATE;
260 1.1 explorer rctl.mask |= RND_FLAG_NO_ESTIMATE;
261 1.1 explorer mflag++;
262 1.1 explorer break;
263 1.1 explorer case 'c':
264 1.1 explorer rctl.flags &= ~RND_FLAG_NO_COLLECT;
265 1.1 explorer rctl.mask |= RND_FLAG_NO_COLLECT;
266 1.1 explorer mflag++;
267 1.1 explorer break;
268 1.1 explorer case 'e':
269 1.1 explorer rctl.flags &= ~RND_FLAG_NO_ESTIMATE;
270 1.1 explorer rctl.mask |= RND_FLAG_NO_ESTIMATE;
271 1.1 explorer mflag++;
272 1.1 explorer break;
273 1.1 explorer case 'l':
274 1.1 explorer lflag++;
275 1.1 explorer break;
276 1.1 explorer case 't':
277 1.1 explorer if (cmd != 0)
278 1.1 explorer usage();
279 1.1 explorer cmd = 't';
280 1.1 explorer
281 1.1 explorer type = find_type(optarg);
282 1.1 explorer break;
283 1.1 explorer case 'd':
284 1.1 explorer if (cmd != 0)
285 1.1 explorer usage();
286 1.1 explorer cmd = 'd';
287 1.1 explorer
288 1.1 explorer type = 0xff;
289 1.1 explorer strncpy(name, optarg, 16);
290 1.1 explorer break;
291 1.6 sommerfe case 's':
292 1.6 sommerfe sflag++;
293 1.6 sommerfe break;
294 1.1 explorer case '?':
295 1.1 explorer default:
296 1.1 explorer usage();
297 1.1 explorer }
298 1.1 explorer
299 1.1 explorer /*
300 1.9 enami * Cannot list and modify at the same time.
301 1.1 explorer */
302 1.6 sommerfe if ((lflag != 0 || sflag != 0) && mflag != 0)
303 1.1 explorer usage();
304 1.1 explorer
305 1.1 explorer /*
306 1.9 enami * Bomb out on no-ops.
307 1.1 explorer */
308 1.6 sommerfe if (lflag == 0 && mflag == 0 && sflag == 0)
309 1.1 explorer usage();
310 1.1 explorer
311 1.1 explorer /*
312 1.9 enami * If not listing, we need a device name or a type.
313 1.1 explorer */
314 1.6 sommerfe if (lflag == 0 && cmd == 0 && sflag == 0)
315 1.1 explorer usage();
316 1.1 explorer
317 1.1 explorer /*
318 1.9 enami * Modify request.
319 1.1 explorer */
320 1.1 explorer if (mflag != 0) {
321 1.1 explorer rctl.type = type;
322 1.1 explorer strncpy(rctl.name, name, 16);
323 1.1 explorer do_ioctl(&rctl);
324 1.1 explorer
325 1.1 explorer exit(0);
326 1.1 explorer }
327 1.1 explorer
328 1.1 explorer /*
329 1.9 enami * List sources.
330 1.1 explorer */
331 1.1 explorer if (lflag != 0)
332 1.1 explorer do_list(cmd == 0, type, name);
333 1.1 explorer
334 1.6 sommerfe if (sflag != 0)
335 1.6 sommerfe do_stats();
336 1.9 enami
337 1.9 enami exit(0);
338 1.1 explorer }
339