xref: /freebsd/sys/dev/md/md.c (revision 95f1a897f9ef27e6213ceed7723eb3c22af47eab)
100a6a3c6SPoul-Henning Kamp /*
200a6a3c6SPoul-Henning Kamp  * ----------------------------------------------------------------------------
300a6a3c6SPoul-Henning Kamp  * "THE BEER-WARE LICENSE" (Revision 42):
400a6a3c6SPoul-Henning Kamp  * <phk@FreeBSD.ORG> wrote this file.  As long as you retain this notice you
500a6a3c6SPoul-Henning Kamp  * can do whatever you want with this stuff. If we meet some day, and you think
600a6a3c6SPoul-Henning Kamp  * this stuff is worth it, you can buy me a beer in return.   Poul-Henning Kamp
700a6a3c6SPoul-Henning Kamp  * ----------------------------------------------------------------------------
800a6a3c6SPoul-Henning Kamp  *
900a6a3c6SPoul-Henning Kamp  * $FreeBSD$
1000a6a3c6SPoul-Henning Kamp  *
1100a6a3c6SPoul-Henning Kamp  */
1200a6a3c6SPoul-Henning Kamp 
1300a6a3c6SPoul-Henning Kamp #include <sys/param.h>
1400a6a3c6SPoul-Henning Kamp #include <sys/systm.h>
1500a6a3c6SPoul-Henning Kamp #include <sys/buf.h>
1600a6a3c6SPoul-Henning Kamp #include <sys/conf.h>
1700a6a3c6SPoul-Henning Kamp #include <sys/devicestat.h>
1827068b01SBrian Feldman #include <sys/disk.h>
1927068b01SBrian Feldman #include <sys/kernel.h>
2027068b01SBrian Feldman #include <sys/malloc.h>
2127068b01SBrian Feldman #include <sys/sysctl.h>
2295f1a897SPoul-Henning Kamp #include <sys/linker.h>
2300a6a3c6SPoul-Henning Kamp 
2433edfabeSPoul-Henning Kamp #ifndef MDNSECT
2533edfabeSPoul-Henning Kamp #define MDNSECT (10000 * 2)
2633edfabeSPoul-Henning Kamp #endif
2733edfabeSPoul-Henning Kamp 
2800a6a3c6SPoul-Henning Kamp MALLOC_DEFINE(M_MD, "MD disk", "Memory Disk");
2900a6a3c6SPoul-Henning Kamp MALLOC_DEFINE(M_MDSECT, "MD sectors", "Memory Disk Sectors");
3000a6a3c6SPoul-Henning Kamp 
3100a6a3c6SPoul-Henning Kamp static int md_debug = 0;
3200a6a3c6SPoul-Henning Kamp SYSCTL_INT(_debug, OID_AUTO, mddebug, CTLFLAG_RW, &md_debug, 0, "");
3300a6a3c6SPoul-Henning Kamp 
3400a6a3c6SPoul-Henning Kamp #define CDEV_MAJOR	95
3500a6a3c6SPoul-Henning Kamp #define BDEV_MAJOR	22
3600a6a3c6SPoul-Henning Kamp 
3700a6a3c6SPoul-Henning Kamp static d_strategy_t mdstrategy;
3800a6a3c6SPoul-Henning Kamp static d_open_t mdopen;
3900a6a3c6SPoul-Henning Kamp static d_ioctl_t mdioctl;
4000a6a3c6SPoul-Henning Kamp 
4100a6a3c6SPoul-Henning Kamp static struct cdevsw md_cdevsw = {
4200a6a3c6SPoul-Henning Kamp         /* open */      mdopen,
4300a6a3c6SPoul-Henning Kamp         /* close */     nullclose,
4400a6a3c6SPoul-Henning Kamp         /* read */      physread,
4500a6a3c6SPoul-Henning Kamp         /* write */     physwrite,
4600a6a3c6SPoul-Henning Kamp         /* ioctl */     mdioctl,
4700a6a3c6SPoul-Henning Kamp         /* poll */      nopoll,
4800a6a3c6SPoul-Henning Kamp         /* mmap */      nommap,
4900a6a3c6SPoul-Henning Kamp         /* strategy */  mdstrategy,
5000a6a3c6SPoul-Henning Kamp         /* name */      "md",
5100a6a3c6SPoul-Henning Kamp         /* maj */       CDEV_MAJOR,
5200a6a3c6SPoul-Henning Kamp         /* dump */      nodump,
5300a6a3c6SPoul-Henning Kamp         /* psize */     nopsize,
5400a6a3c6SPoul-Henning Kamp         /* flags */     D_DISK | D_CANFREE,
5500a6a3c6SPoul-Henning Kamp         /* bmaj */      BDEV_MAJOR
5600a6a3c6SPoul-Henning Kamp };
5700a6a3c6SPoul-Henning Kamp static struct cdevsw mddisk_cdevsw;
5800a6a3c6SPoul-Henning Kamp 
5900a6a3c6SPoul-Henning Kamp struct md_s {
6000a6a3c6SPoul-Henning Kamp 	int unit;
6100a6a3c6SPoul-Henning Kamp 	struct devstat stats;
6200a6a3c6SPoul-Henning Kamp 	struct buf_queue_head buf_queue;
6300a6a3c6SPoul-Henning Kamp 	struct disk disk;
6400a6a3c6SPoul-Henning Kamp 	dev_t dev;
6595f1a897SPoul-Henning Kamp 	int busy;
6695f1a897SPoul-Henning Kamp 	enum {MD_MALLOC, MD_PRELOAD} type;
6700a6a3c6SPoul-Henning Kamp 	unsigned nsect;
6895f1a897SPoul-Henning Kamp 
6995f1a897SPoul-Henning Kamp 	/* MD_MALLOC related fields */
7000a6a3c6SPoul-Henning Kamp 	unsigned nsecp;
7100a6a3c6SPoul-Henning Kamp 	u_char **secp;
7200a6a3c6SPoul-Henning Kamp 
7395f1a897SPoul-Henning Kamp 	/* MD_PRELOAD related fields */
7495f1a897SPoul-Henning Kamp 	u_char *pl_ptr;
7595f1a897SPoul-Henning Kamp 	unsigned pl_len;
7600a6a3c6SPoul-Henning Kamp };
7700a6a3c6SPoul-Henning Kamp 
7800a6a3c6SPoul-Henning Kamp static int mdunits;
7900a6a3c6SPoul-Henning Kamp 
8000a6a3c6SPoul-Henning Kamp static int
8100a6a3c6SPoul-Henning Kamp mdopen(dev_t dev, int flag, int fmt, struct proc *p)
8200a6a3c6SPoul-Henning Kamp {
8300a6a3c6SPoul-Henning Kamp 	struct md_s *sc;
8400a6a3c6SPoul-Henning Kamp 	struct disklabel *dl;
8500a6a3c6SPoul-Henning Kamp 
8600a6a3c6SPoul-Henning Kamp 	if (md_debug)
8700a6a3c6SPoul-Henning Kamp 		printf("mdopen(%s %x %x %p)\n",
8800a6a3c6SPoul-Henning Kamp 			devtoname(dev), flag, fmt, p);
8900a6a3c6SPoul-Henning Kamp 
9000a6a3c6SPoul-Henning Kamp 	sc = dev->si_drv1;
9100a6a3c6SPoul-Henning Kamp 
9200a6a3c6SPoul-Henning Kamp 	dl = &sc->disk.d_label;
9300a6a3c6SPoul-Henning Kamp 	bzero(dl, sizeof(*dl));
9400a6a3c6SPoul-Henning Kamp 	dl->d_secsize = DEV_BSIZE;
9500a6a3c6SPoul-Henning Kamp 	dl->d_nsectors = 1024;
9600a6a3c6SPoul-Henning Kamp 	dl->d_ntracks = 1;
9795f1a897SPoul-Henning Kamp 	dl->d_secpercyl = dl->d_nsectors * dl->d_ntracks;
9800a6a3c6SPoul-Henning Kamp 	dl->d_secperunit = sc->nsect;
9900a6a3c6SPoul-Henning Kamp 	dl->d_ncylinders = dl->d_secperunit / dl->d_secpercyl;
10000a6a3c6SPoul-Henning Kamp 	return (0);
10100a6a3c6SPoul-Henning Kamp }
10200a6a3c6SPoul-Henning Kamp 
10300a6a3c6SPoul-Henning Kamp static int
10400a6a3c6SPoul-Henning Kamp mdioctl(dev_t dev, u_long cmd, caddr_t addr, int flags, struct proc *p)
10500a6a3c6SPoul-Henning Kamp {
10600a6a3c6SPoul-Henning Kamp 
10700a6a3c6SPoul-Henning Kamp 	if (md_debug)
10800a6a3c6SPoul-Henning Kamp 		printf("mdioctl(%s %lx %p %x %p)\n",
10900a6a3c6SPoul-Henning Kamp 			devtoname(dev), cmd, addr, flags, p);
11000a6a3c6SPoul-Henning Kamp 
11100a6a3c6SPoul-Henning Kamp 	return (ENOIOCTL);
11200a6a3c6SPoul-Henning Kamp }
11300a6a3c6SPoul-Henning Kamp 
11400a6a3c6SPoul-Henning Kamp static void
11500a6a3c6SPoul-Henning Kamp mdstrategy(struct buf *bp)
11600a6a3c6SPoul-Henning Kamp {
11700a6a3c6SPoul-Henning Kamp 	int s, i;
11800a6a3c6SPoul-Henning Kamp 	struct md_s *sc;
11900a6a3c6SPoul-Henning Kamp 	devstat_trans_flags dop;
12000a6a3c6SPoul-Henning Kamp 	u_char *secp, **secpp, *dst;
12100a6a3c6SPoul-Henning Kamp 	unsigned secno, nsec, secval, uc;
12200a6a3c6SPoul-Henning Kamp 
12300a6a3c6SPoul-Henning Kamp 	if (md_debug > 1)
12400a6a3c6SPoul-Henning Kamp 		printf("mdstrategy(%p) %s %lx, %d, %ld, %p)\n",
12500a6a3c6SPoul-Henning Kamp 		    bp, devtoname(bp->b_dev), bp->b_flags, bp->b_blkno,
12600a6a3c6SPoul-Henning Kamp 		    bp->b_bcount / DEV_BSIZE, bp->b_data);
12700a6a3c6SPoul-Henning Kamp 
12800a6a3c6SPoul-Henning Kamp 	sc = bp->b_dev->si_drv1;
12900a6a3c6SPoul-Henning Kamp 
13000a6a3c6SPoul-Henning Kamp 	s = splbio();
13100a6a3c6SPoul-Henning Kamp 
13200a6a3c6SPoul-Henning Kamp 	bufqdisksort(&sc->buf_queue, bp);
13300a6a3c6SPoul-Henning Kamp 
13400a6a3c6SPoul-Henning Kamp 	if (sc->busy) {
13500a6a3c6SPoul-Henning Kamp 		splx(s);
13600a6a3c6SPoul-Henning Kamp 		return;
13700a6a3c6SPoul-Henning Kamp 	}
13800a6a3c6SPoul-Henning Kamp 
13900a6a3c6SPoul-Henning Kamp 	sc->busy++;
14000a6a3c6SPoul-Henning Kamp 
14100a6a3c6SPoul-Henning Kamp 	while (1) {
14200a6a3c6SPoul-Henning Kamp 		bp = bufq_first(&sc->buf_queue);
14300a6a3c6SPoul-Henning Kamp 		if (bp)
14400a6a3c6SPoul-Henning Kamp 			bufq_remove(&sc->buf_queue, bp);
14500a6a3c6SPoul-Henning Kamp 		splx(s);
14600a6a3c6SPoul-Henning Kamp 		if (!bp)
14700a6a3c6SPoul-Henning Kamp 			break;
14800a6a3c6SPoul-Henning Kamp 
14900a6a3c6SPoul-Henning Kamp 		devstat_start_transaction(&sc->stats);
15000a6a3c6SPoul-Henning Kamp 
15100a6a3c6SPoul-Henning Kamp 		if (bp->b_flags & B_FREEBUF)
15200a6a3c6SPoul-Henning Kamp 			dop = DEVSTAT_NO_DATA;
15300a6a3c6SPoul-Henning Kamp 		else if (bp->b_flags & B_READ)
15400a6a3c6SPoul-Henning Kamp 			dop = DEVSTAT_READ;
15500a6a3c6SPoul-Henning Kamp 		else
15600a6a3c6SPoul-Henning Kamp 			dop = DEVSTAT_WRITE;
15700a6a3c6SPoul-Henning Kamp 
15895f1a897SPoul-Henning Kamp 		if (sc->type == MD_MALLOC) {
15900a6a3c6SPoul-Henning Kamp 			nsec = bp->b_bcount / DEV_BSIZE;
16000a6a3c6SPoul-Henning Kamp 			secno = bp->b_pblkno;
16100a6a3c6SPoul-Henning Kamp 			dst = bp->b_data;
16200a6a3c6SPoul-Henning Kamp 			while (nsec--) {
16300a6a3c6SPoul-Henning Kamp 
16400a6a3c6SPoul-Henning Kamp 				if (secno < sc->nsecp) {
16500a6a3c6SPoul-Henning Kamp 					secpp = &sc->secp[secno];
16633edfabeSPoul-Henning Kamp 					if ((u_int)*secpp > 255) {
16700a6a3c6SPoul-Henning Kamp 						secp = *secpp;
16800a6a3c6SPoul-Henning Kamp 						secval = 0;
16900a6a3c6SPoul-Henning Kamp 					} else {
17000a6a3c6SPoul-Henning Kamp 						secp = 0;
17133edfabeSPoul-Henning Kamp 						secval = (u_int) *secpp;
17200a6a3c6SPoul-Henning Kamp 					}
17300a6a3c6SPoul-Henning Kamp 				} else {
17400a6a3c6SPoul-Henning Kamp 					secpp = 0;
17500a6a3c6SPoul-Henning Kamp 					secp = 0;
17600a6a3c6SPoul-Henning Kamp 					secval = 0;
17700a6a3c6SPoul-Henning Kamp 				}
17833edfabeSPoul-Henning Kamp 				if (md_debug > 2)
17927068b01SBrian Feldman 					printf("%lx %p %p %d\n", bp->b_flags, secpp, secp, secval);
18000a6a3c6SPoul-Henning Kamp 
18100a6a3c6SPoul-Henning Kamp 				if (bp->b_flags & B_FREEBUF) {
18200a6a3c6SPoul-Henning Kamp 					if (secpp) {
18300a6a3c6SPoul-Henning Kamp 						if (secp)
18400a6a3c6SPoul-Henning Kamp 							FREE(secp, M_MDSECT);
18500a6a3c6SPoul-Henning Kamp 						*secpp = 0;
18600a6a3c6SPoul-Henning Kamp 					}
18700a6a3c6SPoul-Henning Kamp 				} else if (bp->b_flags & B_READ) {
18800a6a3c6SPoul-Henning Kamp 					if (secp) {
18900a6a3c6SPoul-Henning Kamp 						bcopy(secp, dst, DEV_BSIZE);
19000a6a3c6SPoul-Henning Kamp 					} else if (secval) {
19100a6a3c6SPoul-Henning Kamp 						for (i = 0; i < DEV_BSIZE; i++)
19200a6a3c6SPoul-Henning Kamp 							dst[i] = secval;
19300a6a3c6SPoul-Henning Kamp 					} else {
19400a6a3c6SPoul-Henning Kamp 						bzero(dst, DEV_BSIZE);
19500a6a3c6SPoul-Henning Kamp 					}
19600a6a3c6SPoul-Henning Kamp 				} else {
19700a6a3c6SPoul-Henning Kamp 					uc = dst[0];
19800a6a3c6SPoul-Henning Kamp 					for (i = 1; i < DEV_BSIZE; i++)
19900a6a3c6SPoul-Henning Kamp 						if (dst[i] != uc)
20000a6a3c6SPoul-Henning Kamp 							break;
20100a6a3c6SPoul-Henning Kamp 					if (i == DEV_BSIZE && !uc) {
20200a6a3c6SPoul-Henning Kamp 						if (secp)
20300a6a3c6SPoul-Henning Kamp 							FREE(secp, M_MDSECT);
20400a6a3c6SPoul-Henning Kamp 						if (secpp)
20500a6a3c6SPoul-Henning Kamp 							*secpp = (u_char *)uc;
20600a6a3c6SPoul-Henning Kamp 					} else {
20700a6a3c6SPoul-Henning Kamp 						if (!secpp) {
20800a6a3c6SPoul-Henning Kamp 							MALLOC(secpp, u_char **, (secno + nsec + 1) * sizeof(u_char *), M_MD, M_WAITOK);
20900a6a3c6SPoul-Henning Kamp 							bzero(secpp, (secno + nsec + 1) * sizeof(u_char *));
21000a6a3c6SPoul-Henning Kamp 							bcopy(sc->secp, secpp, sc->nsecp * sizeof(u_char *));
21100a6a3c6SPoul-Henning Kamp 							FREE(sc->secp, M_MD);
21200a6a3c6SPoul-Henning Kamp 							sc->secp = secpp;
21300a6a3c6SPoul-Henning Kamp 							sc->nsecp = secno + nsec + 1;
21400a6a3c6SPoul-Henning Kamp 							secpp = &sc->secp[secno];
21500a6a3c6SPoul-Henning Kamp 						}
21600a6a3c6SPoul-Henning Kamp 						if (i == DEV_BSIZE) {
21700a6a3c6SPoul-Henning Kamp 							if (secp)
21800a6a3c6SPoul-Henning Kamp 								FREE(secp, M_MDSECT);
21900a6a3c6SPoul-Henning Kamp 							*secpp = (u_char *)uc;
22000a6a3c6SPoul-Henning Kamp 						} else {
22100a6a3c6SPoul-Henning Kamp 							if (!secp)
22200a6a3c6SPoul-Henning Kamp 								MALLOC(secp, u_char *, DEV_BSIZE, M_MDSECT, M_WAITOK);
22300a6a3c6SPoul-Henning Kamp 							bcopy(dst, secp, DEV_BSIZE);
22400a6a3c6SPoul-Henning Kamp 
22500a6a3c6SPoul-Henning Kamp 							*secpp = secp;
22600a6a3c6SPoul-Henning Kamp 						}
22700a6a3c6SPoul-Henning Kamp 					}
22800a6a3c6SPoul-Henning Kamp 				}
22900a6a3c6SPoul-Henning Kamp 				secno++;
23000a6a3c6SPoul-Henning Kamp 				dst += DEV_BSIZE;
23100a6a3c6SPoul-Henning Kamp 			}
23295f1a897SPoul-Henning Kamp 		} else {
23395f1a897SPoul-Henning Kamp 			if (bp->b_flags & B_FREEBUF) {
23495f1a897SPoul-Henning Kamp 				/* nothing */
23595f1a897SPoul-Henning Kamp 			} else if (bp->b_flags & B_READ) {
23695f1a897SPoul-Henning Kamp 				bcopy(sc->pl_ptr + (secno << DEV_BSHIFT), bp->b_data, bp->b_bcount);
23795f1a897SPoul-Henning Kamp 			} else {
23895f1a897SPoul-Henning Kamp 				bcopy(bp->b_data, sc->pl_ptr + (secno << DEV_BSHIFT), bp->b_bcount);
23995f1a897SPoul-Henning Kamp 			}
24095f1a897SPoul-Henning Kamp 		}
24100a6a3c6SPoul-Henning Kamp 
24200a6a3c6SPoul-Henning Kamp 		bp->b_resid = 0;
24333edfabeSPoul-Henning Kamp 		devstat_end_transaction_buf(&sc->stats, bp);
24400a6a3c6SPoul-Henning Kamp 		biodone(bp);
24500a6a3c6SPoul-Henning Kamp 		s = splbio();
24600a6a3c6SPoul-Henning Kamp 	}
24700a6a3c6SPoul-Henning Kamp 	sc->busy = 0;
24800a6a3c6SPoul-Henning Kamp 	return;
24900a6a3c6SPoul-Henning Kamp }
25000a6a3c6SPoul-Henning Kamp 
25195f1a897SPoul-Henning Kamp static void
25295f1a897SPoul-Henning Kamp mdcreate_preload(u_char *image, unsigned length)
25300a6a3c6SPoul-Henning Kamp {
25400a6a3c6SPoul-Henning Kamp 	struct md_s *sc;
25500a6a3c6SPoul-Henning Kamp 
25600a6a3c6SPoul-Henning Kamp 	MALLOC(sc, struct md_s *,sizeof(*sc), M_MD, M_WAITOK);
25700a6a3c6SPoul-Henning Kamp 	bzero(sc, sizeof(*sc));
25800a6a3c6SPoul-Henning Kamp 	sc->unit = mdunits++;
25995f1a897SPoul-Henning Kamp 	sc->type = MD_PRELOAD;
26095f1a897SPoul-Henning Kamp 	bufq_init(&sc->buf_queue);
26195f1a897SPoul-Henning Kamp 	devstat_add_entry(&sc->stats, "md", sc->unit, DEV_BSIZE,
26295f1a897SPoul-Henning Kamp 		DEVSTAT_NO_ORDERED_TAGS,
26395f1a897SPoul-Henning Kamp 		DEVSTAT_TYPE_DIRECT | DEVSTAT_TYPE_IF_OTHER, 0x190);
26495f1a897SPoul-Henning Kamp 	sc->dev = disk_create(sc->unit, &sc->disk, 0,
26595f1a897SPoul-Henning Kamp 	    &md_cdevsw, &mddisk_cdevsw);
26695f1a897SPoul-Henning Kamp 	sc->dev->si_drv1 = sc;
26795f1a897SPoul-Henning Kamp 	sc->nsect = length / DEV_BSIZE;
26895f1a897SPoul-Henning Kamp 	sc->pl_ptr = image;
26995f1a897SPoul-Henning Kamp 	sc->pl_len = length;
27095f1a897SPoul-Henning Kamp }
27195f1a897SPoul-Henning Kamp 
27295f1a897SPoul-Henning Kamp static void
27395f1a897SPoul-Henning Kamp mdcreate_malloc(void)
27495f1a897SPoul-Henning Kamp {
27595f1a897SPoul-Henning Kamp 	struct md_s *sc;
27695f1a897SPoul-Henning Kamp 
27795f1a897SPoul-Henning Kamp 	MALLOC(sc, struct md_s *,sizeof(*sc), M_MD, M_WAITOK);
27895f1a897SPoul-Henning Kamp 	bzero(sc, sizeof(*sc));
27995f1a897SPoul-Henning Kamp 	sc->unit = mdunits++;
28095f1a897SPoul-Henning Kamp 	sc->type = MD_MALLOC;
28100a6a3c6SPoul-Henning Kamp 
28200a6a3c6SPoul-Henning Kamp 	bufq_init(&sc->buf_queue);
28300a6a3c6SPoul-Henning Kamp 
28400a6a3c6SPoul-Henning Kamp 	devstat_add_entry(&sc->stats, "md", sc->unit, DEV_BSIZE,
28500a6a3c6SPoul-Henning Kamp 		DEVSTAT_NO_ORDERED_TAGS,
28600a6a3c6SPoul-Henning Kamp 		DEVSTAT_TYPE_DIRECT | DEVSTAT_TYPE_IF_OTHER, 0x190);
28700a6a3c6SPoul-Henning Kamp 
28800a6a3c6SPoul-Henning Kamp 	sc->dev = disk_create(sc->unit, &sc->disk, 0,
28900a6a3c6SPoul-Henning Kamp 	    &md_cdevsw, &mddisk_cdevsw);
29000a6a3c6SPoul-Henning Kamp 
29100a6a3c6SPoul-Henning Kamp 	sc->dev->si_drv1 = sc;
29233edfabeSPoul-Henning Kamp 	sc->nsect = MDNSECT;	/* for now */
29300a6a3c6SPoul-Henning Kamp 	MALLOC(sc->secp, u_char **, sizeof(u_char *), M_MD, M_WAITOK);
29400a6a3c6SPoul-Henning Kamp 	bzero(sc->secp, sizeof(u_char *));
29500a6a3c6SPoul-Henning Kamp 	sc->nsecp = 1;
29600a6a3c6SPoul-Henning Kamp }
29700a6a3c6SPoul-Henning Kamp 
29800a6a3c6SPoul-Henning Kamp static void
29900a6a3c6SPoul-Henning Kamp md_drvinit(void *unused)
30000a6a3c6SPoul-Henning Kamp {
30100a6a3c6SPoul-Henning Kamp 
30295f1a897SPoul-Henning Kamp 	caddr_t mod;
30395f1a897SPoul-Henning Kamp 	caddr_t c;
30495f1a897SPoul-Henning Kamp 	u_char *ptr, *name, *type;
30595f1a897SPoul-Henning Kamp 	unsigned len;
30695f1a897SPoul-Henning Kamp 
30795f1a897SPoul-Henning Kamp 	mod = NULL;
30895f1a897SPoul-Henning Kamp 	while ((mod = preload_search_next_name(mod)) != NULL) {
30995f1a897SPoul-Henning Kamp 		name = (char *)preload_search_info(mod, MODINFO_NAME);
31095f1a897SPoul-Henning Kamp 		type = (char *)preload_search_info(mod, MODINFO_TYPE);
31195f1a897SPoul-Henning Kamp 		if (name == NULL)
31295f1a897SPoul-Henning Kamp 			continue;
31395f1a897SPoul-Henning Kamp 		if (type == NULL)
31495f1a897SPoul-Henning Kamp 			continue;
31595f1a897SPoul-Henning Kamp 		if (strcmp(type, "md_image"))
31695f1a897SPoul-Henning Kamp 			continue;
31795f1a897SPoul-Henning Kamp 		c = preload_search_info(mod, MODINFO_ADDR);
31895f1a897SPoul-Henning Kamp 		ptr = *(u_char **)c;
31995f1a897SPoul-Henning Kamp 		c = preload_search_info(mod, MODINFO_SIZE);
32095f1a897SPoul-Henning Kamp 		len = *(unsigned *)c;
32195f1a897SPoul-Henning Kamp 		printf("md%d: Preloaded image <%s> %d bytes at %p\n",
32295f1a897SPoul-Henning Kamp 		   mdunits, name, len, ptr);
32395f1a897SPoul-Henning Kamp 		mdcreate_preload(ptr, len);
32495f1a897SPoul-Henning Kamp 	}
32595f1a897SPoul-Henning Kamp 	mdcreate_malloc();
32600a6a3c6SPoul-Henning Kamp }
32700a6a3c6SPoul-Henning Kamp 
32800a6a3c6SPoul-Henning Kamp SYSINIT(ptcdev,SI_SUB_DRIVERS,SI_ORDER_MIDDLE+CDEV_MAJOR, md_drvinit,NULL)
32900a6a3c6SPoul-Henning Kamp 
330