Commit de77864d authored by Kai Mäkisara's avatar Kai Mäkisara Committed by Trond Myklebust

[PATCH] SCSI tape driver fixes for 2.5.27

This patch contains the following changes to the SCSI tape driver:

- fixes EOM bug in large transfer in fixed block mode
- enables writing trailers by alternating error with ENOSPC and
  attempting a new write
- the write loop cleaned and simplified
- set the scatter/gather list element lengths to sum up to the
  desired transfer length
- MTERASE to use the short erase SCSI command if argument is zero
- set the can-bsr flag for drives with SCSI level >= 3
- C99 structure initializations
- some janitorial changes
parent b43e708b
...@@ -2,7 +2,7 @@ This file contains brief information about the SCSI tape driver. ...@@ -2,7 +2,7 @@ This file contains brief information about the SCSI tape driver.
The driver is currently maintained by Kai M{kisara (email The driver is currently maintained by Kai M{kisara (email
Kai.Makisara@metla.fi) Kai.Makisara@metla.fi)
Last modified: Tue Jun 18 18:13:50 2002 by makisara Last modified: Mon Jul 15 16:30:40 2002 by makisara
BASICS BASICS
...@@ -118,7 +118,6 @@ allowed if ST_BUFFER_WRITES is non-zero. Buffer allocation uses chunks of ...@@ -118,7 +118,6 @@ allowed if ST_BUFFER_WRITES is non-zero. Buffer allocation uses chunks of
memory having sizes 2^n * (page size). Because of this the actual memory having sizes 2^n * (page size). Because of this the actual
buffer size may be larger than the minimum allowable buffer size. buffer size may be larger than the minimum allowable buffer size.
Asynchronous writing. Writing the buffer contents to the tape is Asynchronous writing. Writing the buffer contents to the tape is
started and the write call returns immediately. The status is checked started and the write call returns immediately. The status is checked
at the next tape operation. at the next tape operation.
...@@ -163,12 +162,24 @@ is smaller than the number of segments used in phases 1 and 2, ...@@ -163,12 +162,24 @@ is smaller than the number of segments used in phases 1 and 2,
extending the buffer will always fail. extending the buffer will always fail.
EOM BEHAVIOUR WHEN WRITING
When the end of medium early warning is encountered, the current write
is finished and the number of bytes is returned. The next write
returns -1 and errno is set to ENOSPC. To enable writing a trailer,
the next write is allowed to proceed and, if successful, the number of
bytes is returned. After this, -1 and the number of bytes are
alternately returned until the physical end of medium (or some other
error) is encountered.
MODULE PARAMETERS MODULE PARAMETERS
The buffer size, write threshold, and the maximum number of allocated buffers The buffer size, write threshold, and the maximum number of allocated buffers
are configurable when the driver is loaded as a module. The keywords are: are configurable when the driver is loaded as a module. The keywords are:
buffer_kbs=xxx the buffer size in kilobytes is set to xxx buffer_kbs=xxx the buffer size for fixed block mode is set
to xxx kilobytes
write_threshold_kbs=xxx the write threshold in kilobytes set to xxx write_threshold_kbs=xxx the write threshold in kilobytes set to xxx
max_buffers=xxx the maximum number of tape buffer set to xxx max_buffers=xxx the maximum number of tape buffer set to xxx
max_sg_segs=xxx the maximum number of scatter/gather max_sg_segs=xxx the maximum number of scatter/gather
...@@ -188,17 +199,15 @@ a comma (no spaces allowed). A colon can be used instead of the equal ...@@ -188,17 +199,15 @@ a comma (no spaces allowed). A colon can be used instead of the equal
mark. The definition is prepended by the string st=. Here is an mark. The definition is prepended by the string st=. Here is an
example: example:
st=buffer_kbs:64,max_buffers:2 st=buffer_kbs:64,write_threhold_kbs:60
The following syntax used by the old kernel versions is also supported: The following syntax used by the old kernel versions is also supported:
st=aa[,bb[,cc[,dd]]] st=aa[,bb[,dd]]
where where
aa is the buffer size in 1024 byte units aa is the buffer size for fixed block mode in 1024 byte units
bb is the write threshold in 1024 byte units bb is the write threshold in 1024 byte units
cc is the maximum number of tape buffers to allocate (the number of
buffers is bounded also by the number of drives detected)
dd is the maximum number of scatter/gather segments dd is the maximum number of scatter/gather segments
...@@ -231,7 +240,9 @@ MTOFFL Set device off line (often rewind plus eject). ...@@ -231,7 +240,9 @@ MTOFFL Set device off line (often rewind plus eject).
MTNOP Do nothing except flush the buffers. MTNOP Do nothing except flush the buffers.
MTRETEN Re-tension tape. MTRETEN Re-tension tape.
MTEOM Space to end of recorded data. MTEOM Space to end of recorded data.
MTERASE Erase tape. MTERASE Erase tape. If the argument is zero, the short erase command
is used. The long erase command is used with all other values
of the argument.
MTSEEK Seek to tape block count. Uses Tandberg-compatible seek (QFA) MTSEEK Seek to tape block count. Uses Tandberg-compatible seek (QFA)
for SCSI-1 drives and SCSI-2 seek for SCSI-2 drives. The file and for SCSI-1 drives and SCSI-2 seek for SCSI-2 drives. The file and
block numbers in the status are not valid after a seek. block numbers in the status are not valid after a seek.
......
...@@ -12,13 +12,13 @@ ...@@ -12,13 +12,13 @@
Copyright 1992 - 2002 Kai Makisara Copyright 1992 - 2002 Kai Makisara
email Kai.Makisara@metla.fi email Kai.Makisara@metla.fi
Last modified: Sat Jun 15 13:01:56 2002 by makisara Last modified: Mon Jul 22 13:27:21 2002 by makisara
Some small formal changes - aeb, 950809 Some small formal changes - aeb, 950809
Last modified: 18-JAN-1998 Richard Gooch <rgooch@atnf.csiro.au> Devfs support Last modified: 18-JAN-1998 Richard Gooch <rgooch@atnf.csiro.au> Devfs support
*/ */
static char *verstr = "20020615"; static char *verstr = "20020722";
#include <linux/module.h> #include <linux/module.h>
...@@ -76,11 +76,11 @@ MODULE_DESCRIPTION("SCSI Tape Driver"); ...@@ -76,11 +76,11 @@ MODULE_DESCRIPTION("SCSI Tape Driver");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_PARM(buffer_kbs, "i"); MODULE_PARM(buffer_kbs, "i");
MODULE_PARM_DESC(buffer_kbs, "Default driver buffer size (KB; 32)"); MODULE_PARM_DESC(buffer_kbs, "Default driver buffer size for fixed block mode (KB; 32)");
MODULE_PARM(write_threshold_kbs, "i"); MODULE_PARM(write_threshold_kbs, "i");
MODULE_PARM_DESC(write_threshold_kbs, "Asynchronous write threshold (KB; 30)"); MODULE_PARM_DESC(write_threshold_kbs, "Asynchronous write threshold (KB; 30)");
MODULE_PARM(max_sg_segs, "i"); MODULE_PARM(max_sg_segs, "i");
MODULE_PARM_DESC(max_sg_segs, "Maximum number of scatter/gather segments to use (32)"); MODULE_PARM_DESC(max_sg_segs, "Maximum number of scatter/gather segments to use (64)");
#ifndef MODULE #ifndef MODULE
static struct st_dev_parm { static struct st_dev_parm {
...@@ -139,25 +139,26 @@ static Scsi_Tape **scsi_tapes = NULL; ...@@ -139,25 +139,26 @@ static Scsi_Tape **scsi_tapes = NULL;
static int modes_defined; static int modes_defined;
static ST_buffer *new_tape_buffer(int, int); static ST_buffer *new_tape_buffer(int, int, int);
static int enlarge_buffer(ST_buffer *, int, int); static int enlarge_buffer(ST_buffer *, int, int);
static void normalize_buffer(ST_buffer *); static void normalize_buffer(ST_buffer *);
static int append_to_buffer(const char *, ST_buffer *, int); static int append_to_buffer(const char *, ST_buffer *, int);
static int from_buffer(ST_buffer *, char *, int); static int from_buffer(ST_buffer *, char *, int);
static void buf_to_sg(ST_buffer *, unsigned int);
static int st_attach(Scsi_Device *); static int st_attach(Scsi_Device *);
static int st_detect(Scsi_Device *); static int st_detect(Scsi_Device *);
static void st_detach(Scsi_Device *); static void st_detach(Scsi_Device *);
static struct Scsi_Device_Template st_template = { static struct Scsi_Device_Template st_template = {
module: THIS_MODULE, .module = THIS_MODULE,
name: "tape", .name = "tape",
tag: "st", .tag = "st",
scsi_type: TYPE_TAPE, .scsi_type = TYPE_TAPE,
major: SCSI_TAPE_MAJOR, .major = SCSI_TAPE_MAJOR,
detect: st_detect, .detect = st_detect,
attach: st_attach, .attach = st_attach,
detach: st_detach .detach = st_detach
}; };
static int st_compression(Scsi_Tape *, int); static int st_compression(Scsi_Tape *, int);
...@@ -361,12 +362,11 @@ static Scsi_Request * ...@@ -361,12 +362,11 @@ static Scsi_Request *
if (SRpnt->sr_device->scsi_level <= SCSI_2) if (SRpnt->sr_device->scsi_level <= SCSI_2)
cmd[1] |= (SRpnt->sr_device->lun << 5) & 0xe0; cmd[1] |= (SRpnt->sr_device->lun << 5) & 0xe0;
init_completion(&STp->wait); init_completion(&STp->wait);
SRpnt->sr_use_sg = (bytes > (STp->buffer)->sg[0].length) ? SRpnt->sr_use_sg = (bytes > (STp->buffer)->frp[0].length);
(STp->buffer)->use_sg : 0;
if (SRpnt->sr_use_sg) { if (SRpnt->sr_use_sg) {
bp = (char *) &((STp->buffer)->sg[0]); buf_to_sg(STp->buffer, bytes);
if ((STp->buffer)->sg_segs < SRpnt->sr_use_sg)
SRpnt->sr_use_sg = (STp->buffer)->sg_segs; SRpnt->sr_use_sg = (STp->buffer)->sg_segs;
bp = (char *) &((STp->buffer)->sg[0]);
} else } else
bp = (STp->buffer)->b_data; bp = (STp->buffer)->b_data;
SRpnt->sr_data_direction = direction; SRpnt->sr_data_direction = direction;
...@@ -550,7 +550,6 @@ static int flush_buffer(Scsi_Tape *STp, int seek_next) ...@@ -550,7 +550,6 @@ static int flush_buffer(Scsi_Tape *STp, int seek_next)
if (STp->ready != ST_READY) if (STp->ready != ST_READY)
return 0; return 0;
STps = &(STp->ps[STp->partition]); STps = &(STp->ps[STp->partition]);
if (STps->rw == ST_WRITING) /* Writing */ if (STps->rw == ST_WRITING) /* Writing */
return flush_write_buffer(STp); return flush_write_buffer(STp);
...@@ -1226,14 +1225,15 @@ static ssize_t ...@@ -1226,14 +1225,15 @@ static ssize_t
ssize_t total; ssize_t total;
ssize_t i, do_count, blks, transfer; ssize_t i, do_count, blks, transfer;
ssize_t retval; ssize_t retval;
int write_threshold; int undone;
int doing_write = 0; int async_write;
unsigned char cmd[MAX_COMMAND_SIZE]; unsigned char cmd[MAX_COMMAND_SIZE];
const char *b_point; const char *b_point;
Scsi_Request *SRpnt = NULL; Scsi_Request *SRpnt = NULL;
Scsi_Tape *STp; Scsi_Tape *STp;
ST_mode *STm; ST_mode *STm;
ST_partstat *STps; ST_partstat *STps;
ST_buffer *STbp;
int dev = TAPE_NR(inode->i_rdev); int dev = TAPE_NR(inode->i_rdev);
read_lock(&st_dev_arr_lock); read_lock(&st_dev_arr_lock);
...@@ -1286,12 +1286,13 @@ static ssize_t ...@@ -1286,12 +1286,13 @@ static ssize_t
} }
} }
if ((STp->buffer)->writing) { STbp = STp->buffer;
if (STbp->writing) {
write_behind_check(STp); write_behind_check(STp);
if ((STp->buffer)->syscall_result) { if (STbp->syscall_result) {
DEBC(printk(ST_DEB_MSG "st%d: Async write error (write) %x.\n", DEBC(printk(ST_DEB_MSG "st%d: Async write error (write) %x.\n",
dev, (STp->buffer)->midlevel_result)); dev, STbp->midlevel_result));
if ((STp->buffer)->midlevel_result == INT_MAX) if (STbp->midlevel_result == INT_MAX)
STps->eof = ST_EOM_OK; STps->eof = ST_EOM_OK;
else else
STps->eof = ST_EOM_ERROR; STps->eof = ST_EOM_ERROR;
...@@ -1299,6 +1300,7 @@ static ssize_t ...@@ -1299,6 +1300,7 @@ static ssize_t
} }
if (STps->eof == ST_EOM_OK) { if (STps->eof == ST_EOM_OK) {
STps->eof = ST_EOD_1; /* allow next write */
retval = (-ENOSPC); retval = (-ENOSPC);
goto out; goto out;
} }
...@@ -1316,19 +1318,6 @@ static ssize_t ...@@ -1316,19 +1318,6 @@ static ssize_t
goto out; goto out;
} }
if (!STm->do_buffer_writes) {
#if 0
if (STp->block_size != 0 && (count % STp->block_size) != 0) {
retval = (-EINVAL); /* Write must be integral number of blocks */
goto out;
}
#endif
write_threshold = 1;
} else
write_threshold = (STp->buffer)->buffer_blocks * STp->block_size;
if (!STm->do_async_writes)
write_threshold--;
total = count; total = count;
memset(cmd, 0, MAX_COMMAND_SIZE); memset(cmd, 0, MAX_COMMAND_SIZE);
...@@ -1338,29 +1327,44 @@ static ssize_t ...@@ -1338,29 +1327,44 @@ static ssize_t
STps->rw = ST_WRITING; STps->rw = ST_WRITING;
b_point = buf; b_point = buf;
while ((STp->block_size == 0 && !STm->do_async_writes && count > 0) || while (count > 0) {
(STp->block_size != 0 &&
(STp->buffer)->buffer_bytes + count > write_threshold)) {
doing_write = 1;
if (STp->block_size == 0) if (STp->block_size == 0)
do_count = count; do_count = count;
else { else {
do_count = (STp->buffer)->buffer_blocks * STp->block_size - do_count = STbp->buffer_blocks * STp->block_size -
(STp->buffer)->buffer_bytes; STbp->buffer_bytes;
if (do_count > count) if (do_count > count)
do_count = count; do_count = count;
} }
i = append_to_buffer(b_point, STp->buffer, do_count); i = append_to_buffer(b_point, STbp, do_count);
if (i) { if (i) {
retval = i; retval = i;
goto out; goto out;
} }
count -= do_count;
filp->f_pos += do_count;
b_point += do_count;
async_write = STm->do_async_writes && STps->eof < ST_EOM_OK;
if (STp->block_size != 0)
async_write &= count == 0 &&
(!STm->do_buffer_writes ||
STbp->buffer_bytes >= STp->write_threshold);
if (STp->block_size != 0 && STm->do_buffer_writes && STps->eof < ST_EOM_OK &&
STbp->buffer_bytes < STbp->buffer_size) {
STp->dirty = TRUE;
/* Don't write a buffer that is not full enough. */
if (!async_write && count == 0)
break;
}
if (STp->block_size == 0) if (STp->block_size == 0)
blks = transfer = do_count; blks = transfer = do_count;
else { else {
blks = (STp->buffer)->buffer_bytes / blks = STbp->buffer_bytes /
STp->block_size; STp->block_size;
transfer = blks * STp->block_size; transfer = blks * STp->block_size;
} }
...@@ -1369,47 +1373,59 @@ static ssize_t ...@@ -1369,47 +1373,59 @@ static ssize_t
cmd[4] = blks; cmd[4] = blks;
SRpnt = st_do_scsi(SRpnt, STp, cmd, transfer, SCSI_DATA_WRITE, SRpnt = st_do_scsi(SRpnt, STp, cmd, transfer, SCSI_DATA_WRITE,
STp->timeout, MAX_WRITE_RETRIES, TRUE); STp->timeout, MAX_WRITE_RETRIES, !async_write);
if (!SRpnt) { if (!SRpnt) {
retval = (STp->buffer)->syscall_result; retval = STbp->syscall_result;
goto out; goto out;
} }
if (async_write) {
STbp->writing = transfer;
STp->dirty = !(STbp->writing ==
STbp->buffer_bytes);
SRpnt = NULL; /* Prevent releasing this request! */
DEB( STp->write_pending = 1; )
break;
}
if ((STp->buffer)->syscall_result != 0) { if (STbp->syscall_result != 0) {
DEBC(printk(ST_DEB_MSG "st%d: Error on write:\n", dev)); DEBC(printk(ST_DEB_MSG "st%d: Error on write:\n", dev));
if ((SRpnt->sr_sense_buffer[0] & 0x70) == 0x70 && if ((SRpnt->sr_sense_buffer[0] & 0x70) == 0x70 &&
(SRpnt->sr_sense_buffer[2] & 0x40)) { (SRpnt->sr_sense_buffer[2] & 0x40)) {
if ((SRpnt->sr_sense_buffer[0] & 0x80) != 0) if ((SRpnt->sr_sense_buffer[0] & 0x80) != 0)
transfer = (SRpnt->sr_sense_buffer[3] << 24) | undone = (SRpnt->sr_sense_buffer[3] << 24) |
(SRpnt->sr_sense_buffer[4] << 16) | (SRpnt->sr_sense_buffer[4] << 16) |
(SRpnt->sr_sense_buffer[5] << 8) | (SRpnt->sr_sense_buffer[5] << 8) |
SRpnt->sr_sense_buffer[6]; SRpnt->sr_sense_buffer[6];
else if (STp->block_size == 0 && else if (STp->block_size == 0 &&
(SRpnt->sr_sense_buffer[2] & 0x0f) == (SRpnt->sr_sense_buffer[2] & 0x0f) ==
VOLUME_OVERFLOW) VOLUME_OVERFLOW)
transfer = do_count; undone = do_count;
else else
transfer = 0; undone = 0;
if (STp->block_size != 0) if (STp->block_size != 0)
transfer *= STp->block_size; undone *= STp->block_size;
if (transfer <= do_count) { filp->f_pos -= undone;
filp->f_pos += do_count - transfer; if (undone <= do_count) {
count -= do_count - transfer; /* Only data from this write is not written */
if (STps->drv_block >= 0) { count -= undone;
if (STp->block_size == 0 && do_count -= undone;
transfer < do_count) if (STp->block_size)
STps->drv_block++; blks = (transfer - undone) / STp->block_size;
else if (STp->block_size != 0)
STps->drv_block +=
(do_count - transfer) /
STp->block_size;
}
STps->eof = ST_EOM_OK; STps->eof = ST_EOM_OK;
/* Continue in fixed block mode if all written
in this request but still something left to write
(retval left to zero)
*/
if (STp->block_size == 0 ||
undone > 0 || count == 0)
retval = (-ENOSPC); /* EOM within current request */ retval = (-ENOSPC); /* EOM within current request */
DEBC(printk(ST_DEB_MSG DEBC(printk(ST_DEB_MSG
"st%d: EOM with %d bytes unwritten.\n", "st%d: EOM with %d bytes unwritten.\n",
dev, transfer)); dev, transfer));
} else { } else {
/* Previously buffered data not written */
count -= do_count;
blks = do_count = 0;
STps->eof = ST_EOM_ERROR; STps->eof = ST_EOM_ERROR;
STps->drv_block = (-1); /* Too cautious? */ STps->drv_block = (-1); /* Too cautious? */
retval = (-EIO); /* EOM for old data */ retval = (-EIO); /* EOM for old data */
...@@ -1418,80 +1434,33 @@ static ssize_t ...@@ -1418,80 +1434,33 @@ static ssize_t
dev)); dev));
} }
} else { } else {
filp->f_pos -= do_count;
STps->drv_block = (-1); /* Too cautious? */ STps->drv_block = (-1); /* Too cautious? */
retval = (-EIO); retval = (-EIO);
} }
scsi_release_request(SRpnt);
SRpnt = NULL;
(STp->buffer)->buffer_bytes = 0;
STp->dirty = 0;
if (count < total)
retval = total - count;
goto out;
} }
filp->f_pos += do_count;
b_point += do_count;
count -= do_count;
if (STps->drv_block >= 0) { if (STps->drv_block >= 0) {
if (STp->block_size == 0) if (STp->block_size == 0)
STps->drv_block++; STps->drv_block += (do_count > 0);
else else
STps->drv_block += blks; STps->drv_block += blks;
} }
(STp->buffer)->buffer_bytes = 0;
STbp->buffer_bytes = 0;
STp->dirty = 0; STp->dirty = 0;
}
if (count != 0) {
STp->dirty = 1;
i = append_to_buffer(b_point, STp->buffer, count);
if (i) {
retval = i;
goto out;
}
filp->f_pos += count;
count = 0;
}
if (doing_write && (STp->buffer)->syscall_result != 0) { if (retval) {
retval = (STp->buffer)->syscall_result; if (count < total)
retval = total - count;
goto out; goto out;
} }
if (STm->do_async_writes &&
(((STp->buffer)->buffer_bytes >= STp->write_threshold &&
(STp->buffer)->buffer_bytes >= STp->block_size) ||
STp->block_size == 0)) {
/* Schedule an asynchronous write */
if (STp->block_size == 0)
(STp->buffer)->writing = (STp->buffer)->buffer_bytes;
else
(STp->buffer)->writing = ((STp->buffer)->buffer_bytes /
STp->block_size) * STp->block_size;
STp->dirty = !((STp->buffer)->writing ==
(STp->buffer)->buffer_bytes);
if (STp->block_size == 0)
blks = (STp->buffer)->writing;
else
blks = (STp->buffer)->writing / STp->block_size;
cmd[2] = blks >> 16;
cmd[3] = blks >> 8;
cmd[4] = blks;
DEB( STp->write_pending = 1; )
SRpnt = st_do_scsi(SRpnt, STp, cmd, (STp->buffer)->writing,
SCSI_DATA_WRITE, STp->timeout,
MAX_WRITE_RETRIES, FALSE);
if (SRpnt == NULL) {
retval = (STp->buffer)->syscall_result;
goto out;
} }
SRpnt = NULL; /* Prevent releasing this request! */
} if (STps->eof == ST_EOD_1)
STps->at_sm &= (total == 0); STps->eof = ST_EOM_OK;
if (total > 0) else
STps->eof = ST_NOEOF; STps->eof = ST_NOEOF;
retval = total; retval = total;
...@@ -2425,7 +2394,7 @@ static int st_int_ioctl(Scsi_Tape *STp, unsigned int cmd_in, unsigned long arg) ...@@ -2425,7 +2394,7 @@ static int st_int_ioctl(Scsi_Tape *STp, unsigned int cmd_in, unsigned long arg)
if (STp->write_prot) if (STp->write_prot)
return (-EACCES); return (-EACCES);
cmd[0] = ERASE; cmd[0] = ERASE;
cmd[1] = 1; /* To the end of tape */ cmd[1] = (arg ? 1 : 0); /* Long erase with non-zero argument */
if (STp->immediate) { if (STp->immediate) {
cmd[1] |= 2; /* Don't wait for completion */ cmd[1] |= 2; /* Don't wait for completion */
timeout = STp->timeout; timeout = STp->timeout;
...@@ -3296,7 +3265,7 @@ static int st_ioctl(struct inode *inode, struct file *file, ...@@ -3296,7 +3265,7 @@ static int st_ioctl(struct inode *inode, struct file *file,
/* Try to allocate a new tape buffer. Calling function must not hold /* Try to allocate a new tape buffer. Calling function must not hold
dev_arr_lock. */ dev_arr_lock. */
static ST_buffer * static ST_buffer *
new_tape_buffer(int from_initialization, int need_dma) new_tape_buffer(int from_initialization, int need_dma, int max_sg)
{ {
int i, priority, got = 0, segs = 0; int i, priority, got = 0, segs = 0;
ST_buffer *tb; ST_buffer *tb;
...@@ -3306,26 +3275,29 @@ static ST_buffer * ...@@ -3306,26 +3275,29 @@ static ST_buffer *
else else
priority = GFP_KERNEL; priority = GFP_KERNEL;
i = sizeof(ST_buffer) + (st_max_sg_segs - 1) * sizeof(struct scatterlist); i = sizeof(ST_buffer) + (max_sg - 1) * sizeof(struct scatterlist) +
max_sg * sizeof(struct st_buf_fragment);
tb = kmalloc(i, priority); tb = kmalloc(i, priority);
if (!tb) { if (!tb) {
printk(KERN_NOTICE "st: Can't allocate new tape buffer.\n"); printk(KERN_NOTICE "st: Can't allocate new tape buffer.\n");
return NULL; return NULL;
} }
tb->sg_segs = tb->orig_sg_segs = segs; memset(tb, 0, i);
tb->frp_segs = tb->orig_frp_segs = segs;
tb->use_sg = max_sg;
if (segs > 0) if (segs > 0)
tb->b_data = page_address(tb->sg[0].page); tb->b_data = page_address(tb->sg[0].page);
tb->frp = (struct st_buf_fragment *)(&(tb->sg[0]) + max_sg);
tb->in_use = TRUE; tb->in_use = TRUE;
tb->dma = need_dma; tb->dma = need_dma;
tb->buffer_size = got; tb->buffer_size = got;
tb->writing = 0;
return tb; return tb;
} }
/* Try to allocate a temporary enlarged tape buffer */ /* Try to allocate enough space in the tape buffer */
static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma) static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma)
{ {
int segs, nbr, max_segs, b_size, priority, order, got; int segs, nbr, max_segs, b_size, priority, order, got;
...@@ -3333,12 +3305,11 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma) ...@@ -3333,12 +3305,11 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma)
if (new_size <= STbuffer->buffer_size) if (new_size <= STbuffer->buffer_size)
return TRUE; return TRUE;
normalize_buffer(STbuffer); if (STbuffer->buffer_size <= PAGE_SIZE)
normalize_buffer(STbuffer); /* Avoid extra segment */
max_segs = STbuffer->use_sg; max_segs = STbuffer->use_sg;
if (max_segs > st_max_sg_segs) nbr = max_segs - STbuffer->frp_segs;
max_segs = st_max_sg_segs;
nbr = max_segs - STbuffer->sg_segs;
if (nbr <= 0) if (nbr <= 0)
return FALSE; return FALSE;
...@@ -3350,12 +3321,10 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma) ...@@ -3350,12 +3321,10 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma)
order++, b_size *= 2) order++, b_size *= 2)
; /* empty */ ; /* empty */
for (segs = STbuffer->sg_segs, got = STbuffer->buffer_size; for (segs = STbuffer->frp_segs, got = STbuffer->buffer_size;
segs < max_segs && got < new_size;) { segs < max_segs && got < new_size;) {
STbuffer->sg[segs].page = alloc_pages(priority, order); STbuffer->frp[segs].page = alloc_pages(priority, order);
/* printk("st: allocated %x, order %d\n", STbuffer->sg[segs].page, order); */ if (STbuffer->frp[segs].page == NULL) {
STbuffer->sg[segs].offset = 0;
if (STbuffer->sg[segs].page == NULL) {
if (new_size - got <= (max_segs - segs) * b_size / 2) { if (new_size - got <= (max_segs - segs) * b_size / 2) {
b_size /= 2; /* Large enough for the rest of the buffers */ b_size /= 2; /* Large enough for the rest of the buffers */
order--; order--;
...@@ -3367,16 +3336,16 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma) ...@@ -3367,16 +3336,16 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma)
normalize_buffer(STbuffer); normalize_buffer(STbuffer);
return FALSE; return FALSE;
} }
STbuffer->sg[segs].length = b_size; STbuffer->frp[segs].length = b_size;
STbuffer->sg_segs += 1; STbuffer->frp_segs += 1;
got += b_size; got += b_size;
STbuffer->buffer_size = got; STbuffer->buffer_size = got;
segs++; segs++;
} }
STbuffer->b_data = page_address(STbuffer->sg[0].page); STbuffer->b_data = page_address(STbuffer->frp[0].page);
DEBC(printk(ST_DEB_MSG DEBC(printk(ST_DEB_MSG
"st: Succeeded to enlarge buffer at %p to %d bytes (segs %d->%d, %d).\n", "st: Succeeded to enlarge buffer at %p to %d bytes (segs %d->%d, %d).\n",
STbuffer, got, STbuffer->orig_sg_segs, STbuffer->sg_segs, b_size)); STbuffer, got, STbuffer->orig_frp_segs, STbuffer->frp_segs, b_size));
return TRUE; return TRUE;
} }
...@@ -3385,22 +3354,20 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma) ...@@ -3385,22 +3354,20 @@ static int enlarge_buffer(ST_buffer * STbuffer, int new_size, int need_dma)
/* Release the extra buffer */ /* Release the extra buffer */
static void normalize_buffer(ST_buffer * STbuffer) static void normalize_buffer(ST_buffer * STbuffer)
{ {
int i, order, b_size; int i, order;
for (i = STbuffer->orig_sg_segs; i < STbuffer->sg_segs; i++) { for (i = STbuffer->orig_frp_segs; i < STbuffer->frp_segs; i++) {
for (b_size=PAGE_SIZE, order=0; b_size < STbuffer->sg[i].length; order = get_order(STbuffer->frp[i].length);
order++, b_size *= 2) __free_pages(STbuffer->frp[i].page, order);
; /* empty */ STbuffer->buffer_size -= STbuffer->frp[i].length;
/* printk("st: freeing %x, order %d\n", STbuffer->sg[i].page, order); */
__free_pages(STbuffer->sg[i].page, order);
STbuffer->buffer_size -= STbuffer->sg[i].length;
} }
DEB( DEB(
if (debugging && STbuffer->orig_sg_segs < STbuffer->sg_segs) if (debugging && STbuffer->orig_frp_segs < STbuffer->frp_segs)
printk(ST_DEB_MSG "st: Buffer at %p normalized to %d bytes (segs %d).\n", printk(ST_DEB_MSG "st: Buffer at %p normalized to %d bytes (segs %d->%d).\n",
STbuffer, STbuffer->buffer_size, STbuffer->sg_segs); STbuffer, STbuffer->buffer_size, STbuffer->frp_segs, STbuffer->orig_frp_segs);
) /* end DEB */ ) /* end DEB */
STbuffer->sg_segs = STbuffer->orig_sg_segs; STbuffer->frp_segs = STbuffer->orig_frp_segs;
STbuffer->frp_sg_current = 0;
} }
...@@ -3411,16 +3378,16 @@ static int append_to_buffer(const char *ubp, ST_buffer * st_bp, int do_count) ...@@ -3411,16 +3378,16 @@ static int append_to_buffer(const char *ubp, ST_buffer * st_bp, int do_count)
int i, cnt, res, offset; int i, cnt, res, offset;
for (i = 0, offset = st_bp->buffer_bytes; for (i = 0, offset = st_bp->buffer_bytes;
i < st_bp->sg_segs && offset >= st_bp->sg[i].length; i++) i < st_bp->frp_segs && offset >= st_bp->frp[i].length; i++)
offset -= st_bp->sg[i].length; offset -= st_bp->frp[i].length;
if (i == st_bp->sg_segs) { /* Should never happen */ if (i == st_bp->frp_segs) { /* Should never happen */
printk(KERN_WARNING "st: append_to_buffer offset overflow.\n"); printk(KERN_WARNING "st: append_to_buffer offset overflow.\n");
return (-EIO); return (-EIO);
} }
for (; i < st_bp->sg_segs && do_count > 0; i++) { for (; i < st_bp->frp_segs && do_count > 0; i++) {
cnt = st_bp->sg[i].length - offset < do_count ? cnt = st_bp->frp[i].length - offset < do_count ?
st_bp->sg[i].length - offset : do_count; st_bp->frp[i].length - offset : do_count;
res = copy_from_user(page_address(st_bp->sg[i].page) + offset, ubp, cnt); res = copy_from_user(page_address(st_bp->frp[i].page) + offset, ubp, cnt);
if (res) if (res)
return (-EFAULT); return (-EFAULT);
do_count -= cnt; do_count -= cnt;
...@@ -3444,16 +3411,16 @@ static int from_buffer(ST_buffer * st_bp, char *ubp, int do_count) ...@@ -3444,16 +3411,16 @@ static int from_buffer(ST_buffer * st_bp, char *ubp, int do_count)
int i, cnt, res, offset; int i, cnt, res, offset;
for (i = 0, offset = st_bp->read_pointer; for (i = 0, offset = st_bp->read_pointer;
i < st_bp->sg_segs && offset >= st_bp->sg[i].length; i++) i < st_bp->frp_segs && offset >= st_bp->frp[i].length; i++)
offset -= st_bp->sg[i].length; offset -= st_bp->frp[i].length;
if (i == st_bp->sg_segs) { /* Should never happen */ if (i == st_bp->frp_segs) { /* Should never happen */
printk(KERN_WARNING "st: from_buffer offset overflow.\n"); printk(KERN_WARNING "st: from_buffer offset overflow.\n");
return (-EIO); return (-EIO);
} }
for (; i < st_bp->sg_segs && do_count > 0; i++) { for (; i < st_bp->frp_segs && do_count > 0; i++) {
cnt = st_bp->sg[i].length - offset < do_count ? cnt = st_bp->frp[i].length - offset < do_count ?
st_bp->sg[i].length - offset : do_count; st_bp->frp[i].length - offset : do_count;
res = copy_to_user(ubp, page_address(st_bp->sg[i].page) + offset, cnt); res = copy_to_user(ubp, page_address(st_bp->frp[i].page) + offset, cnt);
if (res) if (res)
return (-EFAULT); return (-EFAULT);
do_count -= cnt; do_count -= cnt;
...@@ -3471,6 +3438,33 @@ static int from_buffer(ST_buffer * st_bp, char *ubp, int do_count) ...@@ -3471,6 +3438,33 @@ static int from_buffer(ST_buffer * st_bp, char *ubp, int do_count)
} }
/* Fill the s/g list up to the length required for this transfer */
static void buf_to_sg(ST_buffer *STbp, unsigned int length)
{
int i;
unsigned int count;
struct scatterlist *sg;
struct st_buf_fragment *frp;
if (length == STbp->frp_sg_current)
return; /* work already done */
sg = &(STbp->sg[0]);
frp = STbp->frp;
for (i=count=0; count < length; i++) {
sg[i].page = frp[i].page;
if (length - count > frp[i].length)
sg[i].length = frp[i].length;
else
sg[i].length = length - count;
count += sg[i].length;
sg[i].offset = 0;
}
STbp->sg_segs = i;
STbp->frp_sg_current = length;
}
/* Validate the options from command line or module parameters */ /* Validate the options from command line or module parameters */
static void validate_options(void) static void validate_options(void)
{ {
...@@ -3587,7 +3581,10 @@ static int st_attach(Scsi_Device * SDp) ...@@ -3587,7 +3581,10 @@ static int st_attach(Scsi_Device * SDp)
return 1; return 1;
} }
buffer = new_tape_buffer(TRUE, (SDp->host)->unchecked_isa_dma); i = SDp->host->sg_tablesize;
if (st_max_sg_segs < i)
i = st_max_sg_segs;
buffer = new_tape_buffer(TRUE, (SDp->host)->unchecked_isa_dma, i);
if (buffer == NULL) { if (buffer == NULL) {
printk(KERN_ERR "st: Can't allocate new tape buffer. Device not attached.\n"); printk(KERN_ERR "st: Can't allocate new tape buffer. Device not attached.\n");
return 1; return 1;
...@@ -3702,7 +3699,6 @@ static int st_attach(Scsi_Device * SDp) ...@@ -3702,7 +3699,6 @@ static int st_attach(Scsi_Device * SDp)
else else
tpnt->tape_type = MT_ISSCSI2; tpnt->tape_type = MT_ISSCSI2;
buffer->use_sg = tpnt->device->host->sg_tablesize;
tpnt->buffer = buffer; tpnt->buffer = buffer;
tpnt->inited = 0; tpnt->inited = 0;
...@@ -3714,7 +3710,7 @@ static int st_attach(Scsi_Device * SDp) ...@@ -3714,7 +3710,7 @@ static int st_attach(Scsi_Device * SDp)
tpnt->use_pf = (SDp->scsi_level >= SCSI_2); tpnt->use_pf = (SDp->scsi_level >= SCSI_2);
tpnt->density = 0; tpnt->density = 0;
tpnt->do_auto_lock = ST_AUTO_LOCK; tpnt->do_auto_lock = ST_AUTO_LOCK;
tpnt->can_bsr = ST_IN_FILE_POS; tpnt->can_bsr = (SDp->scsi_level > 2 ? 1 : ST_IN_FILE_POS); /* BSR mandatory in SCSI3 */
tpnt->can_partitions = 0; tpnt->can_partitions = 0;
tpnt->two_fm = ST_TWO_FM; tpnt->two_fm = ST_TWO_FM;
tpnt->fast_mteom = ST_FAST_MTEOM; tpnt->fast_mteom = ST_FAST_MTEOM;
...@@ -3802,7 +3798,7 @@ static void st_detach(Scsi_Device * SDp) ...@@ -3802,7 +3798,7 @@ static void st_detach(Scsi_Device * SDp)
put_device(&tpnt->driverfs_dev_n[mode]); put_device(&tpnt->driverfs_dev_n[mode]);
} }
if (tpnt->buffer) { if (tpnt->buffer) {
tpnt->buffer->orig_sg_segs = 0; tpnt->buffer->orig_frp_segs = 0;
normalize_buffer(tpnt->buffer); normalize_buffer(tpnt->buffer);
kfree(tpnt->buffer); kfree(tpnt->buffer);
} }
......
...@@ -21,12 +21,20 @@ typedef struct { ...@@ -21,12 +21,20 @@ typedef struct {
int syscall_result; int syscall_result;
Scsi_Request *last_SRpnt; Scsi_Request *last_SRpnt;
unsigned char *b_data; unsigned char *b_data;
unsigned short use_sg; /* zero or number of segments for this adapter */ unsigned short use_sg; /* zero or maximum number of s/g segments for this adapter */
unsigned short sg_segs; /* total number of allocated segments */ unsigned short sg_segs; /* number of segments in s/g list */
unsigned short orig_sg_segs; /* number of segments allocated at first try */ unsigned short orig_frp_segs; /* number of segments allocated at first try */
unsigned short frp_segs; /* number of buffer segments */
unsigned int frp_sg_current; /* driver buffer length currently in s/g list */
struct st_buf_fragment *frp; /* the allocated buffer fragment list */
struct scatterlist sg[1]; /* MUST BE last item */ struct scatterlist sg[1]; /* MUST BE last item */
} ST_buffer; } ST_buffer;
/* The tape buffer fragment descriptor */
struct st_buf_fragment {
struct page *page;
unsigned int length;
};
/* The tape mode definition */ /* The tape mode definition */
typedef struct { typedef struct {
...@@ -147,6 +155,9 @@ typedef struct { ...@@ -147,6 +155,9 @@ typedef struct {
#define ST_EOD 7 #define ST_EOD 7
/* EOD hit while reading => ST_EOD_1 => return zero => ST_EOD_2 => /* EOD hit while reading => ST_EOD_1 => return zero => ST_EOD_2 =>
return zero => ST_EOD, return ENOSPC */ return zero => ST_EOD, return ENOSPC */
/* When writing: ST_EOM_OK == early warning found, write OK
ST_EOD_1 == allow trying new write after early warning
ST_EOM_ERROR == early warning found, not able to write all */
/* Values of rw */ /* Values of rw */
#define ST_IDLE 0 #define ST_IDLE 0
......
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