2  * Copyright (c) 2004-2006 Apple Computer, Inc. All rights reserved. 
   4  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@ 
   6  * This file contains Original Code and/or Modifications of Original Code 
   7  * as defined in and that are subject to the Apple Public Source License 
   8  * Version 2.0 (the 'License'). You may not use this file except in 
   9  * compliance with the License. The rights granted to you under the License 
  10  * may not be used to create, or enable the creation or redistribution of, 
  11  * unlawful or unlicensed copies of an Apple operating system, or to 
  12  * circumvent, violate, or enable the circumvention or violation of, any 
  13  * terms of an Apple operating system software license agreement. 
  15  * Please obtain a copy of the License at 
  16  * http://www.opensource.apple.com/apsl/ and read it before using this file. 
  18  * The Original Code and all software distributed under the License are 
  19  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  20  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  21  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, 
  22  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
  23  * Please see the License for the specific language governing rights and 
  24  * limitations under the License. 
  26  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@ 
  29  * Copyright (c) 1988 University of Utah. 
  30  * Copyright (c) 1990, 1993 
  31  *      The Regents of the University of California.  All rights reserved. 
  33  * This code is derived from software contributed to Berkeley by 
  34  * the Systems Programming Group of the University of Utah Computer 
  37  * Redistribution and use in source and binary forms, with or without 
  38  * modification, are permitted provided that the following conditions 
  40  * 1. Redistributions of source code must retain the above copyright 
  41  *    notice, this list of conditions and the following disclaimer. 
  42  * 2. Redistributions in binary form must reproduce the above copyright 
  43  *    notice, this list of conditions and the following disclaimer in the 
  44  *    documentation and/or other materials provided with the distribution. 
  45  * 3. All advertising materials mentioning features or use of this software 
  46  *    must display the following acknowledgement: 
  47  *      This product includes software developed by the University of 
  48  *      California, Berkeley and its contributors. 
  49  * 4. Neither the name of the University nor the names of its contributors 
  50  *    may be used to endorse or promote products derived from this software 
  51  *    without specific prior written permission. 
  53  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 
  54  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 
  55  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 
  56  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 
  57  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 
  58  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 
  59  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 
  60  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 
  61  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 
  62  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 
  65  * from: Utah Hdr: vn.c 1.13 94/04/02 
  67  *      from: @(#)vn.c  8.6 (Berkeley) 4/1/94 
  68  * $FreeBSD: src/sys/dev/vn/vn.c,v 1.105.2.4 2001/11/18 07:11:00 dillon Exp $ 
  74  * Block interface to a ramdisk.   
  78 #include <sys/param.h> 
  79 #include <sys/kernel.h> 
  80 #include <sys/mount.h> 
  81 #include <sys/namei.h> 
  84 #include <sys/malloc.h> 
  85 #include <sys/mount.h> 
  86 #include <sys/fcntl.h> 
  91 #include <sys/uio_internal.h> 
  92 #include <libkern/libkern.h> 
  95 #include <vm/vm_pager.h> 
  96 #include <mach/memory_object_types.h> 
  97 #include <kern/debug.h> 
  99 #include <miscfs/devfs/devfs.h> 
 102 void            mdevinit(int the_cnt
); 
 104 static open_close_fcn_t mdevopen
; 
 105 static open_close_fcn_t mdevclose
; 
 106 static psize_fcn_t              mdevsize
; 
 107 static strategy_fcn_t   mdevstrategy
; 
 108 static int                              mdevbioctl(dev_t dev
, u_long cmd
, caddr_t data
, int flag
, struct proc 
*p
); 
 109 static int                              mdevcioctl(dev_t dev
, u_long cmd
, caddr_t data
, int flag
, struct proc 
*p
); 
 110 static int                              mdevrw(dev_t dev
, struct uio 
*uio
, int ioflag
); 
 112 #ifdef CONFIG_MEMDEV_INSECURE 
 114 static char *                   nonspace(char *pos
, char *end
); 
 115 static char *                   getspace(char *pos
, char *end
); 
 116 static char *                   cvtnum(char *pos
, char *end
, unsigned int *num
); 
 118 #endif /* CONFIG_MEMDEV_INSECURE */ 
 120 extern void             bcopy_phys(addr64_t from
, addr64_t to
, vm_size_t bytes
); 
 121 extern void             mapping_set_mod(ppnum_t pn
); 
 122 extern ppnum_t  
pmap_find_phys(pmap_t pmap
, addr64_t va
); 
 127  *      D_DISK          we want to look like a disk 
 128  *      D_CANFREE       We support B_FREEBUF 
 131 static struct bdevsw mdevbdevsw 
= { 
 133         /* close */     mdevclose
, 
 134         /* strategy */  mdevstrategy
, 
 135         /* ioctl */     mdevbioctl
, 
 137         /* psize */     mdevsize
, 
 141 static struct cdevsw mdevcdevsw 
= { 
 143         /* close */     mdevclose
, 
 146         /* ioctl */     mdevcioctl
, 
 148         /* reset */     eno_reset
, 
 150         /* select */    eno_select
, 
 152         /* strategy */  eno_strat
, 
 159         uint64_t        mdBase
;         /* file size in bytes */ 
 160         uint32_t        mdSize
;         /* file size in bytes */ 
 161         int                     mdFlags
;        /* flags */ 
 162         int                     mdSecsize
;      /* sector size */ 
 163         int                     mdBDev
;         /* Block device number */ 
 164         int                     mdCDev
;         /* Character device number */ 
 170 #define mdInited        0x01    /* This device defined */ 
 171 #define mdRO            0x02    /* This device is read-only */ 
 172 #define mdPhys          0x04    /* This device is in physical memory */ 
 177 static int mdevioctl(dev_t dev
, u_long cmd
, caddr_t data
, int flag
, struct proc 
*p
, int is_char
); 
 178 dev_t 
mdevadd(int devid
, ppnum_t base
, unsigned int size
, int phys
); 
 179 dev_t 
mdevlookup(int devid
); 
 180 void mdevremoveall(void); 
 182 static  int mdevclose(__unused dev_t dev
, __unused 
int flags
,  
 183                                           __unused 
int devtype
, __unused 
struct proc 
*p
) { 
 188 static  int mdevopen(dev_t dev
, int flags
, __unused 
int devtype
, __unused 
struct proc 
*p
) { 
 192         devid 
= minor(dev
);                                                                     /* Get minor device number */ 
 194         if (devid 
> 16) return (ENXIO
);                                         /* Not valid */ 
 196         if ((flags 
& FWRITE
) && (mdev
[devid
].mdFlags 
& mdRO
)) return (EACCES
);  /* Currently mounted RO */ 
 201 static int mdevrw(dev_t dev
, struct uio 
*uio
, __unused 
int ioflag
) { 
 205         enum uio_seg    saveflag
; 
 207         devid 
= minor(dev
);                                                                     /* Get minor device number */ 
 209         if (devid 
> 16) return (ENXIO
);                                         /* Not valid */ 
 210         if (!(mdev
[devid
].mdFlags 
& mdInited
))  return (ENXIO
); /* Have we actually been defined yet? */ 
 212         mdata 
= ((addr64_t
)mdev
[devid
].mdBase 
<< 12) + uio
->uio_offset
; /* Point to the area in "file" */ 
 214         saveflag 
= uio
->uio_segflg
;                                                     /* Remember what the request is */ 
 216         if (UIO_IS_USER_SPACE(uio
) == 0 && UIO_IS_SYS_SPACE(uio
) == 0) { 
 217           panic("mdevrw - invalid uio_segflg\n");  
 219 #endif /* LP64_DEBUG */ 
 220         /* Make sure we are moving from physical ram if physical device */ 
 221         if (mdev
[devid
].mdFlags 
& mdPhys
) { 
 222                 if (uio
->uio_segflg 
== UIO_USERSPACE64
)  
 223                         uio
->uio_segflg 
= UIO_PHYS_USERSPACE64
;  
 224                 else if (uio
->uio_segflg 
== UIO_USERSPACE32
) 
 225                         uio
->uio_segflg 
= UIO_PHYS_USERSPACE32
;  
 227                         uio
->uio_segflg 
= UIO_PHYS_USERSPACE
;    
 229         status 
= uiomove64(mdata
, uio_resid(uio
), uio
);         /* Move the data */ 
 230         uio
->uio_segflg 
= saveflag
;                                                     /* Restore the flag */ 
 235 static void mdevstrategy(struct buf 
*bp
) { 
 236         unsigned int left
, lop
, csize
; 
 237         vm_offset_t vaddr
, blkoff
; 
 239         addr64_t paddr
, fvaddr
; 
 242         devid 
= minor(buf_device(bp
));                                                  /* Get minor device number */ 
 244         if ((mdev
[devid
].mdFlags 
& mdInited
) == 0) {            /* Have we actually been defined yet? */ 
 245                 buf_seterror(bp
, ENXIO
); 
 250         buf_setresid(bp
, buf_count(bp
));                                                /* Set byte count */ 
 252         blkoff 
= buf_blkno(bp
) * mdev
[devid
].mdSecsize
;         /* Get offset into file */ 
 255  *      Note that reading past end is an error, but reading at end is an EOF.  For these 
 256  *      we just return with resid == count. 
 259         if (blkoff 
>= (mdev
[devid
].mdSize 
<< 12)) {                     /* Are they trying to read/write at/after end? */ 
 260                 if(blkoff 
!= (mdev
[devid
].mdSize 
<< 12)) {              /* Are we trying to read after EOF? */ 
 261                         buf_seterror(bp
, EINVAL
);                                               /* Yeah, this is an error */ 
 263                 buf_biodone(bp
);                                                                /* Return */ 
 267         if ((blkoff 
+ buf_count(bp
)) > (mdev
[devid
].mdSize 
<< 12)) {            /* Will this read go past end? */ 
 268                 buf_setcount(bp
, ((mdev
[devid
].mdSize 
<< 12) - blkoff
));        /* Yes, trim to max */ 
 271          * make sure the buffer's data area is 
 274         if (buf_map(bp
, (caddr_t 
*)&vaddr
)) 
 275                 panic("ramstrategy: buf_map failed\n"); 
 277         fvaddr 
= (mdev
[devid
].mdBase 
<< 12) + blkoff
;           /* Point to offset into ram disk */ 
 279         if (buf_flags(bp
) & B_READ
) {                                   /* Is this a read? */ 
 280                 if(!(mdev
[devid
].mdFlags 
& mdPhys
)) {                   /* Physical mapped disk? */ 
 281                         bcopy((void *)((uintptr_t)fvaddr
), 
 282                                 (void *)vaddr
, (size_t)buf_count(bp
));  /* This is virtual, just get the data */ 
 285                         left 
= buf_count(bp
);                                           /* Init the amount left to copy */ 
 286                         while(left
) {                                                           /* Go until it is all copied */ 
 288                                 lop 
= min((4096 - (vaddr 
& 4095)), (4096 - (fvaddr 
& 4095)));   /* Get smallest amount left on sink and source */ 
 289                                 csize 
= min(lop
, left
);                                 /* Don't move more than we need to */ 
 291                                 pp 
= pmap_find_phys(kernel_pmap
, (addr64_t
)((uintptr_t)vaddr
)); /* Get the sink physical address */ 
 292                                 if(!pp
) {                                                               /* Not found, what gives? */ 
 293                                         panic("mdevstrategy: sink address %016llX not mapped\n", (addr64_t
)((uintptr_t)vaddr
)); 
 295                                 paddr 
= (addr64_t
)(((addr64_t
)pp 
<< 12) | (addr64_t
)(vaddr 
& 4095));    /* Get actual address */ 
 296                                 bcopy_phys(fvaddr
, paddr
, csize
);               /* Copy this on in */ 
 297                                 mapping_set_mod(paddr 
>> 12);                   /* Make sure we know that it is modified */ 
 299                                 left 
= left 
- csize
;                                    /* Calculate what is left */ 
 300                                 vaddr 
= vaddr 
+ csize
;                                  /* Move to next sink address */ 
 301                                 fvaddr 
= fvaddr 
+ csize
;                                /* Bump to next physical address */ 
 305         else {                                                                                          /* This is a write */ 
 306                 if(!(mdev
[devid
].mdFlags 
& mdPhys
)) {                   /* Physical mapped disk? */ 
 307                         bcopy((void *)vaddr
, (void *)((uintptr_t)fvaddr
), 
 308                                 (size_t)buf_count(bp
));         /* This is virtual, just put the data */ 
 311                         left 
= buf_count(bp
);                                           /* Init the amount left to copy */ 
 312                         while(left
) {                                                           /* Go until it is all copied */ 
 314                                 lop 
= min((4096 - (vaddr 
& 4095)), (4096 - (fvaddr 
& 4095)));   /* Get smallest amount left on sink and source */ 
 315                                 csize 
= min(lop
, left
);                                 /* Don't move more than we need to */ 
 317                                 pp 
= pmap_find_phys(kernel_pmap
, (addr64_t
)((uintptr_t)vaddr
)); /* Get the source physical address */ 
 318                                 if(!pp
) {                                                               /* Not found, what gives? */ 
 319                                         panic("mdevstrategy: source address %016llX not mapped\n", (addr64_t
)((uintptr_t)vaddr
)); 
 321                                 paddr 
= (addr64_t
)(((addr64_t
)pp 
<< 12) | (addr64_t
)(vaddr 
& 4095));    /* Get actual address */ 
 323                                 bcopy_phys(paddr
, fvaddr
, csize
);               /* Move this on out */ 
 325                                 left 
= left 
- csize
;                                    /* Calculate what is left */ 
 326                                 vaddr 
= vaddr 
+ csize
;                                  /* Move to next sink address */ 
 327                                 fvaddr 
= fvaddr 
+ csize
;                                /* Bump to next physical address */ 
 332          * buf_unmap takes care of all the cases 
 333          * it will unmap the buffer from kernel 
 334          * virtual space if that was the state 
 339         buf_setresid(bp
, 0);                                                                    /* Nothing more to do */         
 340         buf_biodone(bp
);                                                                        /* Say we've finished */ 
 343 static int mdevbioctl(dev_t dev
, u_long cmd
, caddr_t data
, int flag
, struct proc 
*p
) { 
 344         return (mdevioctl(dev
, cmd
, data
, flag
, p
, 0)); 
 347 static int mdevcioctl(dev_t dev
, u_long cmd
, caddr_t data
, int flag
, struct proc 
*p
) { 
 348         return (mdevioctl(dev
, cmd
, data
, flag
, p
, 1)); 
 351 static int mdevioctl(dev_t dev
, u_long cmd
, caddr_t data
, __unused 
int flag
,  
 352                                          struct proc 
*p
, int is_char
) { 
 358         devid 
= minor(dev
);                                                                     /* Get minor device number */ 
 360         if (devid 
> 16) return (ENXIO
);                                         /* Not valid */ 
 362         error 
= proc_suser(p
);                  /* Are we superman? */ 
 363         if (error
) return (error
);                                                      /* Nope... */ 
 365         f 
= (u_int32_t
*)data
; 
 366         o 
= (u_int64_t 
*)data
; 
 370                 case DKIOCGETMAXBLOCKCOUNTREAD
: 
 374                 case DKIOCGETMAXBLOCKCOUNTWRITE
: 
 378                 case DKIOCGETMAXSEGMENTCOUNTREAD
: 
 382                 case DKIOCGETMAXSEGMENTCOUNTWRITE
: 
 386                 case DKIOCGETBLOCKSIZE
: 
 387                         *f 
= mdev
[devid
].mdSecsize
; 
 390                 case DKIOCSETBLOCKSIZE
: 
 391                         if (is_char
) return (ENODEV
);                           /* We can only do this for a block */ 
 393                         if (*f 
< DEV_BSIZE
) return (EINVAL
);            /* Too short? */ 
 395                         mdev
[devid
].mdSecsize 
= *f
;                                     /* set the new block size */ 
 398                 case DKIOCISWRITABLE
: 
 402                 case DKIOCGETBLOCKCOUNT32
: 
 403                         if(!(mdev
[devid
].mdFlags 
& mdInited
)) return (ENXIO
); 
 404                         *f 
= ((mdev
[devid
].mdSize 
<< 12) + mdev
[devid
].mdSecsize 
- 1) / mdev
[devid
].mdSecsize
; 
 407                 case DKIOCGETBLOCKCOUNT
: 
 408                         if(!(mdev
[devid
].mdFlags 
& mdInited
)) return (ENXIO
); 
 409                         *o 
= ((mdev
[devid
].mdSize 
<< 12) + mdev
[devid
].mdSecsize 
- 1) / mdev
[devid
].mdSecsize
; 
 420 static  int mdevsize(dev_t dev
) { 
 424         devid 
= minor(dev
);                                                                     /* Get minor device number */ 
 425         if (devid 
> 16) return (ENXIO
);                                         /* Not valid */ 
 427         if ((mdev
[devid
].mdFlags 
& mdInited
) == 0) return(-1);          /* Not inited yet */ 
 429         return(mdev
[devid
].mdSecsize
); 
 432 #include <pexpert/pexpert.h> 
 434 void mdevinit(__unused 
int the_cnt
) { 
 436 #ifdef CONFIG_MEMDEV_INSECURE 
 445         ba 
= PE_boot_args();                                                            /* Get the boot arguments */ 
 446         lp 
= ba 
+ 256;                                                                          /* Point to the end */ 
 448         while(1) {                                                                                      /* Step through, looking for our keywords */ 
 449                 phys 
= 0;                                                                               /* Assume virtual memory device */ 
 450                 ba 
= nonspace(ba
, lp
);                                                  /* Find non-space */ 
 451                 if(ba 
>= lp
) return;                                                    /* We are done if no more... */ 
 452                 if(((ba
[0] != 'v') && (ba
[0] != 'p'))   
 453                   || (ba
[1] != 'm') || (ba
[2] != 'd') || (ba
[4] != '=') 
 454                   || (ba
[3] < '0') || (ba
[3] > 'f')  
 455                   || ((ba
[3] > '9') && (ba
[3] < 'a'))) {                /* Is this of form "vmdx=" or "pmdx=" where x is hex digit? */ 
 457                         ba 
= getspace(ba
, lp
);                                          /* Find next white space or end */ 
 458                         continue;                                                                       /* Start looking for the next one */ 
 461                 if(ba
[0] == 'p') phys 
= 1;                                              /* Set physical memory disk */ 
 463                 devid 
= ba
[3] & 0xF;                                                    /* Assume digit */ 
 464                 if(ba
[3] > '9') devid 
+= 9;                                             /* Adjust for hex digits */ 
 466                 ba 
= &ba
[5];                                                                    /* Step past keyword */ 
 467                 ba 
= cvtnum(ba
, lp
, &base
);                                             /* Convert base of memory disk */ 
 468                 if(ba 
>= lp
) return;                                                    /* Malformed one at the end, leave */ 
 469                 if(ba
[0] != '.') continue;                                              /* If not length separater, try next... */ 
 470                 if(base 
& 0xFFF) continue;                                              /* Only allow page aligned stuff */ 
 472                 ba
++;                                                                                   /* Step past '.' */ 
 473                 ba 
= cvtnum(ba
, lp
, &size
);                                             /* Try to convert it */ 
 474                 if(!size 
|| (size 
& 0xFFF)) continue;                   /* Allow only non-zer page size multiples */ 
 475                 if(ba 
< lp
) {                                                                   /* If we are not at end, check end character */ 
 476                         if((ba
[0] != ' ') && (ba
[0] != 0)) continue;    /* End must be null or space */ 
 479                 dev 
= mdevadd(devid
, base 
>> 12, size 
>> 12, phys
);     /* Go add the device */  
 482 #endif /* CONFIG_MEMDEV_INSECURE */ 
 488 #ifdef CONFIG_MEMDEV_INSECURE 
 490 char *nonspace(char *pos
, char *end
) {                                  /* Find next non-space in string */ 
 492         if(pos 
>= end
) return end
;                                                      /* Don't go past end */ 
 493         if(pos
[0] == 0) return end
;                                                     /* If at null, make end */ 
 495         while(1) {                                                                                      /* Keep going */ 
 496                 if(pos
[0] != ' ') return pos
;                                   /* Leave if we found one */ 
 498                 if(pos 
>= end
) return end
;                                              /* Quit if we run off end */ 
 502 char *getspace(char *pos
, char *end
) {                                  /* Find next non-space in string */ 
 504         while(1) {                                                                                      /* Keep going */ 
 505                 if(pos 
>= end
) return end
;                                              /* Don't go past end */ 
 506                 if(pos
[0] == 0) return end
;                                             /* Leave if we hit null */ 
 507                 if(pos
[0] == ' ') return pos
;                                   /* Leave if we found one */ 
 512 char *cvtnum(char *pos
, char *end
, unsigned int *num
) {         /* Convert to a number */ 
 516         *num 
= 0;                                                                                       /* Set answer to 0 to start */   
 519         if(pos 
>= end
) return end
;                                                      /* Don't go past end */ 
 520         if(pos
[0] == 0) return end
;                                                     /* If at null, make end */ 
 522         if(pos
[0] == '0' && ((pos
[1] == 'x') || (pos
[1] == 'x'))) {     /* A hex constant? */ 
 524                 pos 
+= 2;                                                                               /* Point to the number */ 
 527         while(1) {                                                                                      /* Convert it */ 
 529                 if(pos 
>= end
) return end
;                                              /* Don't go past end */ 
 530                 if(pos
[0] == 0) return end
;                                             /* If at null, make end */ 
 531                 if(pos
[0] < '0') return pos
;                                    /* Leave if non-digit */ 
 532                 dig 
= pos
[0] & 0xF;                                                             /* Extract digit */ 
 533                 if(pos
[0] > '9') {                                                              /* Is it bigger than 9? */ 
 534                         if(rad 
== 10) return pos
;                                       /* Leave if not base 10 */ 
 535                         if(!(((pos
[0] >= 'A') && (pos
[0] <= 'F'))  
 536                           || ((pos
[0] >= 'a') && (pos
[0] <= 'f')))) return pos
; /* Leave if bogus char */ 
 537                          dig 
= dig 
+ 9;                                                         /* Adjust for character */ 
 539                 *num 
= (*num 
* rad
) + dig
;                                              /* Accumulate the number */ 
 544 #endif /* CONFIG_MEMDEV_INSECURE */ 
 546 dev_t 
mdevadd(int devid
, ppnum_t base
, unsigned int size
, int phys
) { 
 553                 for(i 
= 0; i 
< 16; i
++) {                                               /* Search all known memory devices */ 
 554                         if(!(mdev
[i
].mdFlags 
& mdInited
)) {                     /* Is this a free one? */ 
 555                                 if(devid 
< 0)devid 
= i
;                                 /* Remember first free one */ 
 556                                 continue;                                                               /* Skip check */ 
 558                         if(!(((base 
+ size 
-1 ) < mdev
[i
].mdBase
) || ((mdev
[i
].mdBase 
+ mdev
[i
].mdSize 
- 1) < base
))) { /* Is there any overlap? */ 
 559                                 panic("mdevadd: attempt to add overlapping memory device at %08lX-%08lX\n", (long) mdev
[i
].mdBase
, (long) mdev
[i
].mdBase 
+ mdev
[i
].mdSize 
- 1); 
 562                 if(devid 
< 0) {                                                                 /* Do we have free slots? */ 
 563                         panic("mdevadd: attempt to add more than 16 memory devices\n"); 
 567                 if(devid 
>= 16) {                                                               /* Giving us something bogus? */ 
 568                         panic("mdevadd: attempt to explicitly add a bogus memory device: %08X\n", devid
); 
 570                 if(mdev
[devid
].mdFlags 
&mdInited
) {                             /* Already there? */ 
 571                         panic("mdevadd: attempt to explicitly add a previously defined memory device: %08X\n", devid
); 
 575         if(mdevBMajor 
< 0) {                                                            /* Have we gotten a major number yet? */ 
 576                 mdevBMajor 
= bdevsw_add(-1, &mdevbdevsw
);               /* Add to the table and figure out a major number */ 
 577                 if (mdevBMajor 
< 0) { 
 578                         printf("mdevadd: error - bdevsw_add() returned %d\n", mdevBMajor
); 
 583         if(mdevCMajor 
< 0) {                                                            /* Have we gotten a major number yet? */ 
 584                 mdevCMajor 
= cdevsw_add_with_bdev(-1, &mdevcdevsw
, mdevBMajor
);         /* Add to the table and figure out a major number */ 
 585                 if (mdevCMajor 
< 0) { 
 586                         printf("ramdevice_init: error - cdevsw_add() returned %d\n", mdevCMajor
); 
 591         mdev
[devid
].mdBDev 
= makedev(mdevBMajor
, devid
);        /* Get the device number */ 
 592         mdev
[devid
].mdbdevb 
= devfs_make_node(mdev
[devid
].mdBDev
, DEVFS_BLOCK
,  /* Make the node */ 
 593                                                   UID_ROOT
, GID_OPERATOR
,  
 594                                                   0600, "md%d", devid
); 
 595         if (mdev
[devid
].mdbdevb 
== NULL
) {                                      /* Did we make one? */ 
 596                 printf("mdevadd: devfs_make_node for block failed!\n"); 
 597                 return -1;                                                                              /* Nope... */ 
 600         mdev
[devid
].mdCDev 
= makedev(mdevCMajor
, devid
);        /* Get the device number */ 
 601         mdev
[devid
].mdcdevb 
= devfs_make_node(mdev
[devid
].mdCDev
, DEVFS_CHAR
,           /* Make the node */ 
 602                                                   UID_ROOT
, GID_OPERATOR
,  
 603                                                   0600, "rmd%d", devid
); 
 604         if (mdev
[devid
].mdcdevb 
== NULL
) {                                      /* Did we make one? */ 
 605                 printf("mdevadd: devfs_make_node for character failed!\n"); 
 606                 return -1;                                                                              /* Nope... */ 
 609         mdev
[devid
].mdBase 
= base
;                                                      /* Set the base address of ram disk */ 
 610         mdev
[devid
].mdSize 
= size
;                                                      /* Set the length of the ram disk */ 
 611         mdev
[devid
].mdSecsize 
= DEV_BSIZE
;                                      /* Set starting block size */ 
 612         if(phys
) mdev
[devid
].mdFlags 
|= mdPhys
;                         /* Show that we are in physical memory */ 
 613         mdev
[devid
].mdFlags 
|= mdInited
;                                        /* Show we are all set up */ 
 614         printf("Added memory device md%x/rmd%x (%08X/%08X) at %08X for %08X\n",  
 615                 devid
, devid
, mdev
[devid
].mdBDev
, mdev
[devid
].mdCDev
, base 
<< 12, size 
<< 12); 
 616         return mdev
[devid
].mdBDev
; 
 620 dev_t 
mdevlookup(int devid
) { 
 622         if((devid 
< 0) || (devid 
> 15)) return -1;                      /* Filter any bogus requests */ 
 623         if(!(mdev
[devid
].mdFlags 
& mdInited
)) return -1;        /* This one hasn't been defined */ 
 624         return mdev
[devid
].mdBDev
;                                                      /* Return the device number */ 
 627 void mdevremoveall(void) { 
 631         for(i 
= 0; i 
< 16; i
++) { 
 632                 if(!(mdev
[i
].mdFlags 
& mdInited
)) continue;     /* Ignore unused mdevs */ 
 634                 devfs_remove(mdev
[i
].mdbdevb
);                  /* Remove the block device */ 
 635                 devfs_remove(mdev
[i
].mdcdevb
);                  /* Remove the character device */ 
 637                 mdev
[i
].mdBase 
= 0;                             /* Clear the mdev's storage */ 
 639                 mdev
[i
].mdSecsize 
= 0;