Commit a049055c authored by Christoph Hellwig's avatar Christoph Hellwig

[PATCH] get rid of ->finish method for highlevel drivers

the ->finish method is a relicat from the old day were we never had
hotplugging and allowed the driver to do fixups after all busses
had been scanned.  Nowdays only sd and sr actually implement it,
and both only defer actions to there that should actually happen in
->attach.  Change both drivers to move that code into ->attach,
clenaup the Templates to use C99 initializers and get rid of the
methods.

This also cleans up some very crude race-avoidable code in those
drivers, btw..
parent 84a26cf5
...@@ -573,14 +573,6 @@ int scsi_register_host(Scsi_Host_Template *shost_tp) ...@@ -573,14 +573,6 @@ int scsi_register_host(Scsi_Host_Template *shost_tp)
} }
} }
} }
/* This does any final handling that is required. */
for (sdev_tp = scsi_devicelist; sdev_tp;
sdev_tp = sdev_tp->next) {
if (sdev_tp->finish && sdev_tp->nr_dev) {
(*sdev_tp->finish) ();
}
}
} }
return 0; return 0;
......
...@@ -549,7 +549,6 @@ struct Scsi_Device_Template ...@@ -549,7 +549,6 @@ struct Scsi_Device_Template
int (*detect)(Scsi_Device *); /* Returns 1 if we can attach this device */ int (*detect)(Scsi_Device *); /* Returns 1 if we can attach this device */
int (*init)(void); /* Sizes arrays based upon number of devices int (*init)(void); /* Sizes arrays based upon number of devices
* detected */ * detected */
void (*finish)(void); /* Perform initialization after attachment */
int (*attach)(Scsi_Device *); /* Attach devices to arrays */ int (*attach)(Scsi_Device *); /* Attach devices to arrays */
void (*detach)(Scsi_Device *); void (*detach)(Scsi_Device *);
int (*init_command)(Scsi_Cmnd *); /* Used by new queueing code. int (*init_command)(Scsi_Cmnd *); /* Used by new queueing code.
......
...@@ -2034,17 +2034,13 @@ int scsi_register_device(struct Scsi_Device_Template *tpnt) ...@@ -2034,17 +2034,13 @@ int scsi_register_device(struct Scsi_Device_Template *tpnt)
} }
} }
/*
* This does any final handling that is required.
*/
if (tpnt->finish && tpnt->nr_dev)
(*tpnt->finish) ();
MOD_INC_USE_COUNT; MOD_INC_USE_COUNT;
if (out_of_space) { if (out_of_space) {
scsi_unregister_device(tpnt); /* easiest way to clean up?? */ scsi_unregister_device(tpnt); /* easiest way to clean up?? */
return 1; return 1;
} else }
return 0; return 0;
} }
......
...@@ -2012,11 +2012,6 @@ static void scsi_scan_selected_lun(struct Scsi_Host *shost, uint channel, ...@@ -2012,11 +2012,6 @@ static void scsi_scan_selected_lun(struct Scsi_Host *shost, uint channel,
__FUNCTION__); __FUNCTION__);
} }
} }
for (sdt = scsi_devicelist; sdt; sdt = sdt->next)
if (sdt->finish && sdt->nr_dev)
(*sdt->finish) ();
} }
} }
......
...@@ -92,7 +92,6 @@ static int sd_revalidate(struct gendisk *); ...@@ -92,7 +92,6 @@ static int sd_revalidate(struct gendisk *);
static void sd_init_onedisk(Scsi_Disk * sdkp, struct gendisk *disk); static void sd_init_onedisk(Scsi_Disk * sdkp, struct gendisk *disk);
static int sd_init(void); static int sd_init(void);
static void sd_finish(void);
static int sd_attach(Scsi_Device *); static int sd_attach(Scsi_Device *);
static int sd_detect(Scsi_Device *); static int sd_detect(Scsi_Device *);
static void sd_detach(Scsi_Device *); static void sd_detach(Scsi_Device *);
...@@ -103,23 +102,19 @@ static int sd_notifier(struct notifier_block *, unsigned long, void *); ...@@ -103,23 +102,19 @@ static int sd_notifier(struct notifier_block *, unsigned long, void *);
static struct notifier_block sd_notifier_block = {sd_notifier, NULL, 0}; static struct notifier_block sd_notifier_block = {sd_notifier, NULL, 0};
static struct Scsi_Device_Template sd_template = { static struct Scsi_Device_Template sd_template = {
module:THIS_MODULE, .module = THIS_MODULE,
name:"disk", .name = "disk",
tag:"sd", .tag = "sd",
scsi_type:TYPE_DISK, .scsi_type = TYPE_DISK,
major:SCSI_DISK0_MAJOR, .major = SCSI_DISK0_MAJOR,
/* .min_major = SCSI_DISK1_MAJOR,
* Secondary range of majors that this driver handles. .max_major = SCSI_DISK7_MAJOR,
*/ .blk = 1,
min_major:SCSI_DISK1_MAJOR, .detect = sd_detect,
max_major:SCSI_DISK7_MAJOR, .init = sd_init,
blk:1, .attach = sd_attach,
detect:sd_detect, .detach = sd_detach,
init:sd_init, .init_command = sd_init_command,
finish:sd_finish,
attach:sd_attach,
detach:sd_detach,
init_command:sd_init_command,
}; };
static void sd_rw_intr(Scsi_Cmnd * SCpnt); static void sd_rw_intr(Scsi_Cmnd * SCpnt);
...@@ -1290,38 +1285,6 @@ static int sd_init() ...@@ -1290,38 +1285,6 @@ static int sd_init()
return 1; return 1;
} }
/**
* sd_finish - called during driver initialization, after all
* the sd_attach() calls are finished.
*
* Note: this function is invoked from the scsi mid-level.
* This function is not called after driver initialization has completed.
* Specifically later device attachments invoke sd_attach() but not
* this function.
**/
static void sd_finish()
{
int k;
Scsi_Disk * sdkp;
SCSI_LOG_HLQUEUE(3, printk("sd_finish: \n"));
for (k = 0; k < sd_template.dev_max; ++k) {
sdkp = sd_get_sdisk(k);
if (sdkp && (0 == sdkp->capacity) && sdkp->device) {
sd_init_onedisk(sdkp, sd_disks[k]);
if (sdkp->has_been_registered)
continue;
set_capacity(sd_disks[k], sdkp->capacity);
sd_disks[k]->private_data = sdkp;
sd_disks[k]->queue = &sdkp->device->request_queue;
add_disk(sd_disks[k]);
sdkp->has_been_registered = 1;
}
}
return;
}
/** /**
* sd_detect - called at the start of driver initialization, once * sd_detect - called at the start of driver initialization, once
* for each scsi device (not just disks) present. * for each scsi device (not just disks) present.
...@@ -1358,13 +1321,12 @@ static int sd_detect(Scsi_Device * sdp) ...@@ -1358,13 +1321,12 @@ static int sd_detect(Scsi_Device * sdp)
**/ **/
static int sd_attach(Scsi_Device * sdp) static int sd_attach(Scsi_Device * sdp)
{ {
Scsi_Disk *sdkp; Scsi_Disk *sdkp = NULL; /* shut up lame gcc warning */
int dsk_nr; int dsk_nr;
unsigned long iflags; unsigned long iflags;
struct gendisk *gd; struct gendisk *gd;
if ((NULL == sdp) || if ((sdp->type != TYPE_DISK) && (sdp->type != TYPE_MOD))
((sdp->type != TYPE_DISK) && (sdp->type != TYPE_MOD)))
return 0; return 0;
gd = alloc_disk(16); gd = alloc_disk(16);
...@@ -1373,15 +1335,16 @@ static int sd_attach(Scsi_Device * sdp) ...@@ -1373,15 +1335,16 @@ static int sd_attach(Scsi_Device * sdp)
SCSI_LOG_HLQUEUE(3, printk("sd_attach: scsi device: <%d,%d,%d,%d>\n", SCSI_LOG_HLQUEUE(3, printk("sd_attach: scsi device: <%d,%d,%d,%d>\n",
sdp->host->host_no, sdp->channel, sdp->id, sdp->lun)); sdp->host->host_no, sdp->channel, sdp->id, sdp->lun));
if (sd_template.nr_dev >= sd_template.dev_max) { if (sd_template.nr_dev >= sd_template.dev_max) {
sdp->attached--;
printk(KERN_ERR "sd_init: no more room for device\n"); printk(KERN_ERR "sd_init: no more room for device\n");
put_disk(gd); goto out;
return 1;
} }
/* Assume sd_attach is not re-entrant (for time being) */ /*
/* Also think about sd_attach() and sd_detach() running coincidentally. */ * Assume sd_attach is not re-entrant (for time being)
* Also think about sd_attach() and sd_detach() running coincidentally.
*/
write_lock_irqsave(&sd_dsk_arr_lock, iflags); write_lock_irqsave(&sd_dsk_arr_lock, iflags);
for (dsk_nr = 0; dsk_nr < sd_template.dev_max; dsk_nr++) { for (dsk_nr = 0; dsk_nr < sd_template.dev_max; dsk_nr++) {
sdkp = sd_dsk_arr[dsk_nr]; sdkp = sd_dsk_arr[dsk_nr];
...@@ -1393,14 +1356,14 @@ static int sd_attach(Scsi_Device * sdp) ...@@ -1393,14 +1356,14 @@ static int sd_attach(Scsi_Device * sdp)
} }
write_unlock_irqrestore(&sd_dsk_arr_lock, iflags); write_unlock_irqrestore(&sd_dsk_arr_lock, iflags);
if (dsk_nr >= sd_template.dev_max) { if (!sdkp || dsk_nr >= sd_template.dev_max) {
/* panic("scsi_devices corrupt (sd)"); overkill */
printk(KERN_ERR "sd_init: sd_dsk_arr corrupted\n"); printk(KERN_ERR "sd_init: sd_dsk_arr corrupted\n");
put_disk(gd); goto out;
return 1;
} }
sd_init_onedisk(sdkp, gd);
sd_template.nr_dev++; sd_template.nr_dev++;
gd->de = sdp->de; gd->de = sdp->de;
gd->major = SD_MAJOR(dsk_nr>>4); gd->major = SD_MAJOR(dsk_nr>>4);
gd->first_minor = (dsk_nr & 15)<<4; gd->first_minor = (dsk_nr & 15)<<4;
...@@ -1412,11 +1375,23 @@ static int sd_attach(Scsi_Device * sdp) ...@@ -1412,11 +1375,23 @@ static int sd_attach(Scsi_Device * sdp)
gd->flags = sdp->removable ? GENHD_FL_REMOVABLE : 0; gd->flags = sdp->removable ? GENHD_FL_REMOVABLE : 0;
gd->driverfs_dev = &sdp->sdev_driverfs_dev; gd->driverfs_dev = &sdp->sdev_driverfs_dev;
gd->flags |= GENHD_FL_DRIVERFS | GENHD_FL_DEVFS; gd->flags |= GENHD_FL_DRIVERFS | GENHD_FL_DEVFS;
gd->private_data = sdkp;
gd->queue = &sdkp->device->request_queue;
set_capacity(gd, sdkp->capacity);
add_disk(gd);
sd_disks[dsk_nr] = gd; sd_disks[dsk_nr] = gd;
printk(KERN_NOTICE "Attached scsi %sdisk %s at scsi%d, channel %d, " printk(KERN_NOTICE "Attached scsi %sdisk %s at scsi%d, channel %d, "
"id %d, lun %d\n", sdp->removable ? "removable " : "", "id %d, lun %d\n", sdp->removable ? "removable " : "",
gd->disk_name, sdp->host->host_no, sdp->channel, sdp->id, sdp->lun); gd->disk_name, sdp->host->host_no, sdp->channel, sdp->id, sdp->lun);
return 0; return 0;
out:
sdp->attached--;
put_disk(gd);
return 1;
} }
static int sd_revalidate(struct gendisk *disk) static int sd_revalidate(struct gendisk *disk)
...@@ -1472,10 +1447,7 @@ static void sd_detach(Scsi_Device * sdp) ...@@ -1472,10 +1447,7 @@ static void sd_detach(Scsi_Device * sdp)
sdkp->capacity = 0; sdkp->capacity = 0;
/* sdkp->detaching = 1; */ /* sdkp->detaching = 1; */
if (sdkp->has_been_registered) {
sdkp->has_been_registered = 0;
del_gendisk(sd_disks[dsk_nr]); del_gendisk(sd_disks[dsk_nr]);
}
sdp->attached--; sdp->attached--;
sd_template.dev_noticed--; sd_template.dev_noticed--;
sd_template.nr_dev--; sd_template.nr_dev--;
......
...@@ -25,7 +25,6 @@ typedef struct scsi_disk { ...@@ -25,7 +25,6 @@ typedef struct scsi_disk {
Scsi_Device *device; Scsi_Device *device;
unsigned char media_present; unsigned char media_present;
unsigned char write_prot; unsigned char write_prot;
unsigned has_been_registered:1;
unsigned WCE:1; /* state of disk WCE bit */ unsigned WCE:1; /* state of disk WCE bit */
unsigned RCD:1; /* state of disk RCD bit */ unsigned RCD:1; /* state of disk RCD bit */
} Scsi_Disk; } Scsi_Disk;
......
...@@ -63,27 +63,24 @@ MODULE_PARM(xa_test, "i"); /* see sr_ioctl.c */ ...@@ -63,27 +63,24 @@ MODULE_PARM(xa_test, "i"); /* see sr_ioctl.c */
#define SR_TIMEOUT (30 * HZ) #define SR_TIMEOUT (30 * HZ)
static int sr_init(void); static int sr_init(void);
static void sr_finish(void);
static int sr_attach(Scsi_Device *); static int sr_attach(Scsi_Device *);
static int sr_detect(Scsi_Device *); static int sr_detect(Scsi_Device *);
static void sr_detach(Scsi_Device *); static void sr_detach(Scsi_Device *);
static int sr_init_command(Scsi_Cmnd *); static int sr_init_command(Scsi_Cmnd *);
static struct Scsi_Device_Template sr_template = static struct Scsi_Device_Template sr_template = {
{ .module = THIS_MODULE,
module:THIS_MODULE, .name = "cdrom",
name:"cdrom", .tag = "sr",
tag:"sr", .scsi_type = TYPE_ROM,
scsi_type:TYPE_ROM, .major = SCSI_CDROM_MAJOR,
major:SCSI_CDROM_MAJOR, .blk = 1,
blk:1, .detect = sr_detect,
detect:sr_detect, .init = sr_init,
init:sr_init, .attach = sr_attach,
finish:sr_finish, .detach = sr_detach,
attach:sr_attach, .init_command = sr_init_command
detach:sr_detach,
init_command:sr_init_command
}; };
static Scsi_CD *scsi_CDs; static Scsi_CD *scsi_CDs;
...@@ -91,6 +88,7 @@ static Scsi_CD *scsi_CDs; ...@@ -91,6 +88,7 @@ static Scsi_CD *scsi_CDs;
static int sr_open(struct cdrom_device_info *, int); static int sr_open(struct cdrom_device_info *, int);
static void get_sectorsize(Scsi_CD *); static void get_sectorsize(Scsi_CD *);
static void get_capabilities(Scsi_CD *); static void get_capabilities(Scsi_CD *);
static int sr_init_one(Scsi_CD *, int);
static int sr_media_change(struct cdrom_device_info *, int); static int sr_media_change(struct cdrom_device_info *, int);
static int sr_packet(struct cdrom_device_info *, struct cdrom_generic_command *); static int sr_packet(struct cdrom_device_info *, struct cdrom_generic_command *);
...@@ -473,10 +471,9 @@ static int sr_attach(Scsi_Device * SDp) ...@@ -473,10 +471,9 @@ static int sr_attach(Scsi_Device * SDp)
if (SDp->type != TYPE_ROM && SDp->type != TYPE_WORM) if (SDp->type != TYPE_ROM && SDp->type != TYPE_WORM)
return 1; return 1;
if (sr_template.nr_dev >= sr_template.dev_max) { if (sr_template.nr_dev >= sr_template.dev_max)
SDp->attached--; goto fail;
return 1;
}
for (cpnt = scsi_CDs, i = 0; i < sr_template.dev_max; i++, cpnt++) for (cpnt = scsi_CDs, i = 0; i < sr_template.dev_max; i++, cpnt++)
if (!cpnt->device) if (!cpnt->device)
break; break;
...@@ -484,6 +481,8 @@ static int sr_attach(Scsi_Device * SDp) ...@@ -484,6 +481,8 @@ static int sr_attach(Scsi_Device * SDp)
if (i >= sr_template.dev_max) if (i >= sr_template.dev_max)
panic("scsi_devices corrupt (sr)"); panic("scsi_devices corrupt (sr)");
if (sr_init_one(cpnt, i))
goto fail;
scsi_CDs[i].device = SDp; scsi_CDs[i].device = SDp;
...@@ -494,6 +493,10 @@ static int sr_attach(Scsi_Device * SDp) ...@@ -494,6 +493,10 @@ static int sr_attach(Scsi_Device * SDp)
printk("Attached scsi CD-ROM %s at scsi%d, channel %d, id %d, lun %d\n", printk("Attached scsi CD-ROM %s at scsi%d, channel %d, id %d, lun %d\n",
scsi_CDs[i].cdi.name, SDp->host->host_no, SDp->channel, SDp->id, SDp->lun); scsi_CDs[i].cdi.name, SDp->host->host_no, SDp->channel, SDp->id, SDp->lun);
return 0; return 0;
fail:
SDp->attached--;
return 1;
} }
...@@ -744,26 +747,16 @@ static int sr_init() ...@@ -744,26 +747,16 @@ static int sr_init()
return 1; return 1;
} }
void sr_finish() static int sr_init_one(Scsi_CD *cd, int first_minor)
{ {
int i;
for (i = 0; i < sr_template.nr_dev; ++i) {
struct gendisk *disk; struct gendisk *disk;
Scsi_CD *cd = &scsi_CDs[i];
/* If we have already seen this, then skip it. Comes up
* with loadable modules. */
if (cd->disk)
continue;
disk = alloc_disk(1); disk = alloc_disk(1);
if (!disk) if (!disk)
continue; return -ENOMEM;
if (cd->disk) {
put_disk(disk);
continue;
}
disk->major = MAJOR_NR; disk->major = MAJOR_NR;
disk->first_minor = i; disk->first_minor = first_minor;
strcpy(disk->disk_name, cd->cdi.name); strcpy(disk->disk_name, cd->cdi.name);
disk->fops = &sr_bdops; disk->fops = &sr_bdops;
disk->flags = GENHD_FL_CD; disk->flags = GENHD_FL_CD;
...@@ -788,6 +781,7 @@ void sr_finish() ...@@ -788,6 +781,7 @@ void sr_finish()
cd->cdi.handle = cd; cd->cdi.handle = cd;
cd->cdi.mask = 0; cd->cdi.mask = 0;
cd->cdi.capacity = 1; cd->cdi.capacity = 1;
/* /*
* FIXME: someone needs to handle a get_capabilities * FIXME: someone needs to handle a get_capabilities
* failure properly ?? * failure properly ??
...@@ -801,7 +795,8 @@ void sr_finish() ...@@ -801,7 +795,8 @@ void sr_finish()
disk->private_data = cd; disk->private_data = cd;
disk->queue = &cd->device->request_queue; disk->queue = &cd->device->request_queue;
add_disk(disk); add_disk(disk);
}
return 0;
} }
static void sr_detach(Scsi_Device * SDp) static void sr_detach(Scsi_Device * SDp)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment