2  *  drivers/cdrom/viocd.c
 
   4  *  iSeries Virtual CD Rom
 
   6  *  Authors: Dave Boutcher <boutcher@us.ibm.com>
 
   7  *           Ryan Arnold <ryanarn@us.ibm.com>
 
   8  *           Colin Devilbiss <devilbis@us.ibm.com>
 
   9  *           Stephen Rothwell <sfr@au1.ibm.com>
 
  11  * (C) Copyright 2000-2004 IBM Corporation
 
  13  * This program is free software;  you can redistribute it and/or
 
  14  * modify it under the terms of the GNU General Public License as
 
  15  * published by the Free Software Foundation; either version 2 of the
 
  16  * License, or (at your option) anyu later version.
 
  18  * This program is distributed in the hope that it will be useful, but
 
  19  * WITHOUT ANY WARRANTY; without even the implied warranty of
 
  20  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 
  21  * General Public License for more details.
 
  23  * You should have received a copy of the GNU General Public License
 
  24  * along with this program; if not, write to the Free Software Foundation,
 
  25  * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 
  27  * This routine provides access to CD ROM drives owned and managed by an
 
  28  * OS/400 partition running on the same box as this Linux partition.
 
  30  * All operations are performed by sending messages back and forth to
 
  31  * the OS/400 partition.
 
  34 #include <linux/major.h>
 
  35 #include <linux/blkdev.h>
 
  36 #include <linux/cdrom.h>
 
  37 #include <linux/errno.h>
 
  38 #include <linux/init.h>
 
  39 #include <linux/dma-mapping.h>
 
  40 #include <linux/module.h>
 
  41 #include <linux/completion.h>
 
  42 #include <linux/proc_fs.h>
 
  43 #include <linux/seq_file.h>
 
  46 #include <asm/scatterlist.h>
 
  47 #include <asm/iseries/hv_types.h>
 
  48 #include <asm/iseries/hv_lp_event.h>
 
  49 #include <asm/iseries/vio.h>
 
  50 #include <asm/firmware.h>
 
  52 #define VIOCD_DEVICE                    "iseries/vcd"
 
  54 #define VIOCD_VERS "1.06"
 
  56 #define VIOCD_KERN_WARNING              KERN_WARNING "viocd: "
 
  57 #define VIOCD_KERN_INFO                 KERN_INFO "viocd: "
 
  60         struct HvLpEvent        event;
 
  67         u64                     offset;         /* On open, max number of disks */
 
  68         u64                     len;            /* On open, size of the disk */
 
  69         u32                     block_size;     /* Only set on open */
 
  70         u32                     media_size;     /* Only set on open */
 
  78         viocdlockdoor = 0x0005,
 
  79         viocdgetinfo = 0x0006,
 
  84  * Should probably make this a module parameter....sigh
 
  86 #define VIOCD_MAX_CD    HVMAXARCHITECTEDVIRTUALCDROMS
 
  88 static const struct vio_error_entry viocd_err_table[] = {
 
  89         {0x0201, EINVAL, "Invalid Range"},
 
  90         {0x0202, EINVAL, "Invalid Token"},
 
  91         {0x0203, EIO, "DMA Error"},
 
  92         {0x0204, EIO, "Use Error"},
 
  93         {0x0205, EIO, "Release Error"},
 
  94         {0x0206, EINVAL, "Invalid CD"},
 
  95         {0x020C, EROFS, "Read Only Device"},
 
  96         {0x020D, ENOMEDIUM, "Changed or Missing Volume (or Varied Off?)"},
 
  97         {0x020E, EIO, "Optical System Error (Varied Off?)"},
 
  98         {0x02FF, EIO, "Internal Error"},
 
  99         {0x3010, EIO, "Changed Volume"},
 
 100         {0xC100, EIO, "Optical System Error"},
 
 105  * This is the structure we use to exchange info between driver and interrupt
 
 108 struct viocd_waitevent {
 
 109         struct completion       com;
 
 115 /* this is a lookup table for the true capabilities of a device */
 
 116 struct capability_entry {
 
 121 static struct capability_entry capability_table[] __initdata = {
 
 122         { "6330", CDC_LOCK | CDC_DVD_RAM | CDC_RAM },
 
 123         { "6331", CDC_LOCK | CDC_DVD_RAM | CDC_RAM },
 
 124         { "6333", CDC_LOCK | CDC_DVD_RAM | CDC_RAM },
 
 125         { "632A", CDC_LOCK | CDC_DVD_RAM | CDC_RAM },
 
 126         { "6321", CDC_LOCK },
 
 131 /* These are our internal structures for keeping track of devices */
 
 132 static int viocd_numdev;
 
 140  * This needs to be allocated since it is passed to the
 
 141  * Hypervisor and we may be a module.
 
 143 static struct cdrom_info *viocd_unitinfo;
 
 144 static dma_addr_t unitinfo_dmaaddr;
 
 147         struct gendisk                  *viocd_disk;
 
 148         struct cdrom_device_info        viocd_info;
 
 151 static struct disk_info viocd_diskinfo[VIOCD_MAX_CD];
 
 153 #define DEVICE_NR(di)   ((di) - &viocd_diskinfo[0])
 
 155 static spinlock_t viocd_reqlock;
 
 160 static int proc_viocd_show(struct seq_file *m, void *v)
 
 164         for (i = 0; i < viocd_numdev; i++) {
 
 165                 seq_printf(m, "viocd device %d is iSeries resource %10.10s"
 
 166                                 "type %4.4s, model %3.3s\n",
 
 167                                 i, viocd_unitinfo[i].rsrcname,
 
 168                                 viocd_unitinfo[i].type,
 
 169                                 viocd_unitinfo[i].model);
 
 174 static int proc_viocd_open(struct inode *inode, struct file *file)
 
 176         return single_open(file, proc_viocd_show, NULL);
 
 179 static const struct file_operations proc_viocd_operations = {
 
 180         .open           = proc_viocd_open,
 
 183         .release        = single_release,
 
 186 static int viocd_blk_open(struct inode *inode, struct file *file)
 
 188         struct disk_info *di = inode->i_bdev->bd_disk->private_data;
 
 189         return cdrom_open(&di->viocd_info, inode, file);
 
 192 static int viocd_blk_release(struct inode *inode, struct file *file)
 
 194         struct disk_info *di = inode->i_bdev->bd_disk->private_data;
 
 195         return cdrom_release(&di->viocd_info, file);
 
 198 static int viocd_blk_ioctl(struct inode *inode, struct file *file,
 
 199                 unsigned cmd, unsigned long arg)
 
 201         struct disk_info *di = inode->i_bdev->bd_disk->private_data;
 
 202         return cdrom_ioctl(file, &di->viocd_info, inode, cmd, arg);
 
 205 static int viocd_blk_media_changed(struct gendisk *disk)
 
 207         struct disk_info *di = disk->private_data;
 
 208         return cdrom_media_changed(&di->viocd_info);
 
 211 struct block_device_operations viocd_fops = {
 
 212         .owner =                THIS_MODULE,
 
 213         .open =                 viocd_blk_open,
 
 214         .release =              viocd_blk_release,
 
 215         .ioctl =                viocd_blk_ioctl,
 
 216         .media_changed =        viocd_blk_media_changed,
 
 219 /* Get info on CD devices from OS/400 */
 
 220 static void __init get_viocd_info(void)
 
 224         struct viocd_waitevent we;
 
 226         viocd_unitinfo = dma_alloc_coherent(iSeries_vio_dev,
 
 227                         sizeof(*viocd_unitinfo) * VIOCD_MAX_CD,
 
 228                         &unitinfo_dmaaddr, GFP_ATOMIC);
 
 229         if (viocd_unitinfo == NULL) {
 
 230                 printk(VIOCD_KERN_WARNING "error allocating unitinfo\n");
 
 234         memset(viocd_unitinfo, 0, sizeof(*viocd_unitinfo) * VIOCD_MAX_CD);
 
 236         init_completion(&we.com);
 
 238         hvrc = HvCallEvent_signalLpEventFast(viopath_hostLp,
 
 239                         HvLpEvent_Type_VirtualIo,
 
 240                         viomajorsubtype_cdio | viocdgetinfo,
 
 241                         HvLpEvent_AckInd_DoAck, HvLpEvent_AckType_ImmediateAck,
 
 242                         viopath_sourceinst(viopath_hostLp),
 
 243                         viopath_targetinst(viopath_hostLp),
 
 244                         (u64)&we, VIOVERSION << 16, unitinfo_dmaaddr, 0,
 
 245                         sizeof(*viocd_unitinfo) * VIOCD_MAX_CD, 0);
 
 246         if (hvrc != HvLpEvent_Rc_Good) {
 
 247                 printk(VIOCD_KERN_WARNING "cdrom error sending event. rc %d\n",
 
 252         wait_for_completion(&we.com);
 
 255                 const struct vio_error_entry *err =
 
 256                         vio_lookup_rc(viocd_err_table, we.sub_result);
 
 257                 printk(VIOCD_KERN_WARNING "bad rc %d:0x%04X on getinfo: %s\n",
 
 258                                 we.rc, we.sub_result, err->msg);
 
 262         for (i = 0; (i < VIOCD_MAX_CD) && viocd_unitinfo[i].rsrcname[0]; i++)
 
 266         if (viocd_numdev == 0) {
 
 267                 dma_free_coherent(iSeries_vio_dev,
 
 268                                 sizeof(*viocd_unitinfo) * VIOCD_MAX_CD,
 
 269                                 viocd_unitinfo, unitinfo_dmaaddr);
 
 270                 viocd_unitinfo = NULL;
 
 274 static int viocd_open(struct cdrom_device_info *cdi, int purpose)
 
 276         struct disk_info *diskinfo = cdi->handle;
 
 277         int device_no = DEVICE_NR(diskinfo);
 
 279         struct viocd_waitevent we;
 
 281         init_completion(&we.com);
 
 282         hvrc = HvCallEvent_signalLpEventFast(viopath_hostLp,
 
 283                         HvLpEvent_Type_VirtualIo,
 
 284                         viomajorsubtype_cdio | viocdopen,
 
 285                         HvLpEvent_AckInd_DoAck, HvLpEvent_AckType_ImmediateAck,
 
 286                         viopath_sourceinst(viopath_hostLp),
 
 287                         viopath_targetinst(viopath_hostLp),
 
 288                         (u64)&we, VIOVERSION << 16, ((u64)device_no << 48),
 
 291                 printk(VIOCD_KERN_WARNING
 
 292                                 "bad rc on HvCallEvent_signalLpEventFast %d\n",
 
 297         wait_for_completion(&we.com);
 
 300                 const struct vio_error_entry *err =
 
 301                         vio_lookup_rc(viocd_err_table, we.sub_result);
 
 302                 printk(VIOCD_KERN_WARNING "bad rc %d:0x%04X on open: %s\n",
 
 303                                 we.rc, we.sub_result, err->msg);
 
 310 static void viocd_release(struct cdrom_device_info *cdi)
 
 312         int device_no = DEVICE_NR((struct disk_info *)cdi->handle);
 
 315         hvrc = HvCallEvent_signalLpEventFast(viopath_hostLp,
 
 316                         HvLpEvent_Type_VirtualIo,
 
 317                         viomajorsubtype_cdio | viocdclose,
 
 318                         HvLpEvent_AckInd_NoAck, HvLpEvent_AckType_ImmediateAck,
 
 319                         viopath_sourceinst(viopath_hostLp),
 
 320                         viopath_targetinst(viopath_hostLp), 0,
 
 321                         VIOVERSION << 16, ((u64)device_no << 48), 0, 0, 0);
 
 323                 printk(VIOCD_KERN_WARNING
 
 324                                 "bad rc on HvCallEvent_signalLpEventFast %d\n",
 
 328 /* Send a read or write request to OS/400 */
 
 329 static int send_request(struct request *req)
 
 332         struct disk_info *diskinfo = req->rq_disk->private_data;
 
 337         struct scatterlist sg;
 
 339         BUG_ON(req->nr_phys_segments > 1);
 
 341         if (rq_data_dir(req) == READ) {
 
 342                 direction = DMA_FROM_DEVICE;
 
 343                 cmd = viomajorsubtype_cdio | viocdread;
 
 345                 direction = DMA_TO_DEVICE;
 
 346                 cmd = viomajorsubtype_cdio | viocdwrite;
 
 349         if (blk_rq_map_sg(req->q, req, &sg) == 0) {
 
 350                 printk(VIOCD_KERN_WARNING
 
 351                                 "error setting up scatter/gather list\n");
 
 355         if (dma_map_sg(diskinfo->dev, &sg, 1, direction) == 0) {
 
 356                 printk(VIOCD_KERN_WARNING "error allocating sg tce\n");
 
 359         dmaaddr = sg_dma_address(&sg);
 
 360         len = sg_dma_len(&sg);
 
 362         hvrc = HvCallEvent_signalLpEventFast(viopath_hostLp,
 
 363                         HvLpEvent_Type_VirtualIo, cmd,
 
 364                         HvLpEvent_AckInd_DoAck,
 
 365                         HvLpEvent_AckType_ImmediateAck,
 
 366                         viopath_sourceinst(viopath_hostLp),
 
 367                         viopath_targetinst(viopath_hostLp),
 
 368                         (u64)req, VIOVERSION << 16,
 
 369                         ((u64)DEVICE_NR(diskinfo) << 48) | dmaaddr,
 
 370                         (u64)req->sector * 512, len, 0);
 
 371         if (hvrc != HvLpEvent_Rc_Good) {
 
 372                 printk(VIOCD_KERN_WARNING "hv error on op %d\n", (int)hvrc);
 
 379 static void viocd_end_request(struct request *req, int uptodate)
 
 381         int nsectors = req->hard_nr_sectors;
 
 384          * Make sure it's fully ended, and ensure that we process
 
 385          * at least one sector.
 
 387         if (blk_pc_request(req))
 
 388                 nsectors = (req->data_len + 511) >> 9;
 
 392         if (end_that_request_first(req, uptodate, nsectors))
 
 394         add_disk_randomness(req->rq_disk);
 
 395         blkdev_dequeue_request(req);
 
 396         end_that_request_last(req, uptodate);
 
 401 static void do_viocd_request(request_queue_t *q)
 
 405         while ((rwreq == 0) && ((req = elv_next_request(q)) != NULL)) {
 
 406                 if (!blk_fs_request(req))
 
 407                         viocd_end_request(req, 0);
 
 408                 else if (send_request(req) < 0) {
 
 409                         printk(VIOCD_KERN_WARNING
 
 410                                         "unable to send message to OS/400!");
 
 411                         viocd_end_request(req, 0);
 
 417 static int viocd_media_changed(struct cdrom_device_info *cdi, int disc_nr)
 
 419         struct viocd_waitevent we;
 
 421         int device_no = DEVICE_NR((struct disk_info *)cdi->handle);
 
 423         init_completion(&we.com);
 
 425         /* Send the open event to OS/400 */
 
 426         hvrc = HvCallEvent_signalLpEventFast(viopath_hostLp,
 
 427                         HvLpEvent_Type_VirtualIo,
 
 428                         viomajorsubtype_cdio | viocdcheck,
 
 429                         HvLpEvent_AckInd_DoAck, HvLpEvent_AckType_ImmediateAck,
 
 430                         viopath_sourceinst(viopath_hostLp),
 
 431                         viopath_targetinst(viopath_hostLp),
 
 432                         (u64)&we, VIOVERSION << 16, ((u64)device_no << 48),
 
 435                 printk(VIOCD_KERN_WARNING "bad rc on HvCallEvent_signalLpEventFast %d\n",
 
 440         wait_for_completion(&we.com);
 
 442         /* Check the return code.  If bad, assume no change */
 
 444                 const struct vio_error_entry *err =
 
 445                         vio_lookup_rc(viocd_err_table, we.sub_result);
 
 446                 printk(VIOCD_KERN_WARNING
 
 447                                 "bad rc %d:0x%04X on check_change: %s; Assuming no change\n",
 
 448                                 we.rc, we.sub_result, err->msg);
 
 455 static int viocd_lock_door(struct cdrom_device_info *cdi, int locking)
 
 458         u64 device_no = DEVICE_NR((struct disk_info *)cdi->handle);
 
 459         /* NOTE: flags is 1 or 0 so it won't overwrite the device_no */
 
 460         u64 flags = !!locking;
 
 461         struct viocd_waitevent we;
 
 463         init_completion(&we.com);
 
 465         /* Send the lockdoor event to OS/400 */
 
 466         hvrc = HvCallEvent_signalLpEventFast(viopath_hostLp,
 
 467                         HvLpEvent_Type_VirtualIo,
 
 468                         viomajorsubtype_cdio | viocdlockdoor,
 
 469                         HvLpEvent_AckInd_DoAck, HvLpEvent_AckType_ImmediateAck,
 
 470                         viopath_sourceinst(viopath_hostLp),
 
 471                         viopath_targetinst(viopath_hostLp),
 
 472                         (u64)&we, VIOVERSION << 16,
 
 473                         (device_no << 48) | (flags << 32), 0, 0, 0);
 
 475                 printk(VIOCD_KERN_WARNING "bad rc on HvCallEvent_signalLpEventFast %d\n",
 
 480         wait_for_completion(&we.com);
 
 487 static int viocd_packet(struct cdrom_device_info *cdi,
 
 488                 struct packet_command *cgc)
 
 490         unsigned int buflen = cgc->buflen;
 
 493         switch (cgc->cmd[0]) {
 
 494         case GPCMD_READ_DISC_INFO:
 
 496                         disc_information *di = (disc_information *)cgc->buffer;
 
 499                                 di->disc_information_length = cpu_to_be16(1);
 
 504                                         (cdi->ops->capability & ~cdi->mask
 
 505                                          & (CDC_DVD_RAM | CDC_RAM)) != 0;
 
 508         case GPCMD_GET_CONFIGURATION:
 
 509                 if (cgc->cmd[3] == CDF_RWRT) {
 
 510                         struct rwrt_feature_desc *rfd = (struct rwrt_feature_desc *)(cgc->buffer + sizeof(struct feature_header));
 
 513                              (sizeof(struct feature_header) + sizeof(*rfd))) &&
 
 514                             (cdi->ops->capability & ~cdi->mask
 
 515                              & (CDC_DVD_RAM | CDC_RAM))) {
 
 516                                 rfd->feature_code = cpu_to_be16(CDF_RWRT);
 
 524                         /* indicate Unknown code */
 
 525                         cgc->sense->sense_key = 0x05;
 
 526                         cgc->sense->asc = 0x20;
 
 527                         cgc->sense->ascq = 0x00;
 
 536 static void restart_all_queues(int first_index)
 
 540         for (i = first_index + 1; i < viocd_numdev; i++)
 
 541                 if (viocd_diskinfo[i].viocd_disk)
 
 542                         blk_run_queue(viocd_diskinfo[i].viocd_disk->queue);
 
 543         for (i = 0; i <= first_index; i++)
 
 544                 if (viocd_diskinfo[i].viocd_disk)
 
 545                         blk_run_queue(viocd_diskinfo[i].viocd_disk->queue);
 
 548 /* This routine handles incoming CD LP events */
 
 549 static void vio_handle_cd_event(struct HvLpEvent *event)
 
 551         struct viocdlpevent *bevent;
 
 552         struct viocd_waitevent *pwe;
 
 553         struct disk_info *di;
 
 559                 /* Notification that a partition went away! */
 
 561         /* First, we should NEVER get an int here...only acks */
 
 562         if (hvlpevent_is_int(event)) {
 
 563                 printk(VIOCD_KERN_WARNING
 
 564                                 "Yikes! got an int in viocd event handler!\n");
 
 565                 if (hvlpevent_need_ack(event)) {
 
 566                         event->xRc = HvLpEvent_Rc_InvalidSubtype;
 
 567                         HvCallEvent_ackLpEvent(event);
 
 571         bevent = (struct viocdlpevent *)event;
 
 573         switch (event->xSubtype & VIOMINOR_SUBTYPE_MASK) {
 
 575                 if (event->xRc == 0) {
 
 576                         di = &viocd_diskinfo[bevent->disk];
 
 577                         blk_queue_hardsect_size(di->viocd_disk->queue,
 
 579                         set_capacity(di->viocd_disk,
 
 581                                         bevent->block_size / 512);
 
 586                 pwe = (struct viocd_waitevent *)event->xCorrelationToken;
 
 588                 pwe->rc = event->xRc;
 
 589                 pwe->sub_result = bevent->sub_result;
 
 594                 pwe = (struct viocd_waitevent *)event->xCorrelationToken;
 
 595                 pwe->changed = bevent->flags;
 
 596                 goto return_complete;
 
 604                  * Since this is running in interrupt mode, we need to
 
 605                  * make sure we're not stepping on any global I/O operations
 
 607                 di = &viocd_diskinfo[bevent->disk];
 
 608                 spin_lock_irqsave(&viocd_reqlock, flags);
 
 609                 dma_unmap_single(di->dev, bevent->token, bevent->len,
 
 610                                 ((event->xSubtype & VIOMINOR_SUBTYPE_MASK) == viocdread)
 
 611                                 ?  DMA_FROM_DEVICE : DMA_TO_DEVICE);
 
 612                 req = (struct request *)bevent->event.xCorrelationToken;
 
 615                 if (event->xRc != HvLpEvent_Rc_Good) {
 
 616                         const struct vio_error_entry *err =
 
 617                                 vio_lookup_rc(viocd_err_table,
 
 619                         printk(VIOCD_KERN_WARNING "request %p failed "
 
 620                                         "with rc %d:0x%04X: %s\n",
 
 622                                         bevent->sub_result, err->msg);
 
 623                         viocd_end_request(req, 0);
 
 625                         viocd_end_request(req, 1);
 
 627                 /* restart handling of incoming requests */
 
 628                 spin_unlock_irqrestore(&viocd_reqlock, flags);
 
 629                 restart_all_queues(bevent->disk);
 
 633                 printk(VIOCD_KERN_WARNING
 
 634                                 "message with invalid subtype %0x04X!\n",
 
 635                                 event->xSubtype & VIOMINOR_SUBTYPE_MASK);
 
 636                 if (hvlpevent_need_ack(event)) {
 
 637                         event->xRc = HvLpEvent_Rc_InvalidSubtype;
 
 638                         HvCallEvent_ackLpEvent(event);
 
 643 static struct cdrom_device_ops viocd_dops = {
 
 645         .release = viocd_release,
 
 646         .media_changed = viocd_media_changed,
 
 647         .lock_door = viocd_lock_door,
 
 648         .generic_packet = viocd_packet,
 
 649         .capability = CDC_CLOSE_TRAY | CDC_OPEN_TRAY | CDC_LOCK | CDC_SELECT_SPEED | CDC_SELECT_DISC | CDC_MULTI_SESSION | CDC_MCN | CDC_MEDIA_CHANGED | CDC_PLAY_AUDIO | CDC_RESET | CDC_DRIVE_STATUS | CDC_GENERIC_PACKET | CDC_CD_R | CDC_CD_RW | CDC_DVD | CDC_DVD_R | CDC_DVD_RAM | CDC_RAM
 
 652 static int __init find_capability(const char *type)
 
 654         struct capability_entry *entry;
 
 656         for(entry = capability_table; entry->type; ++entry)
 
 657                 if(!strncmp(entry->type, type, 4))
 
 659         return entry->capability;
 
 662 static int viocd_probe(struct vio_dev *vdev, const struct vio_device_id *id)
 
 664         struct gendisk *gendisk;
 
 667         struct cdrom_device_info *c;
 
 668         struct cdrom_info *ci;
 
 669         struct request_queue *q;
 
 671         deviceno = vdev->unit_address;
 
 672         if (deviceno >= viocd_numdev)
 
 675         d = &viocd_diskinfo[deviceno];
 
 677         ci = &viocd_unitinfo[deviceno];
 
 679         c->ops = &viocd_dops;
 
 683         c->mask = ~find_capability(ci->type);
 
 684         sprintf(c->name, VIOCD_DEVICE "%c", 'a' + deviceno);
 
 686         if (register_cdrom(c) != 0) {
 
 687                 printk(VIOCD_KERN_WARNING "Cannot register viocd CD-ROM %s!\n",
 
 691         printk(VIOCD_KERN_INFO "cd %s is iSeries resource %10.10s "
 
 692                         "type %4.4s, model %3.3s\n",
 
 693                         c->name, ci->rsrcname, ci->type, ci->model);
 
 694         q = blk_init_queue(do_viocd_request, &viocd_reqlock);
 
 696                 printk(VIOCD_KERN_WARNING "Cannot allocate queue for %s!\n",
 
 698                 goto out_unregister_cdrom;
 
 700         gendisk = alloc_disk(1);
 
 701         if (gendisk == NULL) {
 
 702                 printk(VIOCD_KERN_WARNING "Cannot create gendisk for %s!\n",
 
 704                 goto out_cleanup_queue;
 
 706         gendisk->major = VIOCD_MAJOR;
 
 707         gendisk->first_minor = deviceno;
 
 708         strncpy(gendisk->disk_name, c->name,
 
 709                         sizeof(gendisk->disk_name));
 
 710         blk_queue_max_hw_segments(q, 1);
 
 711         blk_queue_max_phys_segments(q, 1);
 
 712         blk_queue_max_sectors(q, 4096 / 512);
 
 714         gendisk->fops = &viocd_fops;
 
 715         gendisk->flags = GENHD_FL_CD|GENHD_FL_REMOVABLE;
 
 716         set_capacity(gendisk, 0);
 
 717         gendisk->private_data = d;
 
 718         d->viocd_disk = gendisk;
 
 720         gendisk->driverfs_dev = d->dev;
 
 725         blk_cleanup_queue(q);
 
 726 out_unregister_cdrom:
 
 732 static int viocd_remove(struct vio_dev *vdev)
 
 734         struct disk_info *d = &viocd_diskinfo[vdev->unit_address];
 
 736         if (unregister_cdrom(&d->viocd_info) != 0)
 
 737                 printk(VIOCD_KERN_WARNING
 
 738                                 "Cannot unregister viocd CD-ROM %s!\n",
 
 740         del_gendisk(d->viocd_disk);
 
 741         blk_cleanup_queue(d->viocd_disk->queue);
 
 742         put_disk(d->viocd_disk);
 
 747  * viocd_device_table: Used by vio.c to match devices that we
 
 750 static struct vio_device_id viocd_device_table[] __devinitdata = {
 
 751         { "block", "IBM,iSeries-viocd" },
 
 754 MODULE_DEVICE_TABLE(vio, viocd_device_table);
 
 756 static struct vio_driver viocd_driver = {
 
 757         .id_table = viocd_device_table,
 
 758         .probe = viocd_probe,
 
 759         .remove = viocd_remove,
 
 762                 .owner = THIS_MODULE,
 
 766 static int __init viocd_init(void)
 
 768         struct proc_dir_entry *e;
 
 771         if (!firmware_has_feature(FW_FEATURE_ISERIES))
 
 774         if (viopath_hostLp == HvLpIndexInvalid) {
 
 776                 /* If we don't have a host, bail out */
 
 777                 if (viopath_hostLp == HvLpIndexInvalid)
 
 781         printk(VIOCD_KERN_INFO "vers " VIOCD_VERS ", hosting partition %d\n",
 
 784         if (register_blkdev(VIOCD_MAJOR, VIOCD_DEVICE) != 0) {
 
 785                 printk(VIOCD_KERN_WARNING "Unable to get major %d for %s\n",
 
 786                                 VIOCD_MAJOR, VIOCD_DEVICE);
 
 790         ret = viopath_open(viopath_hostLp, viomajorsubtype_cdio,
 
 793                 printk(VIOCD_KERN_WARNING
 
 794                                 "error opening path to host partition %d\n",
 
 799         /* Initialize our request handler */
 
 800         vio_setHandler(viomajorsubtype_cdio, vio_handle_cd_event);
 
 804         spin_lock_init(&viocd_reqlock);
 
 806         ret = vio_register_driver(&viocd_driver);
 
 810         e = create_proc_entry("iSeries/viocd", S_IFREG|S_IRUGO, NULL);
 
 812                 e->owner = THIS_MODULE;
 
 813                 e->proc_fops = &proc_viocd_operations;
 
 819         dma_free_coherent(iSeries_vio_dev,
 
 820                         sizeof(*viocd_unitinfo) * VIOCD_MAX_CD,
 
 821                         viocd_unitinfo, unitinfo_dmaaddr);
 
 822         vio_clearHandler(viomajorsubtype_cdio);
 
 823         viopath_close(viopath_hostLp, viomajorsubtype_cdio, MAX_CD_REQ + 2);
 
 825         unregister_blkdev(VIOCD_MAJOR, VIOCD_DEVICE);
 
 829 static void __exit viocd_exit(void)
 
 831         remove_proc_entry("iSeries/viocd", NULL);
 
 832         vio_unregister_driver(&viocd_driver);
 
 833         if (viocd_unitinfo != NULL)
 
 834                 dma_free_coherent(iSeries_vio_dev,
 
 835                                 sizeof(*viocd_unitinfo) * VIOCD_MAX_CD,
 
 836                                 viocd_unitinfo, unitinfo_dmaaddr);
 
 837         viopath_close(viopath_hostLp, viomajorsubtype_cdio, MAX_CD_REQ + 2);
 
 838         vio_clearHandler(viomajorsubtype_cdio);
 
 839         unregister_blkdev(VIOCD_MAJOR, VIOCD_DEVICE);
 
 842 module_init(viocd_init);
 
 843 module_exit(viocd_exit);
 
 844 MODULE_LICENSE("GPL");