Commit ad66b0cb authored by James Bottomley's avatar James Bottomley

osst: error handling updates

From: 	Willem Riede <osst@riede.org>

important error handling improvements that I've made as the result of
problem reports.
signed-off-by: default avatarWillem Riede <osst@riede.org>
Signed-off-by: default avatarJames Bottomley <James.Bottomley@SteelEye.com>
parent 97a18d92
...@@ -120,10 +120,10 @@ static int debugging = 1; ...@@ -120,10 +120,10 @@ static int debugging = 1;
// #define OSST_INJECT_ERRORS 1 // #define OSST_INJECT_ERRORS 1
#endif #endif
#define MAX_RETRIES 2 /* Do not retry! The drive firmware already retries when appropriate,
#define MAX_READ_RETRIES 0 and when it tries to tell us something, we had better listen... */
#define MAX_WRITE_RETRIES 0 #define MAX_RETRIES 0
#define MAX_READY_RETRIES 0
#define NO_TAPE NOT_READY #define NO_TAPE NOT_READY
#define OSST_WAIT_POSITION_COMPLETE (HZ > 200 ? HZ / 200 : 1) #define OSST_WAIT_POSITION_COMPLETE (HZ > 200 ? HZ / 200 : 1)
...@@ -224,7 +224,7 @@ static int osst_chk_result(struct osst_tape * STp, struct scsi_request * SRpnt) ...@@ -224,7 +224,7 @@ static int osst_chk_result(struct osst_tape * STp, struct scsi_request * SRpnt)
if (driver_byte(result) & DRIVER_SENSE) if (driver_byte(result) & DRIVER_SENSE)
print_req_sense("osst ", SRpnt); print_req_sense("osst ", SRpnt);
} }
// else else
#endif #endif
if (!(driver_byte(result) & DRIVER_SENSE) || if (!(driver_byte(result) & DRIVER_SENSE) ||
((sense[0] & 0x70) == 0x70 && ((sense[0] & 0x70) == 0x70 &&
...@@ -236,7 +236,7 @@ static int osst_chk_result(struct osst_tape * STp, struct scsi_request * SRpnt) ...@@ -236,7 +236,7 @@ static int osst_chk_result(struct osst_tape * STp, struct scsi_request * SRpnt)
SRpnt->sr_cmnd[0] != MODE_SENSE && SRpnt->sr_cmnd[0] != MODE_SENSE &&
SRpnt->sr_cmnd[0] != TEST_UNIT_READY)) { /* Abnormal conditions for tape */ SRpnt->sr_cmnd[0] != TEST_UNIT_READY)) { /* Abnormal conditions for tape */
if (driver_byte(result) & DRIVER_SENSE) { if (driver_byte(result) & DRIVER_SENSE) {
printk(KERN_WARNING "%s:W: Command with sense data: ", name); printk(KERN_WARNING "%s:W: Command with sense data:\n", name);
print_req_sense("osst:", SRpnt); print_req_sense("osst:", SRpnt);
} }
else { else {
...@@ -623,7 +623,7 @@ static int osst_wait_ready(struct osst_tape * STp, struct scsi_request ** aSRpnt ...@@ -623,7 +623,7 @@ static int osst_wait_ready(struct osst_tape * STp, struct scsi_request ** aSRpnt
memset(cmd, 0, MAX_COMMAND_SIZE); memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_READY_RETRIES, TRUE); SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_RETRIES, TRUE);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
if (!SRpnt) return (-EBUSY); if (!SRpnt) return (-EBUSY);
...@@ -644,7 +644,7 @@ static int osst_wait_ready(struct osst_tape * STp, struct scsi_request ** aSRpnt ...@@ -644,7 +644,7 @@ static int osst_wait_ready(struct osst_tape * STp, struct scsi_request ** aSRpnt
memset(cmd, 0, MAX_COMMAND_SIZE); memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_READY_RETRIES, TRUE); SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_RETRIES, TRUE);
} }
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
#if DEBUG #if DEBUG
...@@ -684,7 +684,7 @@ static int osst_wait_for_medium(struct osst_tape * STp, struct scsi_request ** a ...@@ -684,7 +684,7 @@ static int osst_wait_for_medium(struct osst_tape * STp, struct scsi_request ** a
memset(cmd, 0, MAX_COMMAND_SIZE); memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_READY_RETRIES, TRUE); SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_RETRIES, TRUE);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
if (!SRpnt) return (-EBUSY); if (!SRpnt) return (-EBUSY);
...@@ -703,7 +703,7 @@ static int osst_wait_for_medium(struct osst_tape * STp, struct scsi_request ** a ...@@ -703,7 +703,7 @@ static int osst_wait_for_medium(struct osst_tape * STp, struct scsi_request ** a
memset(cmd, 0, MAX_COMMAND_SIZE); memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_READY_RETRIES, TRUE); SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_RETRIES, TRUE);
} }
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
#if DEBUG #if DEBUG
...@@ -755,7 +755,7 @@ static int osst_flush_drive_buffer(struct osst_tape * STp, struct scsi_request * ...@@ -755,7 +755,7 @@ static int osst_flush_drive_buffer(struct osst_tape * STp, struct scsi_request *
cmd[0] = WRITE_FILEMARKS; cmd[0] = WRITE_FILEMARKS;
cmd[1] = 1; cmd[1] = 1;
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_WRITE_RETRIES, TRUE); SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_RETRIES, TRUE);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
if (!SRpnt) return (-EBUSY); if (!SRpnt) return (-EBUSY);
if (STp->buffer->syscall_result) { if (STp->buffer->syscall_result) {
...@@ -828,6 +828,62 @@ static int osst_wait_frame(struct osst_tape * STp, struct scsi_request ** aSRpnt ...@@ -828,6 +828,62 @@ static int osst_wait_frame(struct osst_tape * STp, struct scsi_request ** aSRpnt
return -EBUSY; return -EBUSY;
} }
static int osst_recover_wait_frame(struct osst_tape * STp, struct scsi_request ** aSRpnt, int writing)
{
struct scsi_request * SRpnt;
unsigned char cmd[MAX_COMMAND_SIZE];
unsigned long startwait = jiffies;
int retval = 1;
char * name = tape_name(STp);
if (writing) {
char mybuf[24];
char * olddata = STp->buffer->b_data;
int oldsize = STp->buffer->buffer_size;
/* write zero fm then read pos - if shows write error, try to recover - if no progress, wait */
memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = WRITE_FILEMARKS;
cmd[1] = 1;
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout,
MAX_RETRIES, TRUE);
while (retval && time_before (jiffies, startwait + 5*60*HZ)) {
if (STp->buffer->syscall_result && (SRpnt->sr_sense_buffer[2] & 0x0f) != 2) {
/* some failure - not just not-ready */
retval = osst_write_error_recovery(STp, aSRpnt, 0);
break;
}
set_current_state(TASK_INTERRUPTIBLE);
schedule_timeout (HZ / OSST_POLL_PER_SEC);
STp->buffer->b_data = mybuf; STp->buffer->buffer_size = 24;
memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = READ_POSITION;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 20, SCSI_DATA_READ, STp->timeout,
MAX_RETRIES, TRUE);
retval = ( STp->buffer->syscall_result || (STp->buffer)->b_data[15] > 25 );
STp->buffer->b_data = olddata; STp->buffer->buffer_size = oldsize;
}
if (retval)
printk(KERN_ERR "%s:E: Device did not succeed to write buffered data\n", name);
} else
/* TODO - figure out which error conditions can be handled */
if (STp->buffer->syscall_result)
printk(KERN_WARNING
"%s:W: Recover_wait_frame(read) cannot handle %02x:%02x:%02x\n", name,
(*aSRpnt)->sr_sense_buffer[ 2] & 0x0f,
(*aSRpnt)->sr_sense_buffer[12],
(*aSRpnt)->sr_sense_buffer[13]);
return retval;
}
/* /*
* Read the next OnStream tape frame at the current location * Read the next OnStream tape frame at the current location
*/ */
...@@ -841,10 +897,10 @@ static int osst_read_frame(struct osst_tape * STp, struct scsi_request ** aSRpnt ...@@ -841,10 +897,10 @@ static int osst_read_frame(struct osst_tape * STp, struct scsi_request ** aSRpnt
char * name = tape_name(STp); char * name = tape_name(STp);
#endif #endif
/* TODO: Error handling */
if (STp->poll) if (STp->poll)
retval = osst_wait_frame (STp, aSRpnt, STp->first_frame_position, 0, timeout); if (osst_wait_frame (STp, aSRpnt, STp->first_frame_position, 0, timeout))
retval = osst_recover_wait_frame(STp, aSRpnt, 0);
memset(cmd, 0, MAX_COMMAND_SIZE); memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = READ_6; cmd[0] = READ_6;
cmd[1] = 1; cmd[1] = 1;
...@@ -852,13 +908,13 @@ static int osst_read_frame(struct osst_tape * STp, struct scsi_request ** aSRpnt ...@@ -852,13 +908,13 @@ static int osst_read_frame(struct osst_tape * STp, struct scsi_request ** aSRpnt
#if DEBUG #if DEBUG
if (debugging) if (debugging)
printk(OSST_DEB_MSG "%s:D: Reading frame from OnStream tape\n", name); printk(OSST_DEB_MSG "%s:D: Reading frame from OnStream tape\n", name);
#endif #endif
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_READ, SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_READ,
STp->timeout, MAX_READ_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
if (!SRpnt) if (!SRpnt)
return (-EBUSY); return (-EBUSY);
if ((STp->buffer)->syscall_result) { if ((STp->buffer)->syscall_result) {
retval = 1; retval = 1;
...@@ -930,9 +986,10 @@ static int osst_initiate_read(struct osst_tape * STp, struct scsi_request ** aSR ...@@ -930,9 +986,10 @@ static int osst_initiate_read(struct osst_tape * STp, struct scsi_request ** aSR
#if DEBUG #if DEBUG
printk(OSST_DEB_MSG "%s:D: Start Read Ahead on OnStream tape\n", name); printk(OSST_DEB_MSG "%s:D: Start Read Ahead on OnStream tape\n", name);
#endif #endif
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_READ_RETRIES, TRUE); SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_RETRIES, TRUE);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
retval = STp->buffer->syscall_result; if ((retval = STp->buffer->syscall_result))
printk(KERN_WARNING "%s:W: Error starting read ahead\n", name);
} }
return retval; return retval;
...@@ -972,6 +1029,7 @@ static int osst_get_logical_frame(struct osst_tape * STp, struct scsi_request ** ...@@ -972,6 +1029,7 @@ static int osst_get_logical_frame(struct osst_tape * STp, struct scsi_request **
name, STp->read_error_frame); name, STp->read_error_frame);
#endif #endif
STp->read_error_frame = 0; STp->read_error_frame = 0;
STp->abort_count++;
} }
return (-EIO); return (-EIO);
} }
...@@ -989,10 +1047,11 @@ static int osst_get_logical_frame(struct osst_tape * STp, struct scsi_request ** ...@@ -989,10 +1047,11 @@ static int osst_get_logical_frame(struct osst_tape * STp, struct scsi_request **
position = 0xbb8; position = 0xbb8;
else if (position > STp->eod_frame_ppos || ++bad == 10) { else if (position > STp->eod_frame_ppos || ++bad == 10) {
position = STp->read_error_frame - 1; position = STp->read_error_frame - 1;
bad = 0;
} }
else { else {
position += 39; position += 29;
cnt += 20; cnt += 19;
} }
#if DEBUG #if DEBUG
printk(OSST_DEB_MSG "%s:D: Bad frame detected, positioning tape to block %d\n", printk(OSST_DEB_MSG "%s:D: Bad frame detected, positioning tape to block %d\n",
...@@ -1309,7 +1368,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi ...@@ -1309,7 +1368,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi
cmd[8] = 32768 & 0xff; cmd[8] = 32768 & 0xff;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_READ, SRpnt = osst_do_scsi(SRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_READ,
STp->timeout, MAX_READ_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
if ((STp->buffer)->syscall_result || !SRpnt) { if ((STp->buffer)->syscall_result || !SRpnt) {
printk(KERN_ERR "%s:E: Failed to read frame back from OnStream buffer\n", name); printk(KERN_ERR "%s:E: Failed to read frame back from OnStream buffer\n", name);
...@@ -1358,8 +1417,8 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi ...@@ -1358,8 +1417,8 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi
vfree((void *)buffer); vfree((void *)buffer);
return (-EIO); return (-EIO);
} }
flag = 0;
if ( i >= nframes + pending ) break; if ( i >= nframes + pending ) break;
flag = 0;
} }
osst_copy_to_buffer(STp->buffer, p); osst_copy_to_buffer(STp->buffer, p);
/* /*
...@@ -1381,7 +1440,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi ...@@ -1381,7 +1440,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi
p[0], p[1], p[2], p[3]); p[0], p[1], p[2], p[3]);
#endif #endif
SRpnt = osst_do_scsi(SRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_WRITE, SRpnt = osst_do_scsi(SRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_WRITE,
STp->timeout, MAX_WRITE_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
if (STp->buffer->syscall_result) if (STp->buffer->syscall_result)
flag = 1; flag = 1;
...@@ -1397,7 +1456,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi ...@@ -1397,7 +1456,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi
cmd[0] = WRITE_FILEMARKS; cmd[0] = WRITE_FILEMARKS;
cmd[1] = 1; cmd[1] = 1;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE,
STp->timeout, MAX_WRITE_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
#if DEBUG #if DEBUG
if (debugging) { if (debugging) {
printk(OSST_DEB_MSG "%s:D: Sleeping in re-write wait ready\n", name); printk(OSST_DEB_MSG "%s:D: Sleeping in re-write wait ready\n", name);
...@@ -1412,7 +1471,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi ...@@ -1412,7 +1471,7 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout,
MAX_READY_RETRIES, TRUE); MAX_RETRIES, TRUE);
if (SRpnt->sr_sense_buffer[2] == 2 && SRpnt->sr_sense_buffer[12] == 4 && if (SRpnt->sr_sense_buffer[2] == 2 && SRpnt->sr_sense_buffer[12] == 4 &&
(SRpnt->sr_sense_buffer[13] == 1 || SRpnt->sr_sense_buffer[13] == 8)) { (SRpnt->sr_sense_buffer[13] == 1 || SRpnt->sr_sense_buffer[13] == 8)) {
...@@ -1449,11 +1508,16 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi ...@@ -1449,11 +1508,16 @@ static int osst_read_back_buffer_and_rewrite(struct osst_tape * STp, struct scsi
#endif #endif
osst_get_frame_position(STp, aSRpnt); osst_get_frame_position(STp, aSRpnt);
#if DEBUG #if DEBUG
printk(OSST_DEB_MSG "%s:D: reported frame positions: host = %d, tape = %d\n", printk(OSST_DEB_MSG "%s:D: reported frame positions: host = %d, tape = %d, buffer = %d\n",
name, STp->first_frame_position, STp->last_frame_position); name, STp->first_frame_position, STp->last_frame_position, STp->cur_frames);
#endif #endif
} }
} }
if (flag) {
/* error recovery did not successfully complete */
printk(KERN_ERR "%s:D: Write error recovery failed in %s\n", name,
STp->write_type == OS_WRITE_HEADER?"header":"body");
}
if (!pending) if (!pending)
osst_copy_to_buffer(STp->buffer, p); /* so buffer content == at entry in all cases */ osst_copy_to_buffer(STp->buffer, p); /* so buffer content == at entry in all cases */
vfree((void *)buffer); vfree((void *)buffer);
...@@ -1513,7 +1577,7 @@ static int osst_reposition_and_retry(struct osst_tape * STp, struct scsi_request ...@@ -1513,7 +1577,7 @@ static int osst_reposition_and_retry(struct osst_tape * STp, struct scsi_request
name, STp->frame_seq_number-1, STp->first_frame_position); name, STp->frame_seq_number-1, STp->first_frame_position);
#endif #endif
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_WRITE, SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_WRITE,
STp->timeout, MAX_WRITE_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
if (STp->buffer->syscall_result) { /* additional write error */ if (STp->buffer->syscall_result) { /* additional write error */
...@@ -1551,6 +1615,7 @@ static int osst_reposition_and_retry(struct osst_tape * STp, struct scsi_request ...@@ -1551,6 +1615,7 @@ static int osst_reposition_and_retry(struct osst_tape * STp, struct scsi_request
debugging = 0; debugging = 0;
} }
#endif #endif
set_current_state(TASK_INTERRUPTIBLE);
schedule_timeout(HZ / 10); schedule_timeout(HZ / 10);
} }
printk(KERN_ERR "%s:E: Failed to find valid tape media\n", name); printk(KERN_ERR "%s:E: Failed to find valid tape media\n", name);
...@@ -1636,7 +1701,9 @@ static int osst_write_error_recovery(struct osst_tape * STp, struct scsi_request ...@@ -1636,7 +1701,9 @@ static int osst_write_error_recovery(struct osst_tape * STp, struct scsi_request
if (retval == 0) { if (retval == 0) {
STp->recover_count++; STp->recover_count++;
STp->recover_erreg++; STp->recover_erreg++;
} } else
STp->abort_count++;
STps->rw = rw_state; STps->rw = rw_state;
return retval; return retval;
} }
...@@ -2703,13 +2770,12 @@ static int osst_get_frame_position(struct osst_tape *STp, struct scsi_request ** ...@@ -2703,13 +2770,12 @@ static int osst_get_frame_position(struct osst_tape *STp, struct scsi_request **
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
if (STp->buffer->syscall_result) if (STp->buffer->syscall_result)
result = ((SRpnt->sr_sense_buffer[2] & 0x0f) == 3) ? -EIO : -EINVAL; result = ((SRpnt->sr_sense_buffer[2] & 0x0f) == 3) ? -EIO : -EINVAL; /* 3: Write Error */
if (result == -EINVAL) if (result == -EINVAL)
printk(KERN_ERR "%s:E: Can't read tape position.\n", name); printk(KERN_ERR "%s:E: Can't read tape position.\n", name);
else { else {
if (result == -EIO) { /* re-read position - this needs to preserve media errors */
if (result == -EIO) { /* re-read position */
unsigned char mysense[16]; unsigned char mysense[16];
memcpy (mysense, SRpnt->sr_sense_buffer, 16); memcpy (mysense, SRpnt->sr_sense_buffer, 16);
memset (scmd, 0, MAX_COMMAND_SIZE); memset (scmd, 0, MAX_COMMAND_SIZE);
...@@ -2717,8 +2783,15 @@ static int osst_get_frame_position(struct osst_tape *STp, struct scsi_request ** ...@@ -2717,8 +2783,15 @@ static int osst_get_frame_position(struct osst_tape *STp, struct scsi_request **
STp->buffer->b_data = mybuf; STp->buffer->buffer_size = 24; STp->buffer->b_data = mybuf; STp->buffer->buffer_size = 24;
SRpnt = osst_do_scsi(SRpnt, STp, scmd, 20, SCSI_DATA_READ, SRpnt = osst_do_scsi(SRpnt, STp, scmd, 20, SCSI_DATA_READ,
STp->timeout, MAX_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
#if DEBUG
printk(OSST_DEB_MSG "%s:D: Reread position, reason=[%02x:%02x:%02x], result=[%s%02x:%02x:%02x]\n",
name, mysense[2], mysense[12], mysense[13], STp->buffer->syscall_result?"":"ok:",
SRpnt->sr_sense_buffer[2],SRpnt->sr_sense_buffer[12],SRpnt->sr_sense_buffer[13]);
#endif
if (!STp->buffer->syscall_result) if (!STp->buffer->syscall_result)
memcpy (SRpnt->sr_sense_buffer, mysense, 16); memcpy (SRpnt->sr_sense_buffer, mysense, 16);
else
printk(KERN_WARNING "%s:W: Double error in get position\n", name);
} }
STp->first_frame_position = ((STp->buffer)->b_data[4] << 24) STp->first_frame_position = ((STp->buffer)->b_data[4] << 24)
+ ((STp->buffer)->b_data[5] << 16) + ((STp->buffer)->b_data[5] << 16)
...@@ -2740,7 +2813,7 @@ static int osst_get_frame_position(struct osst_tape *STp, struct scsi_request ** ...@@ -2740,7 +2813,7 @@ static int osst_get_frame_position(struct osst_tape *STp, struct scsi_request **
#endif #endif
if (STp->cur_frames == 0 && STp->first_frame_position != STp->last_frame_position) { if (STp->cur_frames == 0 && STp->first_frame_position != STp->last_frame_position) {
#if DEBUG #if DEBUG
printk(KERN_WARNING "%s:D: Correcting read position %d, %d, %d\n", name, printk(OSST_DEB_MSG "%s:D: Correcting read position %d, %d, %d\n", name,
STp->first_frame_position, STp->last_frame_position, STp->cur_frames); STp->first_frame_position, STp->last_frame_position, STp->cur_frames);
#endif #endif
STp->first_frame_position = STp->last_frame_position; STp->first_frame_position = STp->last_frame_position;
...@@ -2885,9 +2958,9 @@ static int osst_flush_write_buffer(struct osst_tape *STp, struct scsi_request ** ...@@ -2885,9 +2958,9 @@ static int osst_flush_write_buffer(struct osst_tape *STp, struct scsi_request **
if (offset < OS_DATA_SIZE) if (offset < OS_DATA_SIZE)
osst_zero_buffer_tail(STp->buffer); osst_zero_buffer_tail(STp->buffer);
/* TODO: Error handling! */
if (STp->poll) if (STp->poll)
result = osst_wait_frame (STp, aSRpnt, STp->first_frame_position, -50, 120); if (osst_wait_frame (STp, aSRpnt, STp->first_frame_position, -50, 120))
result = osst_recover_wait_frame(STp, aSRpnt, 1);
memset(cmd, 0, MAX_COMMAND_SIZE); memset(cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = WRITE_6; cmd[0] = WRITE_6;
...@@ -2926,7 +2999,7 @@ static int osst_flush_write_buffer(struct osst_tape *STp, struct scsi_request ** ...@@ -2926,7 +2999,7 @@ static int osst_flush_write_buffer(struct osst_tape *STp, struct scsi_request **
#endif #endif
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, transfer, SCSI_DATA_WRITE, SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, transfer, SCSI_DATA_WRITE,
STp->timeout, MAX_WRITE_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
if (!SRpnt) if (!SRpnt)
return (-EBUSY); return (-EBUSY);
...@@ -3056,8 +3129,9 @@ static int osst_write_frame(struct osst_tape * STp, struct scsi_request ** aSRpn ...@@ -3056,8 +3129,9 @@ static int osst_write_frame(struct osst_tape * STp, struct scsi_request ** aSRpn
} }
if (STp->poll) if (STp->poll)
osst_wait_frame (STp, aSRpnt, STp->first_frame_position, -50, 60); if (osst_wait_frame (STp, aSRpnt, STp->first_frame_position, -48, 120))
/* TODO: Check for an error ! */ if (osst_recover_wait_frame(STp, aSRpnt, 1))
return (-EIO);
// osst_build_stats(STp, &SRpnt); // osst_build_stats(STp, &SRpnt);
...@@ -3082,7 +3156,7 @@ static int osst_write_frame(struct osst_tape * STp, struct scsi_request ** aSRpn ...@@ -3082,7 +3156,7 @@ static int osst_write_frame(struct osst_tape * STp, struct scsi_request ** aSRpn
STp->write_pending = 1; STp->write_pending = 1;
#endif #endif
SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_WRITE, STp->timeout, SRpnt = osst_do_scsi(*aSRpnt, STp, cmd, OS_FRAME_SIZE, SCSI_DATA_WRITE, STp->timeout,
MAX_WRITE_RETRIES, synchronous); MAX_RETRIES, synchronous);
if (!SRpnt) if (!SRpnt)
return (-EBUSY); return (-EBUSY);
*aSRpnt = SRpnt; *aSRpnt = SRpnt;
...@@ -4328,9 +4402,9 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp) ...@@ -4328,9 +4402,9 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp)
memset (cmd, 0, MAX_COMMAND_SIZE); memset (cmd, 0, MAX_COMMAND_SIZE);
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(NULL, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_READY_RETRIES, TRUE); SRpnt = osst_do_scsi(NULL, STp, cmd, 0, SCSI_DATA_NONE, STp->timeout, MAX_RETRIES, TRUE);
if (!SRpnt) { if (!SRpnt) {
retval = (STp->buffer)->syscall_result; retval = (STp->buffer)->syscall_result; /* FIXME - valid? */
goto err_out; goto err_out;
} }
if ((SRpnt->sr_sense_buffer[0] & 0x70) == 0x70 && if ((SRpnt->sr_sense_buffer[0] & 0x70) == 0x70 &&
...@@ -4349,7 +4423,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp) ...@@ -4349,7 +4423,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp)
cmd[1] = 1; cmd[1] = 1;
cmd[4] = 1; cmd[4] = 1;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE,
STp->timeout, MAX_READY_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
} }
osst_wait_ready(STp, &SRpnt, (SRpnt->sr_sense_buffer[13]==1?15:3) * 60, 0); osst_wait_ready(STp, &SRpnt, (SRpnt->sr_sense_buffer[13]==1?15:3) * 60, 0);
} }
...@@ -4366,7 +4440,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp) ...@@ -4366,7 +4440,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp)
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE,
STp->timeout, MAX_READY_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
if ((SRpnt->sr_sense_buffer[0] & 0x70) != 0x70 || if ((SRpnt->sr_sense_buffer[0] & 0x70) != 0x70 ||
(SRpnt->sr_sense_buffer[2] & 0x0f) != UNIT_ATTENTION) (SRpnt->sr_sense_buffer[2] & 0x0f) != UNIT_ATTENTION)
break; break;
...@@ -4387,6 +4461,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp) ...@@ -4387,6 +4461,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp)
} }
new_session = TRUE; new_session = TRUE;
STp->recover_count = 0; STp->recover_count = 0;
STp->abort_count = 0;
} }
/* /*
* if we have valid headers from before, and the drive/tape seem untouched, * if we have valid headers from before, and the drive/tape seem untouched,
...@@ -4474,7 +4549,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp) ...@@ -4474,7 +4549,7 @@ static int os_scsi_tape_open(struct inode * inode, struct file * filp)
cmd[0] = TEST_UNIT_READY; cmd[0] = TEST_UNIT_READY;
SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE, SRpnt = osst_do_scsi(SRpnt, STp, cmd, 0, SCSI_DATA_NONE,
STp->timeout, MAX_READY_RETRIES, TRUE); STp->timeout, MAX_RETRIES, TRUE);
if ((SRpnt->sr_sense_buffer[0] & 0x70) != 0x70 || if ((SRpnt->sr_sense_buffer[0] & 0x70) != 0x70 ||
(SRpnt->sr_sense_buffer[2] & 0x0f) == NOT_READY) (SRpnt->sr_sense_buffer[2] & 0x0f) == NOT_READY)
break; break;
...@@ -4658,14 +4733,19 @@ static int os_scsi_tape_flush(struct file * filp) ...@@ -4658,14 +4733,19 @@ static int os_scsi_tape_flush(struct file * filp)
} }
if (SRpnt) scsi_release_request(SRpnt); if (SRpnt) scsi_release_request(SRpnt);
if (STp->recover_count) { if (STp->abort_count || STp->recover_count) {
printk(KERN_INFO "%s:I: %d recovered errors in", name, STp->recover_count); printk(KERN_INFO "%s:I:", name);
if (STp->abort_count)
printk(" %d unrecovered errors", STp->abort_count);
if (STp->recover_count)
printk(" %d recovered errors", STp->recover_count);
if (STp->write_count) if (STp->write_count)
printk(" %d frames written", STp->write_count); printk(" in %d frames written", STp->write_count);
if (STp->read_count) if (STp->read_count)
printk(" %d frames read", STp->read_count); printk(" in %d frames read", STp->read_count);
printk("\n"); printk("\n");
STp->recover_count = 0; STp->recover_count = 0;
STp->abort_count = 0;
} }
STp->write_count = 0; STp->write_count = 0;
STp->read_count = 0; STp->read_count = 0;
...@@ -5088,12 +5168,10 @@ static int enlarge_buffer(struct osst_buffer *STbuffer, int need_dma) ...@@ -5088,12 +5168,10 @@ static int enlarge_buffer(struct osst_buffer *STbuffer, int need_dma)
if (nbr <= 2) if (nbr <= 2)
return FALSE; return FALSE;
priority = GFP_KERNEL; priority = GFP_KERNEL /* | __GFP_NOWARN */;
if (need_dma) if (need_dma)
priority |= GFP_DMA; priority |= GFP_DMA;
priority |= __GFP_NOWARN;
/* Try to allocate the first segment up to OS_DATA_SIZE and the others /* Try to allocate the first segment up to OS_DATA_SIZE and the others
big enough to reach the goal (code assumes no segments in place) */ big enough to reach the goal (code assumes no segments in place) */
for (b_size = OS_DATA_SIZE, order = OSST_FIRST_ORDER; b_size >= PAGE_SIZE; order--, b_size /= 2) { for (b_size = OS_DATA_SIZE, order = OSST_FIRST_ORDER; b_size >= PAGE_SIZE; order--, b_size /= 2) {
......
...@@ -70,7 +70,7 @@ typedef struct { ...@@ -70,7 +70,7 @@ typedef struct {
#define BLOCK_SIZE_PAGE_LENGTH 4 #define BLOCK_SIZE_PAGE_LENGTH 4
#define BUFFER_FILLING_PAGE 0x33 #define BUFFER_FILLING_PAGE 0x33
#define BUFFER_FILLING_PAGE_LENGTH #define BUFFER_FILLING_PAGE_LENGTH 4
#define VENDOR_IDENT_PAGE 0x36 #define VENDOR_IDENT_PAGE 0x36
#define VENDOR_IDENT_PAGE_LENGTH 8 #define VENDOR_IDENT_PAGE_LENGTH 8
...@@ -577,6 +577,7 @@ struct osst_tape { ...@@ -577,6 +577,7 @@ struct osst_tape {
int min_block; int min_block;
int max_block; int max_block;
int recover_count; /* from tape opening */ int recover_count; /* from tape opening */
int abort_count;
int write_count; int write_count;
int read_count; int read_count;
int recover_erreg; /* from last status call */ int recover_erreg; /* from last status call */
......
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