Home | History | Annotate | Line # | Download | only in rndctl
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