| /* |
| * Sony CDU-535 interface device driver |
| * |
| * This is a modified version of the CDU-31A device driver (see below). |
| * Changes were made using documentation for the CDU-531 (which Sony |
| * assures me is very similar to the 535) and partial disassembly of the |
| * DOS driver. I used Minyard's driver and replaced the CDU-31A |
| * commands with the CDU-531 commands. This was complicated by a different |
| * interface protocol with the drive. The driver is still polled. |
| * |
| * Data transfer rate is about 110 Kb/sec, theoretical maximum is 150 Kb/sec. |
| * I tried polling without the sony_sleep during the data transfers but |
| * it did not speed things up any. |
| * |
| * 1993-05-23 (rgj) changed the major number to 21 to get rid of conflict |
| * with CDU-31A driver. This is the also the number from the Linux |
| * Device Driver Registry for the Sony Drive. Hope nobody else is using it. |
| * |
| * 1993-08-29 (rgj) remove the configuring of the interface board address |
| * from the top level configuration, you have to modify it in this file. |
| * |
| * 1995-01-26 Made module-capable (Joel Katz <Stimpson@Panix.COM>) |
| * |
| * 1995-05-20 |
| * Modified to support CDU-510/515 series |
| * (Claudio Porfiri<C.Porfiri@nisms.tei.ericsson.se>) |
| * Fixed to report verify_area() failures |
| * (Heiko Eissfeldt <heiko@colossus.escape.de>) |
| * |
| * 1995-06-01 |
| * More changes to support CDU-510/515 series |
| * (Claudio Porfiri<C.Porfiri@nisms.tei.ericsson.se>) |
| * |
| * November 1999 -- Make kernel-parameter implementation work with 2.3.x |
| * Removed init_module & cleanup_module in favor of |
| * module_init & module_exit. |
| * Torben Mathiasen <tmm@image.dk> |
| * |
| * September 2003 - Fix SMP support by removing cli/sti calls. |
| * Using spinlocks with a wait_queue instead. |
| * Felipe Damasio <felipewd@terra.com.br> |
| * |
| * Things to do: |
| * - handle errors and status better, put everything into a single word |
| * - use interrupts (code mostly there, but a big hole still missing) |
| * - handle multi-session CDs? |
| * - use DMA? |
| * |
| * Known Bugs: |
| * - |
| * |
| * Ken Pizzini (ken@halcyon.com) |
| * |
| * Original by: |
| * Ron Jeppesen (ronj.an@site007.saic.com) |
| * |
| * |
| *------------------------------------------------------------------------ |
| * Sony CDROM interface device driver. |
| * |
| * Corey Minyard (minyard@wf-rch.cirr.com) (CDU-535 complaints to Ken above) |
| * |
| * Colossians 3:17 |
| * |
| * The Sony interface device driver handles Sony interface CDROM |
| * drives and provides a complete block-level interface as well as an |
| * ioctl() interface compatible with the Sun (as specified in |
| * include/linux/cdrom.h). With this interface, CDROMs can be |
| * accessed and standard audio CDs can be played back normally. |
| * |
| * This interface is (unfortunately) a polled interface. This is |
| * because most Sony interfaces are set up with DMA and interrupts |
| * disables. Some (like mine) do not even have the capability to |
| * handle interrupts or DMA. For this reason you will see a bit of |
| * the following: |
| * |
| * snap = jiffies; |
| * while (jiffies-snap < SONY_JIFFIES_TIMEOUT) |
| * { |
| * if (some_condition()) |
| * break; |
| * sony_sleep(); |
| * } |
| * if (some_condition not met) |
| * { |
| * return an_error; |
| * } |
| * |
| * This ugly hack waits for something to happen, sleeping a little |
| * between every try. (The conditional is written so that jiffies |
| * wrap-around is handled properly.) |
| * |
| * One thing about these drives: They talk in MSF (Minute Second Frame) format. |
| * There are 75 frames a second, 60 seconds a minute, and up to 75 minutes on a |
| * disk. The funny thing is that these are sent to the drive in BCD, but the |
| * interface wants to see them in decimal. A lot of conversion goes on. |
| * |
| * Copyright (C) 1993 Corey Minyard |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. |
| * |
| */ |
| |
| |
| # include <linux/module.h> |
| |
| #include <linux/errno.h> |
| #include <linux/signal.h> |
| #include <linux/sched.h> |
| #include <linux/timer.h> |
| #include <linux/fs.h> |
| #include <linux/kernel.h> |
| #include <linux/interrupt.h> |
| #include <linux/ioport.h> |
| #include <linux/hdreg.h> |
| #include <linux/genhd.h> |
| #include <linux/mm.h> |
| #include <linux/slab.h> |
| #include <linux/init.h> |
| |
| #define REALLY_SLOW_IO |
| #include <asm/system.h> |
| #include <asm/io.h> |
| #include <asm/uaccess.h> |
| |
| #include <linux/cdrom.h> |
| |
| #define MAJOR_NR CDU535_CDROM_MAJOR |
| #include <linux/blkdev.h> |
| |
| #define sony535_cd_base_io sonycd535 /* for compatible parameter passing with "insmod" */ |
| #include "sonycd535.h" |
| |
| /* |
| * this is the base address of the interface card for the Sony CDU-535 |
| * CDROM drive. If your jumpers are set for an address other than |
| * this one (the default), change the following line to the |
| * proper address. |
| */ |
| #ifndef CDU535_ADDRESS |
| # define CDU535_ADDRESS 0x340 |
| #endif |
| #ifndef CDU535_INTERRUPT |
| # define CDU535_INTERRUPT 0 |
| #endif |
| #ifndef CDU535_HANDLE |
| # define CDU535_HANDLE "cdu535" |
| #endif |
| #ifndef CDU535_MESSAGE_NAME |
| # define CDU535_MESSAGE_NAME "Sony CDU-535" |
| #endif |
| |
| #define CDU535_BLOCK_SIZE 2048 |
| |
| #ifndef MAX_SPINUP_RETRY |
| # define MAX_SPINUP_RETRY 3 /* 1 is sufficient for most drives... */ |
| #endif |
| #ifndef RETRY_FOR_BAD_STATUS |
| # define RETRY_FOR_BAD_STATUS 100 /* in 10th of second */ |
| #endif |
| |
| #ifndef DEBUG |
| # define DEBUG 1 |
| #endif |
| |
| /* |
| * SONY535_BUFFER_SIZE determines the size of internal buffer used |
| * by the drive. It must be at least 2K and the larger the buffer |
| * the better the transfer rate. It does however take system memory. |
| * On my system I get the following transfer rates using dd to read |
| * 10 Mb off /dev/cdrom. |
| * |
| * 8K buffer 43 Kb/sec |
| * 16K buffer 66 Kb/sec |
| * 32K buffer 91 Kb/sec |
| * 64K buffer 111 Kb/sec |
| * 128K buffer 123 Kb/sec |
| * 512K buffer 123 Kb/sec |
| */ |
| #define SONY535_BUFFER_SIZE (64*1024) |
| |
| /* |
| * if LOCK_DOORS is defined then the eject button is disabled while |
| * the device is open. |
| */ |
| #ifndef NO_LOCK_DOORS |
| # define LOCK_DOORS |
| #endif |
| |
| static int read_subcode(void); |
| static void sony_get_toc(void); |
| static int cdu_open(struct inode *inode, struct file *filp); |
| static inline unsigned int int_to_bcd(unsigned int val); |
| static unsigned int bcd_to_int(unsigned int bcd); |
| static int do_sony_cmd(Byte * cmd, int nCmd, Byte status[2], |
| Byte * response, int n_response, int ignoreStatusBit7); |
| |
| /* The base I/O address of the Sony Interface. This is a variable (not a |
| #define) so it can be easily changed via some future ioctl() */ |
| static unsigned int sony535_cd_base_io = CDU535_ADDRESS; |
| module_param(sony535_cd_base_io, int, 0); |
| |
| /* |
| * The following are I/O addresses of the various registers for the drive. The |
| * comment for the base address also applies here. |
| */ |
| static unsigned short select_unit_reg; |
| static unsigned short result_reg; |
| static unsigned short command_reg; |
| static unsigned short read_status_reg; |
| static unsigned short data_reg; |
| |
| static DEFINE_SPINLOCK(sonycd535_lock); /* queue lock */ |
| static struct request_queue *sonycd535_queue; |
| |
| static int initialized; /* Has the drive been initialized? */ |
| static int sony_disc_changed = 1; /* Has the disk been changed |
| since the last check? */ |
| static int sony_toc_read; /* Has the table of contents been |
| read? */ |
| static unsigned int sony_buffer_size; /* Size in bytes of the read-ahead |
| buffer. */ |
| static unsigned int sony_buffer_sectors; /* Size (in 2048 byte records) of |
| the read-ahead buffer. */ |
| static unsigned int sony_usage; /* How many processes have the |
| drive open. */ |
| |
| static int sony_first_block = -1; /* First OS block (512 byte) in |
| the read-ahead buffer */ |
| static int sony_last_block = -1; /* Last OS block (512 byte) in |
| the read-ahead buffer */ |
| |
| static struct s535_sony_toc *sony_toc; /* Points to the table of |
| contents. */ |
| |
| static struct s535_sony_subcode *last_sony_subcode; /* Points to the last |
| subcode address read */ |
| static Byte **sony_buffer; /* Points to the pointers |
| to the sector buffers */ |
| |
| static int sony_inuse; /* is the drive in use? Only one |
| open at a time allowed */ |
| |
| /* |
| * The audio status uses the values from read subchannel data as specified |
| * in include/linux/cdrom.h. |
| */ |
| static int sony_audio_status = CDROM_AUDIO_NO_STATUS; |
| |
| /* |
| * The following are a hack for pausing and resuming audio play. The drive |
| * does not work as I would expect it, if you stop it then start it again, |
| * the drive seeks back to the beginning and starts over. This holds the |
| * position during a pause so a resume can restart it. It uses the |
| * audio status variable above to tell if it is paused. |
| * I just kept the CDU-31A driver behavior rather than using the PAUSE |
| * command on the CDU-535. |
| */ |
| static Byte cur_pos_msf[3]; |
| static Byte final_pos_msf[3]; |
| |
| /* What IRQ is the drive using? 0 if none. */ |
| static int sony535_irq_used = CDU535_INTERRUPT; |
| |
| /* The interrupt handler will wake this queue up when it gets an interrupt. */ |
| static DECLARE_WAIT_QUEUE_HEAD(cdu535_irq_wait); |
| |
| |
| /* |
| * This routine returns 1 if the disk has been changed since the last |
| * check or 0 if it hasn't. Setting flag to 0 resets the changed flag. |
| */ |
| static int |
| cdu535_check_media_change(struct gendisk *disk) |
| { |
| /* if driver is not initialized, always return 0 */ |
| int retval = initialized ? sony_disc_changed : 0; |
| sony_disc_changed = 0; |
| return retval; |
| } |
| |
| static inline void |
| enable_interrupts(void) |
| { |
| #ifdef USE_IRQ |
| /* |
| * This code was taken from cdu31a.c; it will not |
| * directly work for the cdu535 as written... |
| */ |
| curr_control_reg |= ( SONY_ATTN_INT_EN_BIT |
| | SONY_RES_RDY_INT_EN_BIT |
| | SONY_DATA_RDY_INT_EN_BIT); |
| outb(curr_control_reg, sony_cd_control_reg); |
| #endif |
| } |
| |
| static inline void |
| disable_interrupts(void) |
| { |
| #ifdef USE_IRQ |
| /* |
| * This code was taken from cdu31a.c; it will not |
| * directly work for the cdu535 as written... |
| */ |
| curr_control_reg &= ~(SONY_ATTN_INT_EN_BIT |
| | SONY_RES_RDY_INT_EN_BIT |
| | SONY_DATA_RDY_INT_EN_BIT); |
| outb(curr_control_reg, sony_cd_control_reg); |
| #endif |
| } |
| |
| static irqreturn_t |
| cdu535_interrupt(int irq, void *dev_id) |
| { |
| disable_interrupts(); |
| if (waitqueue_active(&cdu535_irq_wait)) { |
| wake_up(&cdu535_irq_wait); |
| return IRQ_HANDLED; |
| } |
| printk(CDU535_MESSAGE_NAME |
| ": Got an interrupt but nothing was waiting\n"); |
| return IRQ_NONE; |
| } |
| |
| |
| /* |
| * Wait a little while. |
| */ |
| static inline void |
| sony_sleep(void) |
| { |
| if (sony535_irq_used <= 0) { /* poll */ |
| yield(); |
| } else { /* Interrupt driven */ |
| DEFINE_WAIT(wait); |
| |
| spin_lock_irq(&sonycd535_lock); |
| enable_interrupts(); |
| prepare_to_wait(&cdu535_irq_wait, &wait, TASK_INTERRUPTIBLE); |
| spin_unlock_irq(&sonycd535_lock); |
| schedule(); |
| finish_wait(&cdu535_irq_wait, &wait); |
| } |
| } |
| |
| /*------------------start of SONY CDU535 very specific ---------------------*/ |
| |
| /**************************************************************************** |
| * void select_unit( int unit_no ) |
| * |
| * Select the specified unit (0-3) so that subsequent commands reference it |
| ****************************************************************************/ |
| static void |
| select_unit(int unit_no) |
| { |
| unsigned int select_mask = ~(1 << unit_no); |
| outb(select_mask, select_unit_reg); |
| } |
| |
| /*************************************************************************** |
| * int read_result_reg( Byte *data_ptr ) |
| * |
| * Read a result byte from the Sony CDU controller, store in location pointed |
| * to by data_ptr. Return zero on success, TIME_OUT if we did not receive |
| * data. |
| ***************************************************************************/ |
| static int |
| read_result_reg(Byte *data_ptr) |
| { |
| unsigned long snap; |
| int read_status; |
| |
| snap = jiffies; |
| while (jiffies-snap < SONY_JIFFIES_TIMEOUT) { |
| read_status = inb(read_status_reg); |
| if ((read_status & SONY535_RESULT_NOT_READY_BIT) == 0) { |
| #if DEBUG > 1 |
| printk(CDU535_MESSAGE_NAME |
| ": read_result_reg(): readStatReg = 0x%x\n", read_status); |
| #endif |
| *data_ptr = inb(result_reg); |
| return 0; |
| } else { |
| sony_sleep(); |
| } |
| } |
| printk(CDU535_MESSAGE_NAME " read_result_reg: TIME OUT!\n"); |
| return TIME_OUT; |
| } |
| |
| /**************************************************************************** |
| * int read_exec_status( Byte status[2] ) |
| * |
| * Read the execution status of the last command and put into status. |
| * Handles reading second status word if available. Returns 0 on success, |
| * TIME_OUT on failure. |
| ****************************************************************************/ |
| static int |
| read_exec_status(Byte status[2]) |
| { |
| status[1] = 0; |
| if (read_result_reg(&(status[0])) != 0) |
| return TIME_OUT; |
| if ((status[0] & 0x80) != 0) { /* byte two follows */ |
| if (read_result_reg(&(status[1])) != 0) |
| return TIME_OUT; |
| } |
| #if DEBUG > 1 |
| printk(CDU535_MESSAGE_NAME ": read_exec_status: read 0x%x 0x%x\n", |
| status[0], status[1]); |
| #endif |
| return 0; |
| } |
| |
| /**************************************************************************** |
| * int check_drive_status( void ) |
| * |
| * Check the current drive status. Using this before executing a command |
| * takes care of the problem of unsolicited drive status-2 messages. |
| * Add a check of the audio status if we think the disk is playing. |
| ****************************************************************************/ |
| static int |
| check_drive_status(void) |
| { |
| Byte status, e_status[2]; |
| int CDD, ATN; |
| Byte cmd; |
| |
| select_unit(0); |
| if (sony_audio_status == CDROM_AUDIO_PLAY) { /* check status */ |
| outb(SONY535_REQUEST_AUDIO_STATUS, command_reg); |
| if (read_result_reg(&status) == 0) { |
| switch (status) { |
| case 0x0: |
| break; /* play in progress */ |
| case 0x1: |
| break; /* paused */ |
| case 0x3: /* audio play completed */ |
| case 0x5: /* play not requested */ |
| sony_audio_status = CDROM_AUDIO_COMPLETED; |
| read_subcode(); |
| break; |
| case 0x4: /* error during play */ |
| sony_audio_status = CDROM_AUDIO_ERROR; |
| break; |
| } |
| } |
| } |
| /* now check drive status */ |
| outb(SONY535_REQUEST_DRIVE_STATUS_2, command_reg); |
| if (read_result_reg(&status) != 0) |
| return TIME_OUT; |
| |
| #if DEBUG > 1 |
| printk(CDU535_MESSAGE_NAME ": check_drive_status() got 0x%x\n", status); |
| #endif |
| |
| if (status == 0) |
| return 0; |
| |
| ATN = status & 0xf; |
| CDD = (status >> 4) & 0xf; |
| |
| switch (ATN) { |
| case 0x0: |
| break; /* go on to CDD stuff */ |
| case SONY535_ATN_BUSY: |
| if (initialized) |
| printk(CDU535_MESSAGE_NAME " error: drive busy\n"); |
| return CD_BUSY; |
| case SONY535_ATN_EJECT_IN_PROGRESS: |
| printk(CDU535_MESSAGE_NAME " error: eject in progress\n"); |
| sony_audio_status = CDROM_AUDIO_INVALID; |
| return CD_BUSY; |
| case SONY535_ATN_RESET_OCCURRED: |
| case SONY535_ATN_DISC_CHANGED: |
| case SONY535_ATN_RESET_AND_DISC_CHANGED: |
| #if DEBUG > 0 |
| printk(CDU535_MESSAGE_NAME " notice: reset occurred or disc changed\n"); |
| #endif |
| sony_disc_changed = 1; |
| sony_toc_read = 0; |
| sony_audio_status = CDROM_AUDIO_NO_STATUS; |
| sony_first_block = -1; |
| sony_last_block = -1; |
| if (initialized) { |
| cmd = SONY535_SPIN_UP; |
| do_sony_cmd(&cmd, 1, e_status, NULL, 0, 0); |
| sony_get_toc(); |
| } |
| return 0; |
| default: |
| printk(CDU535_MESSAGE_NAME " error: drive busy (ATN=0x%x)\n", ATN); |
| return CD_BUSY; |
| } |
| switch (CDD) { /* the 531 docs are not helpful in decoding this */ |
| case 0x0: /* just use the values from the DOS driver */ |
| case 0x2: |
| case 0xa: |
| break; /* no error */ |
| case 0xc: |
| printk(CDU535_MESSAGE_NAME |
| ": check_drive_status(): CDD = 0xc! Not properly handled!\n"); |
| return CD_BUSY; /* ? */ |
| default: |
| return CD_BUSY; |
| } |
| return 0; |
| } /* check_drive_status() */ |
| |
| /***************************************************************************** |
| * int do_sony_cmd( Byte *cmd, int n_cmd, Byte status[2], |
| * Byte *response, int n_response, int ignore_status_bit7 ) |
| * |
| * Generic routine for executing commands. The command and its parameters |
| * should be placed in the cmd[] array, number of bytes in the command is |
| * stored in nCmd. The response from the command will be stored in the |
| * response array. The number of bytes you expect back (excluding status) |
| * should be passed in n_response. Finally, some |
| * commands set bit 7 of the return status even when there is no second |
| * status byte, on these commands set ignoreStatusBit7 TRUE. |
| * If the command was sent and data received back, then we return 0, |
| * else we return TIME_OUT. You still have to check the status yourself. |
| * You should call check_drive_status() before calling this routine |
| * so that you do not lose notifications of disk changes, etc. |
| ****************************************************************************/ |
| static int |
| do_sony_cmd(Byte * cmd, int n_cmd, Byte status[2], |
| Byte * response, int n_response, int ignore_status_bit7) |
| { |
| int i; |
| |
| /* write out the command */ |
| for (i = 0; i < n_cmd; i++) |
| outb(cmd[i], command_reg); |
| |
| /* read back the status */ |
| if (read_result_reg(status) != 0) |
| return TIME_OUT; |
| if (!ignore_status_bit7 && ((status[0] & 0x80) != 0)) { |
| /* get second status byte */ |
| if (read_result_reg(status + 1) != 0) |
| return TIME_OUT; |
| } else { |
| status[1] = 0; |
| } |
| #if DEBUG > 2 |
| printk(CDU535_MESSAGE_NAME ": do_sony_cmd %x: %x %x\n", |
| *cmd, status[0], status[1]); |
| #endif |
| |
| /* do not know about when I should read set of data and when not to */ |
| if ((status[0] & ((ignore_status_bit7 ? 0x7f : 0xff) & 0x8f)) != 0) |
| return 0; |
| |
| /* else, read in rest of data */ |
| for (i = 0; 0 < n_response; n_response--, i++) |
| if (read_result_reg(response + i) != 0) |
| return TIME_OUT; |
| return 0; |
| } /* do_sony_cmd() */ |
| |
| /************************************************************************** |
| * int set_drive_mode( int mode, Byte status[2] ) |
| * |
| * Set the drive mode to the specified value (mode=0 is audio, mode=e0 |
| * is mode-1 CDROM |
| **************************************************************************/ |
| static int |
| set_drive_mode(int mode, Byte status[2]) |
| { |
| Byte cmd_buff[2]; |
| Byte ret_buff[1]; |
| |
| cmd_buff[0] = SONY535_SET_DRIVE_MODE; |
| cmd_buff[1] = mode; |
| return do_sony_cmd(cmd_buff, 2, status, ret_buff, 1, 1); |
| } |
| |
| /*************************************************************************** |
| * int seek_and_read_N_blocks( Byte params[], int n_blocks, Byte status[2], |
| * Byte *data_buff, int buff_size ) |
| * |
| * Read n_blocks of data from the CDROM starting at position params[0:2], |
| * number of blocks in stored in params[3:5] -- both these are already |
| * int bcd format. |
| * Transfer the data into the buffer pointed at by data_buff. buff_size |
| * gives the number of bytes available in the buffer. |
| * The routine returns number of bytes read in if successful, otherwise |
| * it returns one of the standard error returns. |
| ***************************************************************************/ |
| static int |
| seek_and_read_N_blocks(Byte params[], int n_blocks, Byte status[2], |
| Byte **buff, int buf_size) |
| { |
| Byte cmd_buff[7]; |
| int i; |
| int read_status; |
| unsigned long snap; |
| Byte *data_buff; |
| int sector_count = 0; |
| |
| if (buf_size < CDU535_BLOCK_SIZE * n_blocks) |
| return NO_ROOM; |
| |
| set_drive_mode(SONY535_CDROM_DRIVE_MODE, status); |
| |
| /* send command to read the data */ |
| cmd_buff[0] = SONY535_SEEK_AND_READ_N_BLOCKS_1; |
| for (i = 0; i < 6; i++) |
| cmd_buff[i + 1] = params[i]; |
| for (i = 0; i < 7; i++) |
| outb(cmd_buff[i], command_reg); |
| |
| /* read back the data one block at a time */ |
| while (0 < n_blocks--) { |
| /* wait for data to be ready */ |
| int data_valid = 0; |
| snap = jiffies; |
| while (jiffies-snap < SONY_JIFFIES_TIMEOUT) { |
| read_status = inb(read_status_reg); |
| if ((read_status & SONY535_RESULT_NOT_READY_BIT) == 0) { |
| read_exec_status(status); |
| return BAD_STATUS; |
| } |
| if ((read_status & SONY535_DATA_NOT_READY_BIT) == 0) { |
| /* data is ready, read it */ |
| data_buff = buff[sector_count++]; |
| for (i = 0; i < CDU535_BLOCK_SIZE; i++) |
| *data_buff++ = inb(data_reg); /* unrolling this loop does not seem to help */ |
| data_valid = 1; |
| break; /* exit the timeout loop */ |
| } |
| sony_sleep(); /* data not ready, sleep a while */ |
| } |
| if (!data_valid) |
| return TIME_OUT; /* if we reach this stage */ |
| } |
| |
| /* read all the data, now read the status */ |
| if ((i = read_exec_status(status)) != 0) |
| return i; |
| return CDU535_BLOCK_SIZE * sector_count; |
| } /* seek_and_read_N_blocks() */ |
| |
| /**************************************************************************** |
| * int request_toc_data( Byte status[2], struct s535_sony_toc *toc ) |
| * |
| * Read in the table of contents data. Converts all the bcd data |
| * into integers in the toc structure. |
| ****************************************************************************/ |
| static int |
| request_toc_data(Byte status[2], struct s535_sony_toc *toc) |
| { |
| int to_status; |
| int i, j, n_tracks, track_no; |
| int first_track_num, last_track_num; |
| Byte cmd_no = 0xb2; |
| Byte track_address_buffer[5]; |
| |
| /* read the fixed portion of the table of contents */ |
| if ((to_status = do_sony_cmd(&cmd_no, 1, status, (Byte *) toc, 15, 1)) != 0) |
| return to_status; |
| |
| /* convert the data into integers so we can use them */ |
| first_track_num = bcd_to_int(toc->first_track_num); |
| last_track_num = bcd_to_int(toc->last_track_num); |
| n_tracks = last_track_num - first_track_num + 1; |
| |
| /* read each of the track address descriptors */ |
| for (i = 0; i < n_tracks; i++) { |
| /* read the descriptor into a temporary buffer */ |
| for (j = 0; j < 5; j++) { |
| if (read_result_reg(track_address_buffer + j) != 0) |
| return TIME_OUT; |
| if (j == 1) /* need to convert from bcd */ |
| track_no = bcd_to_int(track_address_buffer[j]); |
| } |
| /* copy the descriptor to proper location - sonycd.c just fills */ |
| memcpy(toc->tracks + i, track_address_buffer, 5); |
| } |
| return 0; |
| } /* request_toc_data() */ |
| |
| /*************************************************************************** |
| * int spin_up_drive( Byte status[2] ) |
| * |
| * Spin up the drive (unless it is already spinning). |
| ***************************************************************************/ |
| static int |
| spin_up_drive(Byte status[2]) |
| { |
| Byte cmd; |
| |
| /* first see if the drive is already spinning */ |
| cmd = SONY535_REQUEST_DRIVE_STATUS_1; |
| if (do_sony_cmd(&cmd, 1, status, NULL, 0, 0) != 0) |
| return TIME_OUT; |
| if ((status[0] & SONY535_STATUS1_NOT_SPINNING) == 0) |
| return 0; /* it's already spinning */ |
| |
| /* otherwise, give the spin-up command */ |
| cmd = SONY535_SPIN_UP; |
| return do_sony_cmd(&cmd, 1, status, NULL, 0, 0); |
| } |
| |
| /*--------------------end of SONY CDU535 very specific ---------------------*/ |
| |
| /* Convert from an integer 0-99 to BCD */ |
| static inline unsigned int |
| int_to_bcd(unsigned int val) |
| { |
| int retval; |
| |
| retval = (val / 10) << 4; |
| retval = retval | val % 10; |
| return retval; |
| } |
| |
| |
| /* Convert from BCD to an integer from 0-99 */ |
| static unsigned int |
| bcd_to_int(unsigned int bcd) |
| { |
| return (((bcd >> 4) & 0x0f) * 10) + (bcd & 0x0f); |
| } |
| |
| |
| /* |
| * Convert a logical sector value (like the OS would want to use for |
| * a block device) to an MSF format. |
| */ |
| static void |
| log_to_msf(unsigned int log, Byte *msf) |
| { |
| log = log + LOG_START_OFFSET; |
| msf[0] = int_to_bcd(log / 4500); |
| log = log % 4500; |
| msf[1] = int_to_bcd(log / 75); |
| msf[2] = int_to_bcd(log % 75); |
| } |
| |
| |
| /* |
| * Convert an MSF format to a logical sector. |
| */ |
| static unsigned int |
| msf_to_log(Byte *msf) |
| { |
| unsigned int log; |
| |
| |
| log = bcd_to_int(msf[2]); |
| log += bcd_to_int(msf[1]) * 75; |
| log += bcd_to_int(msf[0]) * 4500; |
| log = log - LOG_START_OFFSET; |
| |
| return log; |
| } |
| |
| |
| /* |
| * Take in integer size value and put it into a buffer like |
| * the drive would want to see a number-of-sector value. |
| */ |
| static void |
| size_to_buf(unsigned int size, Byte *buf) |
| { |
| buf[0] = size / 65536; |
| size = size % 65536; |
| buf[1] = size / 256; |
| buf[2] = size % 256; |
| } |
| |
| |
| /* |
| * The OS calls this to perform a read or write operation to the drive. |
| * Write obviously fail. Reads to a read ahead of sony_buffer_size |
| * bytes to help speed operations. This especially helps since the OS |
| * may use 1024 byte blocks and the drive uses 2048 byte blocks. Since most |
| * data access on a CD is done sequentially, this saves a lot of operations. |
| */ |
| static void |
| do_cdu535_request(request_queue_t * q) |
| { |
| struct request *req; |
| unsigned int read_size; |
| int block; |
| int nsect; |
| int copyoff; |
| int spin_up_retry; |
| Byte params[10]; |
| Byte status[2]; |
| Byte cmd[2]; |
| |
| while (1) { |
| req = elv_next_request(q); |
| if (!req) |
| return; |
| |
| block = req->sector; |
| nsect = req->nr_sectors; |
| if (!blk_fs_request(req)) { |
| end_request(req, 0); |
| continue; |
| } |
| if (rq_data_dir(req) == WRITE) { |
| end_request(req, 0); |
| continue; |
| } |
| /* |
| * If the block address is invalid or the request goes beyond |
| * the end of the media, return an error. |
| */ |
| if (sony_toc->lead_out_start_lba <= (block/4)) { |
| end_request(req, 0); |
| return; |
| } |
| if (sony_toc->lead_out_start_lba <= ((block + nsect) / 4)) { |
| end_request(req, 0); |
| return; |
| } |
| while (0 < nsect) { |
| /* |
| * If the requested sector is not currently in |
| * the read-ahead buffer, it must be read in. |
| */ |
| if ((block < sony_first_block) || (sony_last_block < block)) { |
| sony_first_block = (block / 4) * 4; |
| log_to_msf(block / 4, params); |
| |
| /* |
| * If the full read-ahead would go beyond the end of the media, trim |
| * it back to read just till the end of the media. |
| */ |
| if (sony_toc->lead_out_start_lba <= ((block / 4) + sony_buffer_sectors)) { |
| sony_last_block = (sony_toc->lead_out_start_lba * 4) - 1; |
| read_size = sony_toc->lead_out_start_lba - (block / 4); |
| } else { |
| sony_last_block = sony_first_block + (sony_buffer_sectors * 4) - 1; |
| read_size = sony_buffer_sectors; |
| } |
| size_to_buf(read_size, ¶ms[3]); |
| |
| /* |
| * Read the data. If the drive was not spinning, |
| * spin it up and try some more. |
| */ |
| for (spin_up_retry=0 ;; ++spin_up_retry) { |
| /* This loop has been modified to support the Sony |
| * CDU-510/515 series, thanks to Claudio Porfiri |
| * <C.Porfiri@nisms.tei.ericsson.se>. |
| */ |
| /* |
| * This part is to deal with very slow hardware. We |
| * try at most MAX_SPINUP_RETRY times to read the same |
| * block. A check for seek_and_read_N_blocks' result is |
| * performed; if the result is wrong, the CDROM's engine |
| * is restarted and the operation is tried again. |
| */ |
| /* |
| * 1995-06-01: The system got problems when downloading |
| * from Slackware CDROM, the problem seems to be: |
| * seek_and_read_N_blocks returns BAD_STATUS and we |
| * should wait for a while before retrying, so a new |
| * part was added to discriminate the return value from |
| * seek_and_read_N_blocks for the various cases. |
| */ |
| int readStatus = seek_and_read_N_blocks(params, read_size, |
| status, sony_buffer, (read_size * CDU535_BLOCK_SIZE)); |
| if (0 <= readStatus) /* Good data; common case, placed first */ |
| break; |
| if (readStatus == NO_ROOM || spin_up_retry == MAX_SPINUP_RETRY) { |
| /* give up */ |
| if (readStatus == NO_ROOM) |
| printk(CDU535_MESSAGE_NAME " No room to read from CD\n"); |
| else |
| printk(CDU535_MESSAGE_NAME " Read error: 0x%.2x\n", |
| status[0]); |
| sony_first_block = -1; |
| sony_last_block = -1; |
| end_request(req, 0); |
| return; |
| } |
| if (readStatus == BAD_STATUS) { |
| /* Sleep for a while, then retry */ |
| set_current_state(TASK_INTERRUPTIBLE); |
| spin_unlock_irq(&sonycd535_lock); |
| schedule_timeout(RETRY_FOR_BAD_STATUS*HZ/10); |
| spin_lock_irq(&sonycd535_lock); |
| } |
| #if DEBUG > 0 |
| printk(CDU535_MESSAGE_NAME |
| " debug: calling spin up when reading data!\n"); |
| #endif |
| cmd[0] = SONY535_SPIN_UP; |
| do_sony_cmd(cmd, 1, status, NULL, 0, 0); |
| } |
| } |
| /* |
| * The data is in memory now, copy it to the buffer and advance to the |
| * next block to read. |
| */ |
| copyoff = block - sony_first_block; |
| memcpy(req->buffer, |
| sony_buffer[copyoff / 4] + 512 * (copyoff % 4), 512); |
| |
| block += 1; |
| nsect -= 1; |
| req->buffer += 512; |
| } |
| |
| end_request(req, 1); |
| } |
| } |
| |
| /* |
| * Read the table of contents from the drive and set sony_toc_read if |
| * successful. |
| */ |
| static void |
| sony_get_toc(void) |
| { |
| Byte status[2]; |
| if (!sony_toc_read) { |
| /* do not call check_drive_status() from here since it can call this routine */ |
| if (request_toc_data(status, sony_toc) < 0) |
| return; |
| sony_toc->lead_out_start_lba = msf_to_log(sony_toc->lead_out_start_msf); |
| sony_toc_read = 1; |
| } |
| } |
| |
| |
| /* |
| * Search for a specific track in the table of contents. track is |
| * passed in bcd format |
| */ |
| static int |
| find_track(int track) |
| { |
| int i; |
| int num_tracks; |
| |
| |
| num_tracks = bcd_to_int(sony_toc->last_track_num) - |
| bcd_to_int(sony_toc->first_track_num) + 1; |
| for (i = 0; i < num_tracks; i++) { |
| if (sony_toc->tracks[i].track == track) { |
| return i; |
| } |
| } |
| |
| return -1; |
| } |
| |
| /* |
| * Read the subcode and put it int last_sony_subcode for future use. |
| */ |
| static int |
| read_subcode(void) |
| { |
| Byte cmd = SONY535_REQUEST_SUB_Q_DATA; |
| Byte status[2]; |
| int dsc_status; |
| |
| if (check_drive_status() != 0) |
| return -EIO; |
| |
| if ((dsc_status = do_sony_cmd(&cmd, 1, status, (Byte *) last_sony_subcode, |
| sizeof(struct s535_sony_subcode), 1)) != 0) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x, %d (read_subcode)\n", |
| status[0], dsc_status); |
| return -EIO; |
| } |
| return 0; |
| } |
| |
| |
| /* |
| * Get the subchannel info like the CDROMSUBCHNL command wants to see it. If |
| * the drive is playing, the subchannel needs to be read (since it would be |
| * changing). If the drive is paused or completed, the subcode information has |
| * already been stored, just use that. The ioctl call wants things in decimal |
| * (not BCD), so all the conversions are done. |
| */ |
| static int |
| sony_get_subchnl_info(void __user *arg) |
| { |
| struct cdrom_subchnl schi; |
| |
| /* Get attention stuff */ |
| if (check_drive_status() != 0) |
| return -EIO; |
| |
| sony_get_toc(); |
| if (!sony_toc_read) { |
| return -EIO; |
| } |
| if (copy_from_user(&schi, arg, sizeof schi)) |
| return -EFAULT; |
| |
| switch (sony_audio_status) { |
| case CDROM_AUDIO_PLAY: |
| if (read_subcode() < 0) { |
| return -EIO; |
| } |
| break; |
| |
| case CDROM_AUDIO_PAUSED: |
| case CDROM_AUDIO_COMPLETED: |
| break; |
| |
| case CDROM_AUDIO_NO_STATUS: |
| schi.cdsc_audiostatus = sony_audio_status; |
| if (copy_to_user(arg, &schi, sizeof schi)) |
| return -EFAULT; |
| return 0; |
| break; |
| |
| case CDROM_AUDIO_INVALID: |
| case CDROM_AUDIO_ERROR: |
| default: |
| return -EIO; |
| } |
| |
| schi.cdsc_audiostatus = sony_audio_status; |
| schi.cdsc_adr = last_sony_subcode->address; |
| schi.cdsc_ctrl = last_sony_subcode->control; |
| schi.cdsc_trk = bcd_to_int(last_sony_subcode->track_num); |
| schi.cdsc_ind = bcd_to_int(last_sony_subcode->index_num); |
| if (schi.cdsc_format == CDROM_MSF) { |
| schi.cdsc_absaddr.msf.minute = bcd_to_int(last_sony_subcode->abs_msf[0]); |
| schi.cdsc_absaddr.msf.second = bcd_to_int(last_sony_subcode->abs_msf[1]); |
| schi.cdsc_absaddr.msf.frame = bcd_to_int(last_sony_subcode->abs_msf[2]); |
| |
| schi.cdsc_reladdr.msf.minute = bcd_to_int(last_sony_subcode->rel_msf[0]); |
| schi.cdsc_reladdr.msf.second = bcd_to_int(last_sony_subcode->rel_msf[1]); |
| schi.cdsc_reladdr.msf.frame = bcd_to_int(last_sony_subcode->rel_msf[2]); |
| } else if (schi.cdsc_format == CDROM_LBA) { |
| schi.cdsc_absaddr.lba = msf_to_log(last_sony_subcode->abs_msf); |
| schi.cdsc_reladdr.lba = msf_to_log(last_sony_subcode->rel_msf); |
| } |
| return copy_to_user(arg, &schi, sizeof schi) ? -EFAULT : 0; |
| } |
| |
| |
| /* |
| * The big ugly ioctl handler. |
| */ |
| static int |
| cdu_ioctl(struct inode *inode, |
| struct file *file, |
| unsigned int cmd, |
| unsigned long arg) |
| { |
| Byte status[2]; |
| Byte cmd_buff[10], params[10]; |
| int i; |
| int dsc_status; |
| void __user *argp = (void __user *)arg; |
| |
| if (check_drive_status() != 0) |
| return -EIO; |
| |
| switch (cmd) { |
| case CDROMSTART: /* Spin up the drive */ |
| if (spin_up_drive(status) < 0) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMSTART)\n", |
| status[0]); |
| return -EIO; |
| } |
| return 0; |
| break; |
| |
| case CDROMSTOP: /* Spin down the drive */ |
| cmd_buff[0] = SONY535_HOLD; |
| do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0); |
| |
| /* |
| * Spin the drive down, ignoring the error if the disk was |
| * already not spinning. |
| */ |
| sony_audio_status = CDROM_AUDIO_NO_STATUS; |
| cmd_buff[0] = SONY535_SPIN_DOWN; |
| dsc_status = do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0); |
| if (((dsc_status < 0) && (dsc_status != BAD_STATUS)) || |
| ((status[0] & ~(SONY535_STATUS1_NOT_SPINNING)) != 0)) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMSTOP)\n", |
| status[0]); |
| return -EIO; |
| } |
| return 0; |
| break; |
| |
| case CDROMPAUSE: /* Pause the drive */ |
| cmd_buff[0] = SONY535_HOLD; /* CDU-31 driver uses AUDIO_STOP, not pause */ |
| if (do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0) != 0) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMPAUSE)\n", |
| status[0]); |
| return -EIO; |
| } |
| /* Get the current position and save it for resuming */ |
| if (read_subcode() < 0) { |
| return -EIO; |
| } |
| cur_pos_msf[0] = last_sony_subcode->abs_msf[0]; |
| cur_pos_msf[1] = last_sony_subcode->abs_msf[1]; |
| cur_pos_msf[2] = last_sony_subcode->abs_msf[2]; |
| sony_audio_status = CDROM_AUDIO_PAUSED; |
| return 0; |
| break; |
| |
| case CDROMRESUME: /* Start the drive after being paused */ |
| set_drive_mode(SONY535_AUDIO_DRIVE_MODE, status); |
| |
| if (sony_audio_status != CDROM_AUDIO_PAUSED) { |
| return -EINVAL; |
| } |
| spin_up_drive(status); |
| |
| /* Start the drive at the saved position. */ |
| cmd_buff[0] = SONY535_PLAY_AUDIO; |
| cmd_buff[1] = 0; /* play back starting at this address */ |
| cmd_buff[2] = cur_pos_msf[0]; |
| cmd_buff[3] = cur_pos_msf[1]; |
| cmd_buff[4] = cur_pos_msf[2]; |
| cmd_buff[5] = SONY535_PLAY_AUDIO; |
| cmd_buff[6] = 2; /* set ending address */ |
| cmd_buff[7] = final_pos_msf[0]; |
| cmd_buff[8] = final_pos_msf[1]; |
| cmd_buff[9] = final_pos_msf[2]; |
| if ((do_sony_cmd(cmd_buff, 5, status, NULL, 0, 0) != 0) || |
| (do_sony_cmd(cmd_buff + 5, 5, status, NULL, 0, 0) != 0)) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMRESUME)\n", |
| status[0]); |
| return -EIO; |
| } |
| sony_audio_status = CDROM_AUDIO_PLAY; |
| return 0; |
| break; |
| |
| case CDROMPLAYMSF: /* Play starting at the given MSF address. */ |
| if (copy_from_user(params, argp, 6)) |
| return -EFAULT; |
| spin_up_drive(status); |
| set_drive_mode(SONY535_AUDIO_DRIVE_MODE, status); |
| |
| /* The parameters are given in int, must be converted */ |
| for (i = 0; i < 3; i++) { |
| cmd_buff[2 + i] = int_to_bcd(params[i]); |
| cmd_buff[7 + i] = int_to_bcd(params[i + 3]); |
| } |
| cmd_buff[0] = SONY535_PLAY_AUDIO; |
| cmd_buff[1] = 0; /* play back starting at this address */ |
| /* cmd_buff[2-4] are filled in for loop above */ |
| cmd_buff[5] = SONY535_PLAY_AUDIO; |
| cmd_buff[6] = 2; /* set ending address */ |
| /* cmd_buff[7-9] are filled in for loop above */ |
| if ((do_sony_cmd(cmd_buff, 5, status, NULL, 0, 0) != 0) || |
| (do_sony_cmd(cmd_buff + 5, 5, status, NULL, 0, 0) != 0)) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMPLAYMSF)\n", |
| status[0]); |
| return -EIO; |
| } |
| /* Save the final position for pauses and resumes */ |
| final_pos_msf[0] = cmd_buff[7]; |
| final_pos_msf[1] = cmd_buff[8]; |
| final_pos_msf[2] = cmd_buff[9]; |
| sony_audio_status = CDROM_AUDIO_PLAY; |
| return 0; |
| break; |
| |
| case CDROMREADTOCHDR: /* Read the table of contents header */ |
| { |
| struct cdrom_tochdr __user *hdr = argp; |
| struct cdrom_tochdr loc_hdr; |
| |
| sony_get_toc(); |
| if (!sony_toc_read) |
| return -EIO; |
| loc_hdr.cdth_trk0 = bcd_to_int(sony_toc->first_track_num); |
| loc_hdr.cdth_trk1 = bcd_to_int(sony_toc->last_track_num); |
| if (copy_to_user(hdr, &loc_hdr, sizeof *hdr)) |
| return -EFAULT; |
| } |
| return 0; |
| break; |
| |
| case CDROMREADTOCENTRY: /* Read a given table of contents entry */ |
| { |
| struct cdrom_tocentry __user *entry = argp; |
| struct cdrom_tocentry loc_entry; |
| int track_idx; |
| Byte *msf_val = NULL; |
| |
| sony_get_toc(); |
| if (!sony_toc_read) { |
| return -EIO; |
| } |
| |
| if (copy_from_user(&loc_entry, entry, sizeof loc_entry)) |
| return -EFAULT; |
| |
| /* Lead out is handled separately since it is special. */ |
| if (loc_entry.cdte_track == CDROM_LEADOUT) { |
| loc_entry.cdte_adr = 0 /*sony_toc->address2 */ ; |
| loc_entry.cdte_ctrl = sony_toc->control2; |
| msf_val = sony_toc->lead_out_start_msf; |
| } else { |
| track_idx = find_track(int_to_bcd(loc_entry.cdte_track)); |
| if (track_idx < 0) |
| return -EINVAL; |
| loc_entry.cdte_adr = 0 /*sony_toc->tracks[track_idx].address */ ; |
| loc_entry.cdte_ctrl = sony_toc->tracks[track_idx].control; |
| msf_val = sony_toc->tracks[track_idx].track_start_msf; |
| } |
| |
| /* Logical buffer address or MSF format requested? */ |
| if (loc_entry.cdte_format == CDROM_LBA) { |
| loc_entry.cdte_addr.lba = msf_to_log(msf_val); |
| } else if (loc_entry.cdte_format == CDROM_MSF) { |
| loc_entry.cdte_addr.msf.minute = bcd_to_int(*msf_val); |
| loc_entry.cdte_addr.msf.second = bcd_to_int(*(msf_val + 1)); |
| loc_entry.cdte_addr.msf.frame = bcd_to_int(*(msf_val + 2)); |
| } |
| if (copy_to_user(entry, &loc_entry, sizeof *entry)) |
| return -EFAULT; |
| } |
| return 0; |
| break; |
| |
| case CDROMPLAYTRKIND: /* Play a track. This currently ignores index. */ |
| { |
| struct cdrom_ti ti; |
| int track_idx; |
| |
| sony_get_toc(); |
| if (!sony_toc_read) |
| return -EIO; |
| |
| if (copy_from_user(&ti, argp, sizeof ti)) |
| return -EFAULT; |
| if ((ti.cdti_trk0 < sony_toc->first_track_num) |
| || (sony_toc->last_track_num < ti.cdti_trk0) |
| || (ti.cdti_trk1 < ti.cdti_trk0)) { |
| return -EINVAL; |
| } |
| track_idx = find_track(int_to_bcd(ti.cdti_trk0)); |
| if (track_idx < 0) |
| return -EINVAL; |
| params[1] = sony_toc->tracks[track_idx].track_start_msf[0]; |
| params[2] = sony_toc->tracks[track_idx].track_start_msf[1]; |
| params[3] = sony_toc->tracks[track_idx].track_start_msf[2]; |
| /* |
| * If we want to stop after the last track, use the lead-out |
| * MSF to do that. |
| */ |
| if (bcd_to_int(sony_toc->last_track_num) <= ti.cdti_trk1) { |
| log_to_msf(msf_to_log(sony_toc->lead_out_start_msf) - 1, |
| &(params[4])); |
| } else { |
| track_idx = find_track(int_to_bcd(ti.cdti_trk1 + 1)); |
| if (track_idx < 0) |
| return -EINVAL; |
| log_to_msf(msf_to_log(sony_toc->tracks[track_idx].track_start_msf) - 1, |
| &(params[4])); |
| } |
| params[0] = 0x03; |
| |
| spin_up_drive(status); |
| |
| set_drive_mode(SONY535_AUDIO_DRIVE_MODE, status); |
| |
| /* Start the drive at the saved position. */ |
| cmd_buff[0] = SONY535_PLAY_AUDIO; |
| cmd_buff[1] = 0; /* play back starting at this address */ |
| cmd_buff[2] = params[1]; |
| cmd_buff[3] = params[2]; |
| cmd_buff[4] = params[3]; |
| cmd_buff[5] = SONY535_PLAY_AUDIO; |
| cmd_buff[6] = 2; /* set ending address */ |
| cmd_buff[7] = params[4]; |
| cmd_buff[8] = params[5]; |
| cmd_buff[9] = params[6]; |
| if ((do_sony_cmd(cmd_buff, 5, status, NULL, 0, 0) != 0) || |
| (do_sony_cmd(cmd_buff + 5, 5, status, NULL, 0, 0) != 0)) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMPLAYTRKIND)\n", |
| status[0]); |
| printk("... Params: %x %x %x %x %x %x %x\n", |
| params[0], params[1], params[2], |
| params[3], params[4], params[5], params[6]); |
| return -EIO; |
| } |
| /* Save the final position for pauses and resumes */ |
| final_pos_msf[0] = params[4]; |
| final_pos_msf[1] = params[5]; |
| final_pos_msf[2] = params[6]; |
| sony_audio_status = CDROM_AUDIO_PLAY; |
| return 0; |
| } |
| |
| case CDROMSUBCHNL: /* Get subchannel info */ |
| return sony_get_subchnl_info(argp); |
| |
| case CDROMVOLCTRL: /* Volume control. What volume does this change, anyway? */ |
| { |
| struct cdrom_volctrl volctrl; |
| |
| if (copy_from_user(&volctrl, argp, sizeof volctrl)) |
| return -EFAULT; |
| cmd_buff[0] = SONY535_SET_VOLUME; |
| cmd_buff[1] = volctrl.channel0; |
| cmd_buff[2] = volctrl.channel1; |
| if (do_sony_cmd(cmd_buff, 3, status, NULL, 0, 0) != 0) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMVOLCTRL)\n", |
| status[0]); |
| return -EIO; |
| } |
| } |
| return 0; |
| |
| case CDROMEJECT: /* Eject the drive */ |
| cmd_buff[0] = SONY535_STOP; |
| do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0); |
| cmd_buff[0] = SONY535_SPIN_DOWN; |
| do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0); |
| |
| sony_audio_status = CDROM_AUDIO_INVALID; |
| cmd_buff[0] = SONY535_EJECT_CADDY; |
| if (do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0) != 0) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (CDROMEJECT)\n", |
| status[0]); |
| return -EIO; |
| } |
| return 0; |
| break; |
| |
| default: |
| return -EINVAL; |
| } |
| } |
| |
| |
| /* |
| * Open the drive for operations. Spin the drive up and read the table of |
| * contents if these have not already been done. |
| */ |
| static int |
| cdu_open(struct inode *inode, |
| struct file *filp) |
| { |
| Byte status[2], cmd_buff[2]; |
| |
| if (sony_inuse) |
| return -EBUSY; |
| if (check_drive_status() != 0) |
| return -EIO; |
| sony_inuse = 1; |
| |
| if (spin_up_drive(status) != 0) { |
| printk(CDU535_MESSAGE_NAME " error 0x%.2x (cdu_open, spin up)\n", |
| status[0]); |
| sony_inuse = 0; |
| return -EIO; |
| } |
| sony_get_toc(); |
| if (!sony_toc_read) { |
| cmd_buff[0] = SONY535_SPIN_DOWN; |
| do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0); |
| sony_inuse = 0; |
| return -EIO; |
| } |
| check_disk_change(inode->i_bdev); |
| sony_usage++; |
| |
| #ifdef LOCK_DOORS |
| /* disable the eject button while mounted */ |
| cmd_buff[0] = SONY535_DISABLE_EJECT_BUTTON; |
| do_sony_cmd(cmd_buff, 1, status, NULL, 0, 0); |
| #endif |
| |
| return 0; |
| } |
| |
| |
| /* |
| * Close the drive. Spin it down if no task is using it. The spin |
| * down will fail if playing audio, so audio play is OK. |
| */ |
| static int |
| cdu_release(struct inode *inode, |
| struct file *filp) |
| { |
| Byte status[2], cmd_no; |
| |
| sony_inuse = 0; |
| |
| if (0 < sony_usage) { |
| sony_usage--; |
| } |
| if (sony_usage == 0) { |
| check_drive_status(); |
| |
| if (sony_audio_status != CDROM_AUDIO_PLAY) { |
| cmd_no = SONY535_SPIN_DOWN; |
| do_sony_cmd(&cmd_no, 1, status, NULL, 0, 0); |
| } |
| #ifdef LOCK_DOORS |
| /* enable the eject button after umount */ |
| cmd_no = SONY535_ENABLE_EJECT_BUTTON; |
| do_sony_cmd(&cmd_no, 1, status, NULL, 0, 0); |
| #endif |
| } |
| return 0; |
| } |
| |
| static struct block_device_operations cdu_fops = |
| { |
| .owner = THIS_MODULE, |
| .open = cdu_open, |
| .release = cdu_release, |
| .ioctl = cdu_ioctl, |
| .media_changed = cdu535_check_media_change, |
| }; |
| |
| static struct gendisk *cdu_disk; |
| |
| /* |
| * Initialize the driver. |
| */ |
| static int __init sony535_init(void) |
| { |
| struct s535_sony_drive_config drive_config; |
| Byte cmd_buff[3]; |
| Byte ret_buff[2]; |
| Byte status[2]; |
| unsigned long snap; |
| int got_result = 0; |
| int tmp_irq; |
| int i; |
| int err; |
| |
| /* Setting the base I/O address to 0 will disable it. */ |
| if ((sony535_cd_base_io == 0xffff)||(sony535_cd_base_io == 0)) |
| return 0; |
| |
| /* Set up all the register locations */ |
| result_reg = sony535_cd_base_io; |
| command_reg = sony535_cd_base_io; |
| data_reg = sony535_cd_base_io + 1; |
| read_status_reg = sony535_cd_base_io + 2; |
| select_unit_reg = sony535_cd_base_io + 3; |
| |
| #ifndef USE_IRQ |
| sony535_irq_used = 0; /* polling only until this is ready... */ |
| #endif |
| /* we need to poll until things get initialized */ |
| tmp_irq = sony535_irq_used; |
| sony535_irq_used = 0; |
| |
| #if DEBUG > 0 |
| printk(KERN_INFO CDU535_MESSAGE_NAME ": probing base address %03X\n", |
| sony535_cd_base_io); |
| #endif |
| /* look for the CD-ROM, follows the procedure in the DOS driver */ |
| inb(select_unit_reg); |
| /* wait for 40 18 Hz ticks (reverse-engineered from DOS driver) */ |
| schedule_timeout_interruptible((HZ+17)*40/18); |
| inb(result_reg); |
| |
| outb(0, read_status_reg); /* does a reset? */ |
| snap = jiffies; |
| while (jiffies-snap < SONY_JIFFIES_TIMEOUT) { |
| select_unit(0); |
| if (inb(result_reg) != 0xff) { |
| got_result = 1; |
| break; |
| } |
| sony_sleep(); |
| } |
| |
| if (!got_result || check_drive_status() == TIME_OUT) |
| goto Enodev; |
| |
| /* CD-ROM drive responded -- get the drive configuration */ |
| cmd_buff[0] = SONY535_INQUIRY; |
| if (do_sony_cmd(cmd_buff, 1, status, (Byte *)&drive_config, 28, 1) != 0) |
| goto Enodev; |
| |
| /* was able to get the configuration, |
| * set drive mode as rest of init |
| */ |
| #if DEBUG > 0 |
| /* 0x50 == CADDY_NOT_INSERTED | NOT_SPINNING */ |
| if ( (status[0] & 0x7f) != 0 && (status[0] & 0x7f) != 0x50 ) |
| printk(CDU535_MESSAGE_NAME |
| "Inquiry command returned status = 0x%x\n", status[0]); |
| #endif |
| /* now ready to use interrupts, if available */ |
| sony535_irq_used = tmp_irq; |
| |
| /* A negative sony535_irq_used will attempt an autoirq. */ |
| if (sony535_irq_used < 0) { |
| unsigned long irq_mask, delay; |
| |
| irq_mask = probe_irq_on(); |
| enable_interrupts(); |
| outb(0, read_status_reg); /* does a reset? */ |
| delay = jiffies + HZ/10; |
| while (time_before(jiffies, delay)) ; |
| |
| sony535_irq_used = probe_irq_off(irq_mask); |
| disable_interrupts(); |
| } |
| if (sony535_irq_used > 0) { |
| if (request_irq(sony535_irq_used, cdu535_interrupt, |
| IRQF_DISABLED, CDU535_HANDLE, NULL)) { |
| printk("Unable to grab IRQ%d for the " CDU535_MESSAGE_NAME |
| " driver; polling instead.\n", sony535_irq_used); |
| sony535_irq_used = 0; |
| } |
| } |
| cmd_buff[0] = SONY535_SET_DRIVE_MODE; |
| cmd_buff[1] = 0x0; /* default audio */ |
| if (do_sony_cmd(cmd_buff, 2, status, ret_buff, 1, 1) != 0) |
| goto Enodev_irq; |
| |
| /* set the drive mode successful, we are set! */ |
| sony_buffer_size = SONY535_BUFFER_SIZE; |
| sony_buffer_sectors = sony_buffer_size / CDU535_BLOCK_SIZE; |
| |
| printk(KERN_INFO CDU535_MESSAGE_NAME " I/F CDROM : %8.8s %16.16s %4.4s", |
| drive_config.vendor_id, |
| drive_config.product_id, |
| drive_config.product_rev_level); |
| printk(" base address %03X, ", sony535_cd_base_io); |
| if (tmp_irq > 0) |
| printk("IRQ%d, ", tmp_irq); |
| printk("using %d byte buffer\n", sony_buffer_size); |
| |
| if (register_blkdev(MAJOR_NR, CDU535_HANDLE)) { |
| err = -EIO; |
| goto out1; |
| } |
| sonycd535_queue = blk_init_queue(do_cdu535_request, &sonycd535_lock); |
| if (!sonycd535_queue) { |
| err = -ENOMEM; |
| goto out1a; |
| } |
| |
| blk_queue_hardsect_size(sonycd535_queue, CDU535_BLOCK_SIZE); |
| sony_toc = kmalloc(sizeof(struct s535_sony_toc), GFP_KERNEL); |
| err = -ENOMEM; |
| if (!sony_toc) |
| goto out2; |
| last_sony_subcode = kmalloc(sizeof(struct s535_sony_subcode), GFP_KERNEL); |
| if (!last_sony_subcode) |
| goto out3; |
| sony_buffer = kmalloc(sizeof(Byte *) * sony_buffer_sectors, GFP_KERNEL); |
| if (!sony_buffer) |
| goto out4; |
| for (i = 0; i < sony_buffer_sectors; i++) { |
| sony_buffer[i] = kmalloc(CDU535_BLOCK_SIZE, GFP_KERNEL); |
| if (!sony_buffer[i]) { |
| while (--i>=0) |
| kfree(sony_buffer[i]); |
| goto out5; |
| } |
| } |
| initialized = 1; |
| |
| cdu_disk = alloc_disk(1); |
| if (!cdu_disk) |
| goto out6; |
| cdu_disk->major = MAJOR_NR; |
| cdu_disk->first_minor = 0; |
| cdu_disk->fops = &cdu_fops; |
| sprintf(cdu_disk->disk_name, "cdu"); |
| |
| if (!request_region(sony535_cd_base_io, 4, CDU535_HANDLE)) { |
| printk(KERN_WARNING"sonycd535: Unable to request region 0x%x\n", |
| sony535_cd_base_io); |
| goto out7; |
| } |
| cdu_disk->queue = sonycd535_queue; |
| add_disk(cdu_disk); |
| return 0; |
| |
| out7: |
| put_disk(cdu_disk); |
| out6: |
| for (i = 0; i < sony_buffer_sectors; i++) |
| kfree(sony_buffer[i]); |
| out5: |
| kfree(sony_buffer); |
| out4: |
| kfree(last_sony_subcode); |
| out3: |
| kfree(sony_toc); |
| out2: |
| blk_cleanup_queue(sonycd535_queue); |
| out1a: |
| unregister_blkdev(MAJOR_NR, CDU535_HANDLE); |
| out1: |
| if (sony535_irq_used) |
| free_irq(sony535_irq_used, NULL); |
| return err; |
| Enodev_irq: |
| if (sony535_irq_used) |
| free_irq(sony535_irq_used, NULL); |
| Enodev: |
| printk("Did not find a " CDU535_MESSAGE_NAME " drive\n"); |
| return -EIO; |
| } |
| |
| #ifndef MODULE |
| |
| /* |
| * accept "kernel command line" parameters |
| * (added by emoenke@gwdg.de) |
| * |
| * use: tell LILO: |
| * sonycd535=0x320 |
| * |
| * the address value has to be the existing CDROM port address. |
| */ |
| static int __init |
| sonycd535_setup(char *strings) |
| { |
| int ints[3]; |
| (void)get_options(strings, ARRAY_SIZE(ints), ints); |
| /* if IRQ change and default io base desired, |
| * then call with io base of 0 |
| */ |
| if (ints[0] > 0) |
| if (ints[1] != 0) |
| sony535_cd_base_io = ints[1]; |
| if (ints[0] > 1) |
| sony535_irq_used = ints[2]; |
| if ((strings != NULL) && (*strings != '\0')) |
| printk(CDU535_MESSAGE_NAME |
| ": Warning: Unknown interface type: %s\n", strings); |
| |
| return 1; |
| } |
| |
| __setup("sonycd535=", sonycd535_setup); |
| |
| #endif /* MODULE */ |
| |
| static void __exit |
| sony535_exit(void) |
| { |
| int i; |
| |
| release_region(sony535_cd_base_io, 4); |
| for (i = 0; i < sony_buffer_sectors; i++) |
| kfree(sony_buffer[i]); |
| kfree(sony_buffer); |
| kfree(last_sony_subcode); |
| kfree(sony_toc); |
| del_gendisk(cdu_disk); |
| put_disk(cdu_disk); |
| blk_cleanup_queue(sonycd535_queue); |
| if (unregister_blkdev(MAJOR_NR, CDU535_HANDLE) == -EINVAL) |
| printk("Uh oh, couldn't unregister " CDU535_HANDLE "\n"); |
| else |
| printk(KERN_INFO CDU535_HANDLE " module released\n"); |
| } |
| |
| module_init(sony535_init); |
| module_exit(sony535_exit); |
| |
| |
| MODULE_LICENSE("GPL"); |
| MODULE_ALIAS_BLOCKDEV_MAJOR(CDU535_CDROM_MAJOR); |