Commit 8179c97e authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] ide-tape update

From: Bartlomiej Zolnierkiewicz <B.Zolnierkiewicz@elka.pw.edu.pl>,
      Stuart Hayes <stuart_hayes@dell.com>

- Check drive's write protect bit, try to return appropriate
  errors when attempting to write a write-protected tape.

- Moved "idetape_read_position" call in idetape_chrdev_open
  after the "wait_ready" call.

- Added IDETAPE_MEDIUM_PRESENT flag so driver would know
  not to rewind tape after ejecting it.

- Fixed bug with ide_abort_pipeline (it was deleting stages
  from tape->next_stage to end, instead of from
  new_last_stage->next (tape->next_stage was set to NULL
  by idetape_discard_read_pipeline before calling!).

- Made improvements to idetape_wait_ready.

- Added a few comments here and there.

- Made MTOFFL unlock tape drive door before attempting to eject.

- Added fixes to get Seagate STT3401A Travan working:
  Handle drives that don't support 0-length reads/writes increased timeout
  (retension takes ~10 minutes before irq is returned).
  Fixed request mode page packet command byte 3.

Also remove code depending on NO_LONGER_REQUIRED to match 2.4.x (me).
parent 14209d06
/* /*
* linux/drivers/ide/ide-tape.c Version 1.17b Oct, 2002 * linux/drivers/ide/ide-tape.c Version 1.18 Nov, 2003
* *
* Copyright (C) 1995 - 1999 Gadi Oxman <gadio@netvision.net.il> * Copyright (C) 1995 - 1999 Gadi Oxman <gadio@netvision.net.il>
* *
...@@ -422,7 +422,7 @@ ...@@ -422,7 +422,7 @@
* sharing a (fast) ATA-2 disk with any (slow) new ATAPI device. * sharing a (fast) ATA-2 disk with any (slow) new ATAPI device.
*/ */
#define IDETAPE_VERSION "1.17b-ac1" #define IDETAPE_VERSION "1.18"
#include <linux/config.h> #include <linux/config.h>
#include <linux/module.h> #include <linux/module.h>
...@@ -450,9 +450,6 @@ ...@@ -450,9 +450,6 @@
#include <asm/unaligned.h> #include <asm/unaligned.h>
#include <asm/bitops.h> #include <asm/bitops.h>
#define NO_LONGER_REQUIRED (1)
/* /*
* OnStream support * OnStream support
*/ */
...@@ -652,9 +649,11 @@ typedef struct { ...@@ -652,9 +649,11 @@ typedef struct {
#define IDETAPE_PC_STACK (10 + IDETAPE_MAX_PC_RETRIES) #define IDETAPE_PC_STACK (10 + IDETAPE_MAX_PC_RETRIES)
/* /*
* Some tape drives require a long irq timeout * Some drives (for example, Seagate STT3401A Travan) require a very long
* timeout, because they don't return an interrupt or clear their busy bit
* until after the command completes (even retension commands).
*/ */
#define IDETAPE_WAIT_CMD (60*HZ) #define IDETAPE_WAIT_CMD (900*HZ)
/* /*
* The following parameter is used to select the point in the internal * The following parameter is used to select the point in the internal
...@@ -1032,6 +1031,10 @@ typedef struct { ...@@ -1032,6 +1031,10 @@ typedef struct {
/* the door is currently locked */ /* the door is currently locked */
int door_locked; int door_locked;
/* the tape hardware is write protected */
char drv_write_prot;
/* the tape is write protected (hardware or opened as read-only) */
char write_prot;
/* /*
* OnStream flags * OnStream flags
...@@ -1164,6 +1167,8 @@ typedef struct { ...@@ -1164,6 +1167,8 @@ typedef struct {
#define IDETAPE_DRQ_INTERRUPT 6 /* DRQ interrupt device */ #define IDETAPE_DRQ_INTERRUPT 6 /* DRQ interrupt device */
#define IDETAPE_READ_ERROR 7 #define IDETAPE_READ_ERROR 7
#define IDETAPE_PIPELINE_ACTIVE 8 /* pipeline active */ #define IDETAPE_PIPELINE_ACTIVE 8 /* pipeline active */
/* 0 = no tape is loaded, so we don't rewind after ejecting */
#define IDETAPE_MEDIUM_PRESENT 9
/* /*
* Supported ATAPI tape drives packet commands * Supported ATAPI tape drives packet commands
...@@ -1665,6 +1670,20 @@ static void idetape_analyze_error (ide_drive_t *drive, idetape_request_sense_res ...@@ -1665,6 +1670,20 @@ static void idetape_analyze_error (ide_drive_t *drive, idetape_request_sense_res
idetape_update_buffers(pc); idetape_update_buffers(pc);
} }
/*
* If error was the result of a zero-length read or write command,
* with sense key=5, asc=0x22, ascq=0, let it slide. Some drives
* (i.e. Seagate STT3401A Travan) don't support 0-length read/writes.
*/
if ((pc->c[0] == IDETAPE_READ_CMD || pc->c[0] == IDETAPE_WRITE_CMD)
&& pc->c[4] == 0 && pc->c[3] == 0 && pc->c[2] == 0) { /* length==0 */
if (result->sense_key == 5) {
/* don't report an error, everything's ok */
pc->error = 0;
/* don't retry read/write */
set_bit(PC_ABORT, &pc->flags);
}
}
if (pc->c[0] == IDETAPE_READ_CMD && result->filemark) { if (pc->c[0] == IDETAPE_READ_CMD && result->filemark) {
pc->error = IDETAPE_ERROR_FILEMARK; pc->error = IDETAPE_ERROR_FILEMARK;
set_bit(PC_ABORT, &pc->flags); set_bit(PC_ABORT, &pc->flags);
...@@ -1805,10 +1824,15 @@ static void idetape_remove_stage_head (ide_drive_t *drive) ...@@ -1805,10 +1824,15 @@ static void idetape_remove_stage_head (ide_drive_t *drive)
} }
} }
static void idetape_abort_pipeline (ide_drive_t *drive, idetape_stage_t *last_stage) /*
* This will free all the pipeline stages starting from new_last_stage->next
* to the end of the list, and point tape->last_stage to new_last_stage.
*/
static void idetape_abort_pipeline(ide_drive_t *drive,
idetape_stage_t *new_last_stage)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_stage_t *stage = tape->next_stage; idetape_stage_t *stage = new_last_stage->next;
idetape_stage_t *nstage; idetape_stage_t *nstage;
#if IDETAPE_DEBUG_LOG #if IDETAPE_DEBUG_LOG
...@@ -1822,9 +1846,9 @@ static void idetape_abort_pipeline (ide_drive_t *drive, idetape_stage_t *last_st ...@@ -1822,9 +1846,9 @@ static void idetape_abort_pipeline (ide_drive_t *drive, idetape_stage_t *last_st
--tape->nr_pending_stages; --tape->nr_pending_stages;
stage = nstage; stage = nstage;
} }
tape->last_stage = last_stage; if (new_last_stage)
if (last_stage) new_last_stage->next = NULL;
last_stage->next = NULL; tape->last_stage = new_last_stage;
tape->next_stage = NULL; tape->next_stage = NULL;
} }
...@@ -2430,7 +2454,14 @@ static void idetape_create_mode_sense_cmd (idetape_pc_t *pc, u8 page_code) ...@@ -2430,7 +2454,14 @@ static void idetape_create_mode_sense_cmd (idetape_pc_t *pc, u8 page_code)
if (page_code != IDETAPE_BLOCK_DESCRIPTOR) if (page_code != IDETAPE_BLOCK_DESCRIPTOR)
pc->c[1] = 8; /* DBD = 1 - Don't return block descriptors */ pc->c[1] = 8; /* DBD = 1 - Don't return block descriptors */
pc->c[2] = page_code; pc->c[2] = page_code;
pc->c[3] = 255; /* Don't limit the returned information */ /*
* Changed pc->c[3] to 0 (255 will at best return unused info).
*
* For SCSI this byte is defined as subpage instead of high byte
* of length and some IDE drives seem to interpret it this way
* and return an error when 255 is used.
*/
pc->c[3] = 0;
pc->c[4] = 255; /* (We will just discard data in that case) */ pc->c[4] = 255; /* (We will just discard data in that case) */
if (page_code == IDETAPE_BLOCK_DESCRIPTOR) if (page_code == IDETAPE_BLOCK_DESCRIPTOR)
pc->request_transfer = 12; pc->request_transfer = 12;
...@@ -2544,8 +2575,9 @@ static ide_startstop_t idetape_media_access_finished (ide_drive_t *drive) ...@@ -2544,8 +2575,9 @@ static ide_startstop_t idetape_media_access_finished (ide_drive_t *drive)
if (status.b.dsc) { if (status.b.dsc) {
if (status.b.check) { if (status.b.check) {
/* Error detected */ /* Error detected */
printk(KERN_ERR "ide-tape: %s: I/O error, ",tape->name); if (pc->c[0] != IDETAPE_TEST_UNIT_READY_CMD)
printk(KERN_ERR "ide-tape: %s: I/O error, ",
tape->name);
/* Retry operation */ /* Retry operation */
return idetape_retry_pc(drive); return idetape_retry_pc(drive);
} }
...@@ -3295,25 +3327,28 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout) ...@@ -3295,25 +3327,28 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t pc; idetape_pc_t pc;
int load_attempted = 0;
/* /*
* Wait for the tape to become ready * Wait for the tape to become ready
*/ */
set_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags);
timeout += jiffies; timeout += jiffies;
while (time_before(jiffies, timeout)) { while (time_before(jiffies, timeout)) {
idetape_create_test_unit_ready_cmd(&pc); idetape_create_test_unit_ready_cmd(&pc);
if (!__idetape_queue_pc_tail(drive, &pc)) if (!__idetape_queue_pc_tail(drive, &pc))
return 0; return 0;
if (tape->sense_key == 2 && tape->asc == 4 && tape->ascq == 2) { if ((tape->sense_key == 2 && tape->asc == 4 && tape->ascq == 2)
|| (tape->asc == 0x3A)) { /* no media */
if (load_attempted)
return -ENOMEDIUM;
idetape_create_load_unload_cmd(drive, &pc, IDETAPE_LU_LOAD_MASK); idetape_create_load_unload_cmd(drive, &pc, IDETAPE_LU_LOAD_MASK);
__idetape_queue_pc_tail(drive, &pc); __idetape_queue_pc_tail(drive, &pc);
idetape_create_test_unit_ready_cmd(&pc); load_attempted = 1;
if (!__idetape_queue_pc_tail(drive, &pc)) /* not about to be ready */
return 0; } else if (!(tape->sense_key == 2 && tape->asc == 4 &&
}
if (!(tape->sense_key == 2 && tape->asc == 4 &&
(tape->ascq == 1 || tape->ascq == 8))) (tape->ascq == 1 || tape->ascq == 8)))
break; return -EIO;
current->state = TASK_INTERRUPTIBLE; current->state = TASK_INTERRUPTIBLE;
schedule_timeout(HZ / 10); schedule_timeout(HZ / 10);
} }
...@@ -3369,25 +3404,10 @@ static int idetape_read_position (ide_drive_t *drive) ...@@ -3369,25 +3404,10 @@ static int idetape_read_position (ide_drive_t *drive)
printk(KERN_INFO "ide-tape: Reached idetape_read_position\n"); printk(KERN_INFO "ide-tape: Reached idetape_read_position\n");
#endif /* IDETAPE_DEBUG_LOG */ #endif /* IDETAPE_DEBUG_LOG */
#ifdef NO_LONGER_REQUIRED
idetape_flush_tape_buffers(drive);
#endif
idetape_create_read_position_cmd(&pc); idetape_create_read_position_cmd(&pc);
if (idetape_queue_pc_tail(drive, &pc)) if (idetape_queue_pc_tail(drive, &pc))
return -1; return -1;
position = tape->first_frame_position; position = tape->first_frame_position;
#ifdef NO_LONGER_REQUIRED
if (tape->onstream) {
if ((position != tape->last_frame_position - tape->blocks_in_buffer) &&
(position != tape->last_frame_position + tape->blocks_in_buffer)) {
if (tape->blocks_in_buffer == 0) {
printk("ide-tape: %s: correcting read position %d, %d, %d\n", tape->name, position, tape->last_frame_position, tape->blocks_in_buffer);
position = tape->last_frame_position;
tape->first_frame_position = position;
}
}
}
#endif
return position; return position;
} }
...@@ -3436,6 +3456,8 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive) ...@@ -3436,6 +3456,8 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive)
if (tape->chrdev_direction != idetape_direction_read) if (tape->chrdev_direction != idetape_direction_read)
return 0; return 0;
/* Remove merge stage. */
cnt = tape->merge_stage_size / tape->tape_block_size; cnt = tape->merge_stage_size / tape->tape_block_size;
if (test_and_clear_bit(IDETAPE_FILEMARK, &tape->flags)) if (test_and_clear_bit(IDETAPE_FILEMARK, &tape->flags))
++cnt; /* Filemarks count as 1 sector */ ++cnt; /* Filemarks count as 1 sector */
...@@ -3444,9 +3466,12 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive) ...@@ -3444,9 +3466,12 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive)
__idetape_kfree_stage(tape->merge_stage); __idetape_kfree_stage(tape->merge_stage);
tape->merge_stage = NULL; tape->merge_stage = NULL;
} }
/* Clear pipeline flags. */
clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags); clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags);
tape->chrdev_direction = idetape_direction_none; tape->chrdev_direction = idetape_direction_none;
/* Remove pipeline stages. */
if (tape->first_stage == NULL) if (tape->first_stage == NULL)
return 0; return 0;
...@@ -4059,7 +4084,10 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages) ...@@ -4059,7 +4084,10 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages)
* Issue a read 0 command to ensure that DSC handshake * Issue a read 0 command to ensure that DSC handshake
* is switched from completion mode to buffer available * is switched from completion mode to buffer available
* mode. * mode.
* No point in issuing this if DSC overlap isn't supported,
* some drives (Seagate STT3401A) will return an error.
*/ */
if (drive->dsc_overlap) {
bytes_read = idetape_queue_rw_tail(drive, REQ_IDETAPE_READ, 0, tape->merge_stage->bh); bytes_read = idetape_queue_rw_tail(drive, REQ_IDETAPE_READ, 0, tape->merge_stage->bh);
if (bytes_read < 0) { if (bytes_read < 0) {
__idetape_kfree_stage(tape->merge_stage); __idetape_kfree_stage(tape->merge_stage);
...@@ -4068,6 +4096,7 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages) ...@@ -4068,6 +4096,7 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages)
return bytes_read; return bytes_read;
} }
} }
}
if (tape->restart_speed_control_req) if (tape->restart_speed_control_req)
idetape_restart_speed_control(drive); idetape_restart_speed_control(drive);
memset(&rq, 0, sizeof(rq)); memset(&rq, 0, sizeof(rq));
...@@ -4898,6 +4927,10 @@ static ssize_t idetape_chrdev_write (struct file *file, const char *buf, ...@@ -4898,6 +4927,10 @@ static ssize_t idetape_chrdev_write (struct file *file, const char *buf,
return -ENXIO; return -ENXIO;
} }
/* The drive is write protected. */
if (tape->write_prot)
return -EACCES;
#if IDETAPE_DEBUG_LOG #if IDETAPE_DEBUG_LOG
if (tape->debug_level >= 3) if (tape->debug_level >= 3)
printk(KERN_INFO "ide-tape: Reached idetape_chrdev_write, " printk(KERN_INFO "ide-tape: Reached idetape_chrdev_write, "
...@@ -4979,7 +5012,10 @@ static ssize_t idetape_chrdev_write (struct file *file, const char *buf, ...@@ -4979,7 +5012,10 @@ static ssize_t idetape_chrdev_write (struct file *file, const char *buf,
* Issue a write 0 command to ensure that DSC handshake * Issue a write 0 command to ensure that DSC handshake
* is switched from completion mode to buffer available * is switched from completion mode to buffer available
* mode. * mode.
* No point in issuing this if DSC overlap isn't supported,
* some drives (Seagate STT3401A) will return an error.
*/ */
if (drive->dsc_overlap) {
retval = idetape_queue_rw_tail(drive, REQ_IDETAPE_WRITE, 0, tape->merge_stage->bh); retval = idetape_queue_rw_tail(drive, REQ_IDETAPE_WRITE, 0, tape->merge_stage->bh);
if (retval < 0) { if (retval < 0) {
__idetape_kfree_stage(tape->merge_stage); __idetape_kfree_stage(tape->merge_stage);
...@@ -4987,6 +5023,7 @@ static ssize_t idetape_chrdev_write (struct file *file, const char *buf, ...@@ -4987,6 +5023,7 @@ static ssize_t idetape_chrdev_write (struct file *file, const char *buf,
tape->chrdev_direction = idetape_direction_none; tape->chrdev_direction = idetape_direction_none;
return retval; return retval;
} }
}
#if ONSTREAM_DEBUG #if ONSTREAM_DEBUG
if (tape->debug_level >= 2) if (tape->debug_level >= 2)
printk("ide-tape: first_frame_position %d\n", printk("ide-tape: first_frame_position %d\n",
...@@ -5141,7 +5178,7 @@ int idetape_seek_logical_blk (ide_drive_t *drive, int logical_blk_num) ...@@ -5141,7 +5178,7 @@ int idetape_seek_logical_blk (ide_drive_t *drive, int logical_blk_num)
* Note: * Note:
* *
* MTBSF and MTBSFM are not supported when the tape doesn't * MTBSF and MTBSFM are not supported when the tape doesn't
* supports spacing over filemarks in the reverse direction. * support spacing over filemarks in the reverse direction.
* In this case, MTFSFM is also usually not supported (it is * In this case, MTFSFM is also usually not supported (it is
* supported in the rare case in which we crossed the filemark * supported in the rare case in which we crossed the filemark
* during our read-ahead pipelined operation mode). * during our read-ahead pipelined operation mode).
...@@ -5211,6 +5248,8 @@ static int idetape_mtioctop (ide_drive_t *drive,short mt_op,int mt_count) ...@@ -5211,6 +5248,8 @@ static int idetape_mtioctop (ide_drive_t *drive,short mt_op,int mt_count)
} }
switch (mt_op) { switch (mt_op) {
case MTWEOF: case MTWEOF:
if (tape->write_prot)
return -EACCES;
idetape_discard_read_pipeline(drive, 1); idetape_discard_read_pipeline(drive, 1);
for (i = 0; i < mt_count; i++) { for (i = 0; i < mt_count; i++) {
retval = idetape_write_filemark(drive); retval = idetape_write_filemark(drive);
...@@ -5231,9 +5270,21 @@ static int idetape_mtioctop (ide_drive_t *drive,short mt_op,int mt_count) ...@@ -5231,9 +5270,21 @@ static int idetape_mtioctop (ide_drive_t *drive,short mt_op,int mt_count)
return (idetape_queue_pc_tail(drive, &pc)); return (idetape_queue_pc_tail(drive, &pc));
case MTUNLOAD: case MTUNLOAD:
case MTOFFL: case MTOFFL:
/*
* If door is locked, attempt to unlock before
* attempting to eject.
*/
if (tape->door_locked) {
if (idetape_create_prevent_cmd(drive, &pc, 0))
if (!idetape_queue_pc_tail(drive, &pc))
tape->door_locked = DOOR_UNLOCKED;
}
idetape_discard_read_pipeline(drive, 0); idetape_discard_read_pipeline(drive, 0);
idetape_create_load_unload_cmd(drive, &pc,!IDETAPE_LU_LOAD_MASK); idetape_create_load_unload_cmd(drive, &pc,!IDETAPE_LU_LOAD_MASK);
return (idetape_queue_pc_tail(drive, &pc)); retval = idetape_queue_pc_tail(drive, &pc);
if (!retval)
clear_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags);
return retval;
case MTNOP: case MTNOP:
idetape_discard_read_pipeline(drive, 0); idetape_discard_read_pipeline(drive, 0);
return (idetape_flush_tape_buffers(drive)); return (idetape_flush_tape_buffers(drive));
...@@ -5409,6 +5460,8 @@ static int idetape_chrdev_ioctl (struct inode *inode, struct file *file, unsigne ...@@ -5409,6 +5460,8 @@ static int idetape_chrdev_ioctl (struct inode *inode, struct file *file, unsigne
mtget.mt_gstat |= GMT_EOD(0xffffffff); mtget.mt_gstat |= GMT_EOD(0xffffffff);
if (position <= OS_DATA_STARTFRAME1) if (position <= OS_DATA_STARTFRAME1)
mtget.mt_gstat |= GMT_BOT(0xffffffff); mtget.mt_gstat |= GMT_BOT(0xffffffff);
} else if (tape->drv_write_prot) {
mtget.mt_gstat |= GMT_WR_PROT(0xffffffff);
} }
if (copy_to_user((char *) arg,(char *) &mtget, sizeof(struct mtget))) if (copy_to_user((char *) arg,(char *) &mtget, sizeof(struct mtget)))
return -EFAULT; return -EFAULT;
...@@ -5530,6 +5583,8 @@ static int idetape_analyze_headers (ide_drive_t *drive) ...@@ -5530,6 +5583,8 @@ static int idetape_analyze_headers (ide_drive_t *drive)
return 1; return 1;
} }
static void idetape_get_blocksize_from_block_descriptor(ide_drive_t *drive);
/* /*
* Our character device open function. * Our character device open function.
*/ */
...@@ -5539,6 +5594,7 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -5539,6 +5594,7 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
ide_drive_t *drive; ide_drive_t *drive;
idetape_tape_t *tape; idetape_tape_t *tape;
idetape_pc_t pc; idetape_pc_t pc;
int retval;
#if IDETAPE_DEBUG_LOG #if IDETAPE_DEBUG_LOG
printk(KERN_INFO "ide-tape: Reached idetape_chrdev_open\n"); printk(KERN_INFO "ide-tape: Reached idetape_chrdev_open\n");
...@@ -5552,11 +5608,7 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -5552,11 +5608,7 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
if (test_and_set_bit(IDETAPE_BUSY, &tape->flags)) if (test_and_set_bit(IDETAPE_BUSY, &tape->flags))
return -EBUSY; return -EBUSY;
if (!tape->onstream) { if (tape->onstream) {
idetape_read_position(drive);
if (!test_bit(IDETAPE_ADDRESS_VALID, &tape->flags))
(void) idetape_rewind_tape(drive);
} else {
if (minor & 64) { if (minor & 64) {
tape->tape_block_size = tape->stage_size = 32768 + 512; tape->tape_block_size = tape->stage_size = 32768 + 512;
tape->raw = 1; tape->raw = 1;
...@@ -5566,16 +5618,42 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -5566,16 +5618,42 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
} }
idetape_onstream_mode_sense_tape_parameter_page(drive, tape->debug_level); idetape_onstream_mode_sense_tape_parameter_page(drive, tape->debug_level);
} }
if (idetape_wait_ready(drive, 60 * HZ)) { retval = idetape_wait_ready(drive, 60 * HZ);
if (retval) {
clear_bit(IDETAPE_BUSY, &tape->flags); clear_bit(IDETAPE_BUSY, &tape->flags);
printk(KERN_ERR "ide-tape: %s: drive not ready\n", tape->name); printk(KERN_ERR "ide-tape: %s: drive not ready\n", tape->name);
return -EBUSY; return retval;
} }
if (tape->onstream)
idetape_read_position(drive); idetape_read_position(drive);
if (!test_bit(IDETAPE_ADDRESS_VALID, &tape->flags))
(void)idetape_rewind_tape(drive);
if (tape->chrdev_direction != idetape_direction_read) if (tape->chrdev_direction != idetape_direction_read)
clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags); clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags);
/* Read block size and write protect status from drive. */
idetape_get_blocksize_from_block_descriptor(drive);
/* Set write protect flag if device is opened as read-only. */
if ((filp->f_flags & O_ACCMODE) == O_RDONLY)
tape->write_prot = 1;
else
tape->write_prot = tape->drv_write_prot;
/* Make sure drive isn't write protected if user wants to write. */
if (tape->write_prot) {
if ((filp->f_flags & O_ACCMODE) == O_WRONLY ||
(filp->f_flags & O_ACCMODE) == O_RDWR) {
clear_bit(IDETAPE_BUSY, &tape->flags);
return -EROFS;
}
}
/*
* Lock the tape drive door so user can't eject.
* Analyze headers for OnStream drives.
*/
if (tape->chrdev_direction == idetape_direction_none) { if (tape->chrdev_direction == idetape_direction_none) {
if (idetape_create_prevent_cmd(drive, &pc, 1)) { if (idetape_create_prevent_cmd(drive, &pc, 1)) {
if (!idetape_queue_pc_tail(drive, &pc)) { if (!idetape_queue_pc_tail(drive, &pc)) {
...@@ -5638,7 +5716,7 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp) ...@@ -5638,7 +5716,7 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp)
__idetape_kfree_stage(tape->cache_stage); __idetape_kfree_stage(tape->cache_stage);
tape->cache_stage = NULL; tape->cache_stage = NULL;
} }
if (minor < 128) if (minor < 128 && test_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags))
(void) idetape_rewind_tape(drive); (void) idetape_rewind_tape(drive);
if (tape->chrdev_direction == idetape_direction_none) { if (tape->chrdev_direction == idetape_direction_none) {
if (tape->door_locked == DOOR_LOCKED) { if (tape->door_locked == DOOR_LOCKED) {
...@@ -6059,6 +6137,8 @@ static void idetape_get_blocksize_from_block_descriptor(ide_drive_t *drive) ...@@ -6059,6 +6137,8 @@ static void idetape_get_blocksize_from_block_descriptor(ide_drive_t *drive)
header = (idetape_mode_parameter_header_t *) pc.buffer; header = (idetape_mode_parameter_header_t *) pc.buffer;
block_descrp = (idetape_parameter_block_descriptor_t *) (pc.buffer + sizeof(idetape_mode_parameter_header_t)); block_descrp = (idetape_parameter_block_descriptor_t *) (pc.buffer + sizeof(idetape_mode_parameter_header_t));
tape->tape_block_size =( block_descrp->length[0]<<16) + (block_descrp->length[1]<<8) + block_descrp->length[2]; tape->tape_block_size =( block_descrp->length[0]<<16) + (block_descrp->length[1]<<8) + block_descrp->length[2];
tape->drv_write_prot = (header->dsp & 0x80) >> 7;
#if IDETAPE_DEBUG_INFO #if IDETAPE_DEBUG_INFO
printk(KERN_INFO "ide-tape: Adjusted block size - %d\n", tape->tape_block_size); printk(KERN_INFO "ide-tape: Adjusted block size - %d\n", tape->tape_block_size);
#endif /* IDETAPE_DEBUG_INFO */ #endif /* IDETAPE_DEBUG_INFO */
...@@ -6139,6 +6219,9 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor) ...@@ -6139,6 +6219,9 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor)
} }
} }
#endif /* CONFIG_BLK_DEV_IDEPCI */ #endif /* CONFIG_BLK_DEV_IDEPCI */
/* Seagate Travan drives do not support DSC overlap. */
if (strstr(drive->id->model, "Seagate STT3401"))
drive->dsc_overlap = 0;
tape->drive = drive; tape->drive = drive;
tape->minor = minor; tape->minor = minor;
tape->name[0] = 'h'; tape->name[0] = 'h';
......
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