mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-17 15:27:36 +00:00
f7ea2f55d1
There is only cdevsw (which should be renamed in a later edit to deventry or something). cdevsw contains the union of what were in both bdevsw an cdevsw entries. The bdevsw[] table stiff exists and is a second pointer to the cdevsw entry of the device. it's major is in d_bmaj rather than d_maj. some cleanup still to happen (e.g. dsopen now gets two pointers to the same cdevsw struct instead of one to a bdevsw and one to a cdevsw). rawread()/rawwrite() went away as part of this though it's not strictly the same patch, just that it involves all the same lines in the drivers. cdroms no longer have write() entries (they did have rawwrite (?)). tapes no longer have support for bdev operations. Reviewed by: Eivind Eklund and Mike Smith Changes suggested by eivind.
222 lines
5.4 KiB
C
222 lines
5.4 KiB
C
/*-
|
|
* Parts Copyright (c) 1995 Terrence R. Lambert
|
|
* Copyright (c) 1995 Julian R. Elischer
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by Terrence R. Lambert.
|
|
* 4. The name Terrence R. Lambert may not be used to endorse or promote
|
|
* products derived from this software without specific prior written
|
|
* permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY Julian R. Elischer ``AS IS'' AND ANY
|
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE TERRENCE R. LAMBERT BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* $Id: kern_conf.c,v 1.26 1998/06/26 18:14:25 phk Exp $
|
|
*/
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/module.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/vnode.h>
|
|
|
|
#define NUMBDEV 128
|
|
#define NUMCDEV 256
|
|
#define bdevsw_ALLOCSTART (NUMBDEV/2)
|
|
#define cdevsw_ALLOCSTART (NUMCDEV/2)
|
|
|
|
struct cdevsw *bdevsw[NUMBDEV];
|
|
int nblkdev = NUMBDEV;
|
|
struct cdevsw *cdevsw[NUMCDEV];
|
|
int nchrdev = NUMCDEV;
|
|
|
|
/*
|
|
* Routine to convert from character to block device number.
|
|
*
|
|
* A minimal stub routine can always return NODEV.
|
|
*/
|
|
dev_t
|
|
chrtoblk(dev_t dev)
|
|
{
|
|
struct bdevsw *bd;
|
|
struct cdevsw *cd;
|
|
|
|
if(cd = cdevsw[major(dev)]) {
|
|
if (cd->d_bmaj != -1)
|
|
return(makedev(cd->d_bmaj,minor(dev)));
|
|
}
|
|
return(NODEV);
|
|
}
|
|
|
|
/*
|
|
* (re)place an entry in the bdevsw or cdevsw table
|
|
* return the slot used in major(*descrip)
|
|
*/
|
|
static int
|
|
bdevsw_add(dev_t *descrip,
|
|
struct cdevsw *newentry,
|
|
struct cdevsw **oldentry)
|
|
{
|
|
int i ;
|
|
|
|
if ( (int)*descrip == NODEV) { /* auto (0 is valid) */
|
|
/*
|
|
* Search the table looking for a slot...
|
|
*/
|
|
for (i = bdevsw_ALLOCSTART; i < nblkdev; i++)
|
|
if (bdevsw[i] == NULL)
|
|
break; /* found one! */
|
|
/* out of allocable slots? */
|
|
if (i >= nblkdev) {
|
|
return ENFILE;
|
|
}
|
|
} else { /* assign */
|
|
i = major(*descrip);
|
|
if (i < 0 || i >= nblkdev) {
|
|
return EINVAL;
|
|
}
|
|
}
|
|
|
|
/* maybe save old */
|
|
if (oldentry) {
|
|
*oldentry = bdevsw[i];
|
|
}
|
|
if (newentry) {
|
|
newentry->d_bmaj = i;
|
|
}
|
|
/* replace with new */
|
|
bdevsw[i] = newentry;
|
|
|
|
/* done! let them know where we put it */
|
|
*descrip = makedev(i,0);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
cdevsw_add(dev_t *descrip,
|
|
struct cdevsw *newentry,
|
|
struct cdevsw **oldentry)
|
|
{
|
|
int i ;
|
|
|
|
if ( (int)*descrip == NODEV) { /* auto (0 is valid) */
|
|
/*
|
|
* Search the table looking for a slot...
|
|
*/
|
|
for (i = cdevsw_ALLOCSTART; i < nchrdev; i++)
|
|
if (cdevsw[i] == NULL)
|
|
break; /* found one! */
|
|
/* out of allocable slots? */
|
|
if (i >= nchrdev) {
|
|
return ENFILE;
|
|
}
|
|
} else { /* assign */
|
|
i = major(*descrip);
|
|
if (i < 0 || i >= nchrdev) {
|
|
return EINVAL;
|
|
}
|
|
}
|
|
|
|
/* maybe save old */
|
|
if (oldentry) {
|
|
*oldentry = cdevsw[i];
|
|
}
|
|
if (newentry) {
|
|
newentry->d_bmaj = -1;
|
|
newentry->d_maj = i;
|
|
}
|
|
/* replace with new */
|
|
cdevsw[i] = newentry;
|
|
|
|
/* done! let them know where we put it */
|
|
*descrip = makedev(i,0);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* note must call cdevsw_add before bdevsw_add due to d_bmaj hack.
|
|
*/
|
|
void
|
|
cdevsw_add_generic(int bdev, int cdev, struct cdevsw *cdevsw)
|
|
{
|
|
dev_t dev;
|
|
|
|
dev = makedev(cdev, 0);
|
|
cdevsw_add(&dev, cdevsw, NULL);
|
|
dev = makedev(bdev, 0);
|
|
bdevsw_add(&dev, cdevsw, NULL);
|
|
}
|
|
|
|
int
|
|
cdevsw_module_handler(module_t mod, modeventtype_t what, void* arg)
|
|
{
|
|
struct cdevsw_module_data* data = (struct cdevsw_module_data*) arg;
|
|
int error;
|
|
|
|
switch (what) {
|
|
case MOD_LOAD:
|
|
if (error = cdevsw_add(&data->dev, data->cdevsw, NULL))
|
|
return error;
|
|
break;
|
|
|
|
case MOD_UNLOAD:
|
|
if (error = cdevsw_add(&data->dev, NULL, NULL))
|
|
return error;
|
|
break;
|
|
}
|
|
|
|
if (data->chainevh)
|
|
return data->chainevh(mod, what, data->chainarg);
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
bdevsw_module_handler(module_t mod, modeventtype_t what, void* arg)
|
|
{
|
|
struct bdevsw_module_data* data = (struct bdevsw_module_data*) arg;
|
|
int error;
|
|
|
|
switch (what) {
|
|
case MOD_LOAD:
|
|
if (error = cdevsw_add(&data->cdev, data->cdevsw, NULL))
|
|
return error;
|
|
if (error = bdevsw_add(&data->bdev, data->cdevsw, NULL)) {
|
|
cdevsw_add(&data->bdev, NULL, NULL);
|
|
return error;
|
|
}
|
|
break;
|
|
|
|
case MOD_UNLOAD:
|
|
if (error = bdevsw_add(&data->bdev, NULL, NULL))
|
|
return error;
|
|
if (error = cdevsw_add(&data->cdev, NULL, NULL))
|
|
return error;
|
|
break;
|
|
}
|
|
|
|
if (data->chainevh)
|
|
return data->chainevh(mod, what, data->chainarg);
|
|
else
|
|
return 0;
|
|
}
|