xref: /freebsd/sbin/mdconfig/mdconfig.c (revision b830359bc58e3733d84002a4e2c438c378ff5b30)
170d586c0SPoul-Henning Kamp /*
270d586c0SPoul-Henning Kamp  * ----------------------------------------------------------------------------
370d586c0SPoul-Henning Kamp  * "THE BEER-WARE LICENSE" (Revision 42):
470d586c0SPoul-Henning Kamp  * <phk@FreeBSD.ORG> wrote this file.  As long as you retain this notice you
570d586c0SPoul-Henning Kamp  * can do whatever you want with this stuff. If we meet some day, and you think
670d586c0SPoul-Henning Kamp  * this stuff is worth it, you can buy me a beer in return.   Poul-Henning Kamp
770d586c0SPoul-Henning Kamp  * ----------------------------------------------------------------------------
870d586c0SPoul-Henning Kamp  *
970d586c0SPoul-Henning Kamp  * $FreeBSD$
1070d586c0SPoul-Henning Kamp  *
1170d586c0SPoul-Henning Kamp  */
1270d586c0SPoul-Henning Kamp 
1370d586c0SPoul-Henning Kamp #include <stdio.h>
1470d586c0SPoul-Henning Kamp #include <stdlib.h>
1570d586c0SPoul-Henning Kamp #include <fcntl.h>
1670d586c0SPoul-Henning Kamp #include <unistd.h>
17b830359bSPawel Jakub Dawidek #include <inttypes.h>
18b830359bSPawel Jakub Dawidek #include <libutil.h>
1970d586c0SPoul-Henning Kamp #include <string.h>
2070d586c0SPoul-Henning Kamp #include <err.h>
21b830359bSPawel Jakub Dawidek 
2270d586c0SPoul-Henning Kamp #include <sys/ioctl.h>
2370d586c0SPoul-Henning Kamp #include <sys/param.h>
2457e9624eSPoul-Henning Kamp #include <sys/module.h>
2557e9624eSPoul-Henning Kamp #include <sys/linker.h>
2670d586c0SPoul-Henning Kamp #include <sys/mdioctl.h>
27b830359bSPawel Jakub Dawidek #include <sys/stat.h>
28174b5e9aSPoul-Henning Kamp #include <sys/sysctl.h>
29174b5e9aSPoul-Henning Kamp #include <sys/queue.h>
30174b5e9aSPoul-Henning Kamp 
31174b5e9aSPoul-Henning Kamp int	 list(const int);
323fa96e66SDima Dorfman void	 mdmaybeload(void);
33174b5e9aSPoul-Henning Kamp int	 query(const int, const int);
343fa96e66SDima Dorfman void	 usage(void);
3570d586c0SPoul-Henning Kamp 
3670d586c0SPoul-Henning Kamp struct md_ioctl mdio;
3770d586c0SPoul-Henning Kamp 
38174b5e9aSPoul-Henning Kamp enum {UNSET, ATTACH, DETACH, LIST} action = UNSET;
3970d586c0SPoul-Henning Kamp 
40f79c46d3SRobert Watson int nflag;
41f79c46d3SRobert Watson 
42c2ef0b73SPoul-Henning Kamp void
43c2ef0b73SPoul-Henning Kamp usage()
44c2ef0b73SPoul-Henning Kamp {
45d3974088SDag-Erling Smørgrav 	fprintf(stderr, "usage:\n");
46f79c46d3SRobert Watson 	fprintf(stderr, "\tmdconfig -a -t type [-n] [-o [no]option]... [ -f file] [-s size] [-S sectorsize] [-u unit]\n");
478f8def9eSPoul-Henning Kamp 	fprintf(stderr, "\tmdconfig -d -u unit\n");
48f79c46d3SRobert Watson 	fprintf(stderr, "\tmdconfig -l [-n] [-u unit]\n");
49c2ef0b73SPoul-Henning Kamp 	fprintf(stderr, "\t\ttype = {malloc, preload, vnode, swap}\n");
503f6f9216SPoul-Henning Kamp 	fprintf(stderr, "\t\toption = {cluster, compress, reserve}\n");
51c2ef0b73SPoul-Henning Kamp 	fprintf(stderr, "\t\tsize = %%d (512 byte blocks), %%dk (kB), %%dm (MB) or %%dg (GB)\n");
52c2ef0b73SPoul-Henning Kamp 	exit(1);
53c2ef0b73SPoul-Henning Kamp }
54c2ef0b73SPoul-Henning Kamp 
5570d586c0SPoul-Henning Kamp int
5670d586c0SPoul-Henning Kamp main(int argc, char **argv)
5770d586c0SPoul-Henning Kamp {
5870d586c0SPoul-Henning Kamp 	int ch, fd, i;
59c2ef0b73SPoul-Henning Kamp 	char *p;
60c2ef0b73SPoul-Henning Kamp 	int cmdline = 0;
6170d586c0SPoul-Henning Kamp 
62b830359bSPawel Jakub Dawidek 	bzero(&mdio, sizeof(mdio));
6370d586c0SPoul-Henning Kamp 	for (;;) {
64f79c46d3SRobert Watson 		ch = getopt(argc, argv, "ab:df:lno:s:S:t:u:x:y:");
6570d586c0SPoul-Henning Kamp 		if (ch == -1)
6670d586c0SPoul-Henning Kamp 			break;
6770d586c0SPoul-Henning Kamp 		switch (ch) {
6870d586c0SPoul-Henning Kamp 		case 'a':
69c2ef0b73SPoul-Henning Kamp 			if (cmdline != 0)
70c2ef0b73SPoul-Henning Kamp 				usage();
7170d586c0SPoul-Henning Kamp 			action = ATTACH;
72c2ef0b73SPoul-Henning Kamp 			cmdline = 1;
7370d586c0SPoul-Henning Kamp 			break;
7470d586c0SPoul-Henning Kamp 		case 'd':
75c2ef0b73SPoul-Henning Kamp 			if (cmdline != 0)
76c2ef0b73SPoul-Henning Kamp 				usage();
7770d586c0SPoul-Henning Kamp 			action = DETACH;
788f8def9eSPoul-Henning Kamp 			mdio.md_options = MD_AUTOUNIT;
798f8def9eSPoul-Henning Kamp 			cmdline = 3;
80c2ef0b73SPoul-Henning Kamp 			break;
81174b5e9aSPoul-Henning Kamp 		case 'l':
82174b5e9aSPoul-Henning Kamp 			if (cmdline != 0)
83174b5e9aSPoul-Henning Kamp 				usage();
84174b5e9aSPoul-Henning Kamp 			action = LIST;
85174b5e9aSPoul-Henning Kamp 			mdio.md_options = MD_AUTOUNIT;
86174b5e9aSPoul-Henning Kamp 			cmdline = 3;
87174b5e9aSPoul-Henning Kamp 			break;
88f79c46d3SRobert Watson 		case 'n':
89f79c46d3SRobert Watson 			nflag = 1;
90f79c46d3SRobert Watson 			break;
91c2ef0b73SPoul-Henning Kamp 		case 't':
92c2ef0b73SPoul-Henning Kamp 			if (cmdline != 1)
93c2ef0b73SPoul-Henning Kamp 				usage();
94c2ef0b73SPoul-Henning Kamp 			if (!strcmp(optarg, "malloc")) {
95c2ef0b73SPoul-Henning Kamp 				mdio.md_type = MD_MALLOC;
96c2ef0b73SPoul-Henning Kamp 				mdio.md_options = MD_AUTOUNIT | MD_COMPRESS;
97c2ef0b73SPoul-Henning Kamp 			} else if (!strcmp(optarg, "preload")) {
98c2ef0b73SPoul-Henning Kamp 				mdio.md_type = MD_PRELOAD;
99c2ef0b73SPoul-Henning Kamp 				mdio.md_options = 0;
100c2ef0b73SPoul-Henning Kamp 			} else if (!strcmp(optarg, "vnode")) {
101c2ef0b73SPoul-Henning Kamp 				mdio.md_type = MD_VNODE;
102c2ef0b73SPoul-Henning Kamp 				mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
103c2ef0b73SPoul-Henning Kamp 			} else if (!strcmp(optarg, "swap")) {
104c2ef0b73SPoul-Henning Kamp 				mdio.md_type = MD_SWAP;
105c2ef0b73SPoul-Henning Kamp 				mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
106c2ef0b73SPoul-Henning Kamp 			} else {
107c2ef0b73SPoul-Henning Kamp 				usage();
108c2ef0b73SPoul-Henning Kamp 			}
109c2ef0b73SPoul-Henning Kamp 			cmdline=2;
11070d586c0SPoul-Henning Kamp 			break;
11170d586c0SPoul-Henning Kamp 		case 'f':
112ed23a390SMaxim Sobolev 			if (cmdline != 1 && cmdline != 2)
113c2ef0b73SPoul-Henning Kamp 				usage();
114ed23a390SMaxim Sobolev 			if (cmdline == 1) {
115ed23a390SMaxim Sobolev 				/* Imply ``-t vnode'' */
116ed23a390SMaxim Sobolev 				mdio.md_type = MD_VNODE;
117ed23a390SMaxim Sobolev 				mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
118252bcf45SYaroslav Tykhiy 				cmdline = 2;
119ed23a390SMaxim Sobolev 			}
1203b42f2f3SPoul-Henning Kamp 			mdio.md_file = optarg;
121e869d377SPoul-Henning Kamp 			fd = open(optarg, O_RDONLY);
122e869d377SPoul-Henning Kamp 			if (fd < 0)
123e869d377SPoul-Henning Kamp 				err(1, "could not open %s", optarg);
124b830359bSPawel Jakub Dawidek 			else if (mdio.md_mediasize == 0) {
125b830359bSPawel Jakub Dawidek 				struct stat sb;
126b830359bSPawel Jakub Dawidek 
127b830359bSPawel Jakub Dawidek 				if (fstat(fd, &sb) == -1)
128b830359bSPawel Jakub Dawidek 					err(1, "could not stat %s", optarg);
129b830359bSPawel Jakub Dawidek 				mdio.md_mediasize = sb.st_size;
130b830359bSPawel Jakub Dawidek 			}
131e869d377SPoul-Henning Kamp 			close(fd);
13270d586c0SPoul-Henning Kamp 			break;
13370d586c0SPoul-Henning Kamp 		case 'o':
134c2ef0b73SPoul-Henning Kamp 			if (cmdline != 2)
135c2ef0b73SPoul-Henning Kamp 				usage();
1367a6b2b64SPoul-Henning Kamp 			if (!strcmp(optarg, "async"))
1377a6b2b64SPoul-Henning Kamp 				mdio.md_options |= MD_ASYNC;
1387a6b2b64SPoul-Henning Kamp 			else if (!strcmp(optarg, "noasync"))
1397a6b2b64SPoul-Henning Kamp 				mdio.md_options &= ~MD_ASYNC;
1407a6b2b64SPoul-Henning Kamp 			else if (!strcmp(optarg, "cluster"))
14170d586c0SPoul-Henning Kamp 				mdio.md_options |= MD_CLUSTER;
14270d586c0SPoul-Henning Kamp 			else if (!strcmp(optarg, "nocluster"))
14370d586c0SPoul-Henning Kamp 				mdio.md_options &= ~MD_CLUSTER;
144c2ef0b73SPoul-Henning Kamp 			else if (!strcmp(optarg, "compress"))
145c2ef0b73SPoul-Henning Kamp 				mdio.md_options |= MD_COMPRESS;
146c2ef0b73SPoul-Henning Kamp 			else if (!strcmp(optarg, "nocompress"))
147c2ef0b73SPoul-Henning Kamp 				mdio.md_options &= ~MD_COMPRESS;
14826a0ee75SDima Dorfman 			else if (!strcmp(optarg, "force"))
14926a0ee75SDima Dorfman 				mdio.md_options |= MD_FORCE;
15026a0ee75SDima Dorfman 			else if (!strcmp(optarg, "noforce"))
15126a0ee75SDima Dorfman 				mdio.md_options &= ~MD_FORCE;
152d31ba625SJohn-Mark Gurney 			else if (!strcmp(optarg, "readonly"))
153d31ba625SJohn-Mark Gurney 				mdio.md_options |= MD_READONLY;
154d31ba625SJohn-Mark Gurney 			else if (!strcmp(optarg, "noreadonly"))
155d31ba625SJohn-Mark Gurney 				mdio.md_options &= ~MD_READONLY;
15670d586c0SPoul-Henning Kamp 			else if (!strcmp(optarg, "reserve"))
15770d586c0SPoul-Henning Kamp 				mdio.md_options |= MD_RESERVE;
15870d586c0SPoul-Henning Kamp 			else if (!strcmp(optarg, "noreserve"))
15970d586c0SPoul-Henning Kamp 				mdio.md_options &= ~MD_RESERVE;
16070d586c0SPoul-Henning Kamp 			else
161d31ba625SJohn-Mark Gurney 				errx(1, "Unknown option: %s.", optarg);
16270d586c0SPoul-Henning Kamp 			break;
163ebe789d6SPoul-Henning Kamp 		case 'S':
164ebe789d6SPoul-Henning Kamp 			if (cmdline != 2)
165ebe789d6SPoul-Henning Kamp 				usage();
166b830359bSPawel Jakub Dawidek 			mdio.md_sectorsize = strtoul(optarg, &p, 0);
167ebe789d6SPoul-Henning Kamp 			break;
16870d586c0SPoul-Henning Kamp 		case 's':
169c2ef0b73SPoul-Henning Kamp 			if (cmdline != 2)
170c2ef0b73SPoul-Henning Kamp 				usage();
171b830359bSPawel Jakub Dawidek 			mdio.md_mediasize = (off_t)strtoumax(optarg, &p, 0);
172c2ef0b73SPoul-Henning Kamp 			if (p == NULL || *p == '\0')
173b830359bSPawel Jakub Dawidek 				mdio.md_mediasize *= DEV_BSIZE;
174c2ef0b73SPoul-Henning Kamp 			else if (*p == 'k' || *p == 'K')
175b830359bSPawel Jakub Dawidek 				mdio.md_mediasize <<= 10;
176c2ef0b73SPoul-Henning Kamp 			else if (*p == 'm' || *p == 'M')
177b830359bSPawel Jakub Dawidek 				mdio.md_mediasize <<= 20;
178c2ef0b73SPoul-Henning Kamp 			else if (*p == 'g' || *p == 'G')
179b830359bSPawel Jakub Dawidek 				mdio.md_mediasize <<= 30;
180b830359bSPawel Jakub Dawidek 			else if (*p == 't' || *p == 'T') {
181b830359bSPawel Jakub Dawidek 				mdio.md_mediasize <<= 30;
182b830359bSPawel Jakub Dawidek 				mdio.md_mediasize <<= 10;
183b830359bSPawel Jakub Dawidek 			} else
184c2ef0b73SPoul-Henning Kamp 				errx(1, "Unknown suffix on -s argument");
18570d586c0SPoul-Henning Kamp 			break;
18670d586c0SPoul-Henning Kamp 		case 'u':
1878f8def9eSPoul-Henning Kamp 			if (cmdline != 2 && cmdline != 3)
188c2ef0b73SPoul-Henning Kamp 				usage();
189fb1023d6SPoul-Henning Kamp 			if (!strncmp(optarg, "/dev/", 5))
190fb1023d6SPoul-Henning Kamp 				optarg += 5;
191174b5e9aSPoul-Henning Kamp 			if (!strncmp(optarg, MD_NAME, sizeof(MD_NAME) - 1))
19278baea25SDima Dorfman 				optarg += sizeof(MD_NAME) - 1;
1932885b421SDima Dorfman 			mdio.md_unit = strtoul(optarg, &p, 0);
1948a50130bSAlexander Kabaev 			if (mdio.md_unit == (unsigned)ULONG_MAX || *p != '\0')
1952885b421SDima Dorfman 				errx(1, "bad unit: %s", optarg);
19670d586c0SPoul-Henning Kamp 			mdio.md_options &= ~MD_AUTOUNIT;
19770d586c0SPoul-Henning Kamp 			break;
1984e8bfe14SPoul-Henning Kamp 		case 'x':
1994e8bfe14SPoul-Henning Kamp 			if (cmdline != 2)
2004e8bfe14SPoul-Henning Kamp 				usage();
2014e8bfe14SPoul-Henning Kamp 			mdio.md_fwsectors = strtoul(optarg, &p, 0);
2024e8bfe14SPoul-Henning Kamp 			break;
2034e8bfe14SPoul-Henning Kamp 		case 'y':
2044e8bfe14SPoul-Henning Kamp 			if (cmdline != 2)
2054e8bfe14SPoul-Henning Kamp 				usage();
2064e8bfe14SPoul-Henning Kamp 			mdio.md_fwheads = strtoul(optarg, &p, 0);
2074e8bfe14SPoul-Henning Kamp 			break;
20870d586c0SPoul-Henning Kamp 		default:
209c2ef0b73SPoul-Henning Kamp 			usage();
21070d586c0SPoul-Henning Kamp 		}
21170d586c0SPoul-Henning Kamp 	}
21253d745bcSDima Dorfman 	mdio.md_version = MDIOVERSION;
21370d586c0SPoul-Henning Kamp 
21457e9624eSPoul-Henning Kamp 	mdmaybeload();
215174b5e9aSPoul-Henning Kamp 	fd = open("/dev/" MDCTL_NAME, O_RDWR, 0);
21670d586c0SPoul-Henning Kamp 	if (fd < 0)
217174b5e9aSPoul-Henning Kamp 		err(1, "open(/dev/%s)", MDCTL_NAME);
2182885b421SDima Dorfman 	if (cmdline == 2
2192885b421SDima Dorfman 	    && (mdio.md_type == MD_MALLOC || mdio.md_type == MD_SWAP))
220b830359bSPawel Jakub Dawidek 		if (mdio.md_mediasize == 0)
2212885b421SDima Dorfman 			errx(1, "must specify -s for -t malloc or -t swap");
222252bcf45SYaroslav Tykhiy 	if (cmdline == 2 && mdio.md_type == MD_VNODE)
223252bcf45SYaroslav Tykhiy 		if (mdio.md_file == NULL)
224252bcf45SYaroslav Tykhiy 			errx(1, "must specify -f for -t vnode");
225174b5e9aSPoul-Henning Kamp 	if (action == LIST) {
226174b5e9aSPoul-Henning Kamp 		if (mdio.md_options & MD_AUTOUNIT)
227174b5e9aSPoul-Henning Kamp 			list(fd);
228174b5e9aSPoul-Henning Kamp 		else
229174b5e9aSPoul-Henning Kamp 			query(fd, mdio.md_unit);
230174b5e9aSPoul-Henning Kamp 	} else if (action == ATTACH) {
231252bcf45SYaroslav Tykhiy 		if (cmdline < 2)
232252bcf45SYaroslav Tykhiy 			usage();
23370d586c0SPoul-Henning Kamp 		i = ioctl(fd, MDIOCATTACH, &mdio);
234174b5e9aSPoul-Henning Kamp 		if (i < 0)
235174b5e9aSPoul-Henning Kamp 			err(1, "ioctl(/dev/%s)", MDCTL_NAME);
23683da2a90SPoul-Henning Kamp 		if (mdio.md_options & MD_AUTOUNIT)
237f79c46d3SRobert Watson 			printf("%s%d\n", nflag ? "" : MD_NAME, mdio.md_unit);
23883da2a90SPoul-Henning Kamp 	} else if (action == DETACH) {
2398f8def9eSPoul-Henning Kamp 		if (mdio.md_options & MD_AUTOUNIT)
2408f8def9eSPoul-Henning Kamp 			usage();
241c2ef0b73SPoul-Henning Kamp 		i = ioctl(fd, MDIOCDETACH, &mdio);
24270d586c0SPoul-Henning Kamp 		if (i < 0)
243174b5e9aSPoul-Henning Kamp 			err(1, "ioctl(/dev/%s)", MDCTL_NAME);
24483da2a90SPoul-Henning Kamp 	} else
24583da2a90SPoul-Henning Kamp 		usage();
246174b5e9aSPoul-Henning Kamp 	close (fd);
247174b5e9aSPoul-Henning Kamp 	return (0);
248174b5e9aSPoul-Henning Kamp }
249174b5e9aSPoul-Henning Kamp 
250174b5e9aSPoul-Henning Kamp struct dl {
251174b5e9aSPoul-Henning Kamp 	int		unit;
252174b5e9aSPoul-Henning Kamp 	SLIST_ENTRY(dl)	slist;
253174b5e9aSPoul-Henning Kamp };
254174b5e9aSPoul-Henning Kamp 
255174b5e9aSPoul-Henning Kamp SLIST_HEAD(, dl) dlist = SLIST_HEAD_INITIALIZER(&dlist);
256174b5e9aSPoul-Henning Kamp 
257174b5e9aSPoul-Henning Kamp int
258174b5e9aSPoul-Henning Kamp list(const int fd)
259174b5e9aSPoul-Henning Kamp {
260c894b25aSDima Dorfman 	int unit;
261174b5e9aSPoul-Henning Kamp 
262e39eff98SPoul-Henning Kamp 	if (ioctl(fd, MDIOCLIST, &mdio) < 0)
263e39eff98SPoul-Henning Kamp 		err(1, "ioctl(/dev/%s)", MDCTL_NAME);
264e39eff98SPoul-Henning Kamp 	for (unit = 0; unit < mdio.md_pad[0] && unit < MDNPAD - 1; unit++) {
265f79c46d3SRobert Watson 		printf("%s%s%d", unit > 0 ? " " : "",
266f79c46d3SRobert Watson 		    nflag ? "" : MD_NAME, mdio.md_pad[unit + 1]);
267174b5e9aSPoul-Henning Kamp 	}
268e39eff98SPoul-Henning Kamp 	if (mdio.md_pad[0] - unit > 0)
269e39eff98SPoul-Henning Kamp 		printf(" ... %d more", mdio.md_pad[0] - unit);
270e39eff98SPoul-Henning Kamp 	printf("\n");
271174b5e9aSPoul-Henning Kamp 	return (0);
272174b5e9aSPoul-Henning Kamp }
273174b5e9aSPoul-Henning Kamp 
274b830359bSPawel Jakub Dawidek static void
275b830359bSPawel Jakub Dawidek prthumanval(int64_t bytes)
276b830359bSPawel Jakub Dawidek {
277b830359bSPawel Jakub Dawidek 	char buf[6];
278b830359bSPawel Jakub Dawidek 
279b830359bSPawel Jakub Dawidek 	humanize_number(buf, sizeof(buf) - (bytes < 0 ? 0 : 1),
280b830359bSPawel Jakub Dawidek 	    bytes, "", HN_AUTOSCALE, HN_B | HN_NOSPACE | HN_DECIMAL);
281b830359bSPawel Jakub Dawidek 	(void)printf("%6s", buf);
282b830359bSPawel Jakub Dawidek }
283b830359bSPawel Jakub Dawidek 
284174b5e9aSPoul-Henning Kamp int
285174b5e9aSPoul-Henning Kamp query(const int fd, const int unit)
286174b5e9aSPoul-Henning Kamp {
287174b5e9aSPoul-Henning Kamp 
28853d745bcSDima Dorfman 	mdio.md_version = MDIOVERSION;
289174b5e9aSPoul-Henning Kamp 	mdio.md_unit = unit;
290174b5e9aSPoul-Henning Kamp 
291174b5e9aSPoul-Henning Kamp 	if (ioctl(fd, MDIOCQUERY, &mdio) < 0)
292174b5e9aSPoul-Henning Kamp 		err(1, "ioctl(/dev/%s)", MDCTL_NAME);
293174b5e9aSPoul-Henning Kamp 
294b830359bSPawel Jakub Dawidek 	(void)printf("%s%d\t", MD_NAME, mdio.md_unit);
295174b5e9aSPoul-Henning Kamp 	switch (mdio.md_type) {
296174b5e9aSPoul-Henning Kamp 	case MD_MALLOC:
297b830359bSPawel Jakub Dawidek 		(void)printf("malloc");
298174b5e9aSPoul-Henning Kamp 		break;
299174b5e9aSPoul-Henning Kamp 	case MD_PRELOAD:
300b830359bSPawel Jakub Dawidek 		(void)printf("preload");
301174b5e9aSPoul-Henning Kamp 		break;
302174b5e9aSPoul-Henning Kamp 	case MD_SWAP:
303b830359bSPawel Jakub Dawidek 		(void)printf("swap");
304174b5e9aSPoul-Henning Kamp 		break;
305174b5e9aSPoul-Henning Kamp 	case MD_VNODE:
306b830359bSPawel Jakub Dawidek 		(void)printf("vnode");
307174b5e9aSPoul-Henning Kamp 		break;
308174b5e9aSPoul-Henning Kamp 	}
309b830359bSPawel Jakub Dawidek 	printf("\t");
310b830359bSPawel Jakub Dawidek 	prthumanval(mdio.md_mediasize);
311b830359bSPawel Jakub Dawidek 	printf("\n");
312174b5e9aSPoul-Henning Kamp 
31370d586c0SPoul-Henning Kamp 	return (0);
31470d586c0SPoul-Henning Kamp }
31570d586c0SPoul-Henning Kamp 
31657e9624eSPoul-Henning Kamp void
31757e9624eSPoul-Henning Kamp mdmaybeload(void)
31857e9624eSPoul-Henning Kamp {
31957e9624eSPoul-Henning Kamp         struct module_stat mstat;
32057e9624eSPoul-Henning Kamp         int fileid, modid;
32110b0e058SDima Dorfman         const char *name;
32257e9624eSPoul-Henning Kamp 	char *cp;
32357e9624eSPoul-Henning Kamp 
324a246f097SJohn-Mark Gurney 	name = MD_MODNAME;
32557e9624eSPoul-Henning Kamp         /* scan files in kernel */
32657e9624eSPoul-Henning Kamp         mstat.version = sizeof(struct module_stat);
32757e9624eSPoul-Henning Kamp         for (fileid = kldnext(0); fileid > 0; fileid = kldnext(fileid)) {
32857e9624eSPoul-Henning Kamp                 /* scan modules in file */
32957e9624eSPoul-Henning Kamp                 for (modid = kldfirstmod(fileid); modid > 0;
33057e9624eSPoul-Henning Kamp                      modid = modfnext(modid)) {
33157e9624eSPoul-Henning Kamp                         if (modstat(modid, &mstat) < 0)
33257e9624eSPoul-Henning Kamp                                 continue;
33357e9624eSPoul-Henning Kamp                         /* strip bus name if present */
33457e9624eSPoul-Henning Kamp                         if ((cp = strchr(mstat.name, '/')) != NULL) {
33557e9624eSPoul-Henning Kamp                                 cp++;
33657e9624eSPoul-Henning Kamp                         } else {
33757e9624eSPoul-Henning Kamp                                 cp = mstat.name;
33857e9624eSPoul-Henning Kamp                         }
33957e9624eSPoul-Henning Kamp                         /* already loaded? */
34057e9624eSPoul-Henning Kamp                         if (!strcmp(name, cp))
34157e9624eSPoul-Henning Kamp                                 return;
34257e9624eSPoul-Henning Kamp                 }
34357e9624eSPoul-Henning Kamp         }
34457e9624eSPoul-Henning Kamp         /* not present, we should try to load it */
34557e9624eSPoul-Henning Kamp         kldload(name);
34657e9624eSPoul-Henning Kamp }
34757e9624eSPoul-Henning Kamp 
348