mirror of
https://github.com/FEX-Emu/linux.git
synced 2024-12-14 21:01:29 +00:00
USB storage: sg chaining support
[PATCH] USB storage: sg chaining support Modify usb_stor_access_xfer_buf() to take a pointer to an sg entry pointer, so we can keep track of that instead of passing around an integer index (which we can't use when dealing with multiple scatterlist arrays). Signed-off-by: Jens Axboe <jens.axboe@oracle.com>
This commit is contained in:
parent
53d412fce0
commit
1f6f31a03e
@ -798,12 +798,13 @@ static int alauda_read_data(struct us_data *us, unsigned long address,
|
||||
{
|
||||
unsigned char *buffer;
|
||||
u16 lba, max_lba;
|
||||
unsigned int page, len, index, offset;
|
||||
unsigned int page, len, offset;
|
||||
unsigned int blockshift = MEDIA_INFO(us).blockshift;
|
||||
unsigned int pageshift = MEDIA_INFO(us).pageshift;
|
||||
unsigned int blocksize = MEDIA_INFO(us).blocksize;
|
||||
unsigned int pagesize = MEDIA_INFO(us).pagesize;
|
||||
unsigned int uzonesize = MEDIA_INFO(us).uzonesize;
|
||||
struct scatterlist *sg;
|
||||
int result;
|
||||
|
||||
/*
|
||||
@ -827,7 +828,8 @@ static int alauda_read_data(struct us_data *us, unsigned long address,
|
||||
max_lba = MEDIA_INFO(us).capacity >> (blockshift + pageshift);
|
||||
|
||||
result = USB_STOR_TRANSPORT_GOOD;
|
||||
index = offset = 0;
|
||||
offset = 0;
|
||||
sg = NULL;
|
||||
|
||||
while (sectors > 0) {
|
||||
unsigned int zone = lba / uzonesize; /* integer division */
|
||||
@ -873,7 +875,7 @@ static int alauda_read_data(struct us_data *us, unsigned long address,
|
||||
|
||||
/* Store the data in the transfer buffer */
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&index, &offset, TO_XFER_BUF);
|
||||
&sg, &offset, TO_XFER_BUF);
|
||||
|
||||
page = 0;
|
||||
lba++;
|
||||
@ -891,11 +893,12 @@ static int alauda_write_data(struct us_data *us, unsigned long address,
|
||||
unsigned int sectors)
|
||||
{
|
||||
unsigned char *buffer, *blockbuffer;
|
||||
unsigned int page, len, index, offset;
|
||||
unsigned int page, len, offset;
|
||||
unsigned int blockshift = MEDIA_INFO(us).blockshift;
|
||||
unsigned int pageshift = MEDIA_INFO(us).pageshift;
|
||||
unsigned int blocksize = MEDIA_INFO(us).blocksize;
|
||||
unsigned int pagesize = MEDIA_INFO(us).pagesize;
|
||||
struct scatterlist *sg;
|
||||
u16 lba, max_lba;
|
||||
int result;
|
||||
|
||||
@ -929,7 +932,8 @@ static int alauda_write_data(struct us_data *us, unsigned long address,
|
||||
max_lba = MEDIA_INFO(us).capacity >> (pageshift + blockshift);
|
||||
|
||||
result = USB_STOR_TRANSPORT_GOOD;
|
||||
index = offset = 0;
|
||||
offset = 0;
|
||||
sg = NULL;
|
||||
|
||||
while (sectors > 0) {
|
||||
/* Write as many sectors as possible in this block */
|
||||
@ -946,7 +950,7 @@ static int alauda_write_data(struct us_data *us, unsigned long address,
|
||||
|
||||
/* Get the data from the transfer buffer */
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&index, &offset, FROM_XFER_BUF);
|
||||
&sg, &offset, FROM_XFER_BUF);
|
||||
|
||||
result = alauda_write_lba(us, lba, page, pages, buffer,
|
||||
blockbuffer);
|
||||
|
@ -98,7 +98,8 @@ static int datafab_read_data(struct us_data *us,
|
||||
unsigned char thistime;
|
||||
unsigned int totallen, alloclen;
|
||||
int len, result;
|
||||
unsigned int sg_idx = 0, sg_offset = 0;
|
||||
unsigned int sg_offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
// we're working in LBA mode. according to the ATA spec,
|
||||
// we can support up to 28-bit addressing. I don't know if Datafab
|
||||
@ -155,7 +156,7 @@ static int datafab_read_data(struct us_data *us,
|
||||
|
||||
// Store the data in the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&sg_idx, &sg_offset, TO_XFER_BUF);
|
||||
&sg, &sg_offset, TO_XFER_BUF);
|
||||
|
||||
sector += thistime;
|
||||
totallen -= len;
|
||||
@ -181,7 +182,8 @@ static int datafab_write_data(struct us_data *us,
|
||||
unsigned char thistime;
|
||||
unsigned int totallen, alloclen;
|
||||
int len, result;
|
||||
unsigned int sg_idx = 0, sg_offset = 0;
|
||||
unsigned int sg_offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
// we're working in LBA mode. according to the ATA spec,
|
||||
// we can support up to 28-bit addressing. I don't know if Datafab
|
||||
@ -217,7 +219,7 @@ static int datafab_write_data(struct us_data *us,
|
||||
|
||||
// Get the data from the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&sg_idx, &sg_offset, FROM_XFER_BUF);
|
||||
&sg, &sg_offset, FROM_XFER_BUF);
|
||||
|
||||
command[0] = 0;
|
||||
command[1] = thistime;
|
||||
|
@ -119,7 +119,8 @@ static int jumpshot_read_data(struct us_data *us,
|
||||
unsigned char thistime;
|
||||
unsigned int totallen, alloclen;
|
||||
int len, result;
|
||||
unsigned int sg_idx = 0, sg_offset = 0;
|
||||
unsigned int sg_offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
// we're working in LBA mode. according to the ATA spec,
|
||||
// we can support up to 28-bit addressing. I don't know if Jumpshot
|
||||
@ -170,7 +171,7 @@ static int jumpshot_read_data(struct us_data *us,
|
||||
|
||||
// Store the data in the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&sg_idx, &sg_offset, TO_XFER_BUF);
|
||||
&sg, &sg_offset, TO_XFER_BUF);
|
||||
|
||||
sector += thistime;
|
||||
totallen -= len;
|
||||
@ -195,7 +196,8 @@ static int jumpshot_write_data(struct us_data *us,
|
||||
unsigned char thistime;
|
||||
unsigned int totallen, alloclen;
|
||||
int len, result, waitcount;
|
||||
unsigned int sg_idx = 0, sg_offset = 0;
|
||||
unsigned int sg_offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
// we're working in LBA mode. according to the ATA spec,
|
||||
// we can support up to 28-bit addressing. I don't know if Jumpshot
|
||||
@ -225,7 +227,7 @@ static int jumpshot_write_data(struct us_data *us,
|
||||
|
||||
// Get the data from the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&sg_idx, &sg_offset, FROM_XFER_BUF);
|
||||
&sg, &sg_offset, FROM_XFER_BUF);
|
||||
|
||||
command[0] = 0;
|
||||
command[1] = thistime;
|
||||
|
@ -157,7 +157,7 @@ void usb_stor_transparent_scsi_command(struct scsi_cmnd *srb,
|
||||
* pick up from where this one left off. */
|
||||
|
||||
unsigned int usb_stor_access_xfer_buf(unsigned char *buffer,
|
||||
unsigned int buflen, struct scsi_cmnd *srb, unsigned int *index,
|
||||
unsigned int buflen, struct scsi_cmnd *srb, struct scatterlist **sgptr,
|
||||
unsigned int *offset, enum xfer_buf_dir dir)
|
||||
{
|
||||
unsigned int cnt;
|
||||
@ -184,16 +184,17 @@ unsigned int usb_stor_access_xfer_buf(unsigned char *buffer,
|
||||
* located in high memory -- then kmap() will map it to a temporary
|
||||
* position in the kernel's virtual address space. */
|
||||
} else {
|
||||
struct scatterlist *sg =
|
||||
(struct scatterlist *) srb->request_buffer
|
||||
+ *index;
|
||||
struct scatterlist *sg = *sgptr;
|
||||
|
||||
if (!sg)
|
||||
sg = (struct scatterlist *) srb->request_buffer;
|
||||
|
||||
/* This loop handles a single s-g list entry, which may
|
||||
* include multiple pages. Find the initial page structure
|
||||
* and the starting offset within the page, and update
|
||||
* the *offset and *index values for the next loop. */
|
||||
cnt = 0;
|
||||
while (cnt < buflen && *index < srb->use_sg) {
|
||||
while (cnt < buflen) {
|
||||
struct page *page = sg->page +
|
||||
((sg->offset + *offset) >> PAGE_SHIFT);
|
||||
unsigned int poff =
|
||||
@ -209,8 +210,7 @@ unsigned int usb_stor_access_xfer_buf(unsigned char *buffer,
|
||||
|
||||
/* Transfer continues to next s-g entry */
|
||||
*offset = 0;
|
||||
++*index;
|
||||
++sg;
|
||||
sg = sg_next(sg);
|
||||
}
|
||||
|
||||
/* Transfer the data for all the pages in this
|
||||
@ -234,6 +234,7 @@ unsigned int usb_stor_access_xfer_buf(unsigned char *buffer,
|
||||
sglen -= plen;
|
||||
}
|
||||
}
|
||||
*sgptr = sg;
|
||||
}
|
||||
|
||||
/* Return the amount actually transferred */
|
||||
@ -245,9 +246,10 @@ unsigned int usb_stor_access_xfer_buf(unsigned char *buffer,
|
||||
void usb_stor_set_xfer_buf(unsigned char *buffer,
|
||||
unsigned int buflen, struct scsi_cmnd *srb)
|
||||
{
|
||||
unsigned int index = 0, offset = 0;
|
||||
unsigned int offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
usb_stor_access_xfer_buf(buffer, buflen, srb, &index, &offset,
|
||||
usb_stor_access_xfer_buf(buffer, buflen, srb, &sg, &offset,
|
||||
TO_XFER_BUF);
|
||||
if (buflen < srb->request_bufflen)
|
||||
srb->resid = srb->request_bufflen - buflen;
|
||||
|
@ -52,7 +52,7 @@ extern void usb_stor_transparent_scsi_command(struct scsi_cmnd*,
|
||||
enum xfer_buf_dir {TO_XFER_BUF, FROM_XFER_BUF};
|
||||
|
||||
extern unsigned int usb_stor_access_xfer_buf(unsigned char *buffer,
|
||||
unsigned int buflen, struct scsi_cmnd *srb, unsigned int *index,
|
||||
unsigned int buflen, struct scsi_cmnd *srb, struct scatterlist **,
|
||||
unsigned int *offset, enum xfer_buf_dir dir);
|
||||
|
||||
extern void usb_stor_set_xfer_buf(unsigned char *buffer,
|
||||
|
@ -705,7 +705,8 @@ sddr09_read_data(struct us_data *us,
|
||||
unsigned char *buffer;
|
||||
unsigned int lba, maxlba, pba;
|
||||
unsigned int page, pages;
|
||||
unsigned int len, index, offset;
|
||||
unsigned int len, offset;
|
||||
struct scatterlist *sg;
|
||||
int result;
|
||||
|
||||
// Figure out the initial LBA and page
|
||||
@ -730,7 +731,8 @@ sddr09_read_data(struct us_data *us,
|
||||
// contiguous LBA's. Another exercise left to the student.
|
||||
|
||||
result = 0;
|
||||
index = offset = 0;
|
||||
offset = 0;
|
||||
sg = NULL;
|
||||
|
||||
while (sectors > 0) {
|
||||
|
||||
@ -777,7 +779,7 @@ sddr09_read_data(struct us_data *us,
|
||||
|
||||
// Store the data in the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&index, &offset, TO_XFER_BUF);
|
||||
&sg, &offset, TO_XFER_BUF);
|
||||
|
||||
page = 0;
|
||||
lba++;
|
||||
@ -931,7 +933,8 @@ sddr09_write_data(struct us_data *us,
|
||||
unsigned int pagelen, blocklen;
|
||||
unsigned char *blockbuffer;
|
||||
unsigned char *buffer;
|
||||
unsigned int len, index, offset;
|
||||
unsigned int len, offset;
|
||||
struct scatterlist *sg;
|
||||
int result;
|
||||
|
||||
// Figure out the initial LBA and page
|
||||
@ -968,7 +971,8 @@ sddr09_write_data(struct us_data *us,
|
||||
}
|
||||
|
||||
result = 0;
|
||||
index = offset = 0;
|
||||
offset = 0;
|
||||
sg = NULL;
|
||||
|
||||
while (sectors > 0) {
|
||||
|
||||
@ -987,7 +991,7 @@ sddr09_write_data(struct us_data *us,
|
||||
|
||||
// Get the data from the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&index, &offset, FROM_XFER_BUF);
|
||||
&sg, &offset, FROM_XFER_BUF);
|
||||
|
||||
result = sddr09_write_lba(us, lba, page, pages,
|
||||
buffer, blockbuffer);
|
||||
|
@ -167,7 +167,8 @@ static int sddr55_read_data(struct us_data *us,
|
||||
unsigned long address;
|
||||
|
||||
unsigned short pages;
|
||||
unsigned int len, index, offset;
|
||||
unsigned int len, offset;
|
||||
struct scatterlist *sg;
|
||||
|
||||
// Since we only read in one block at a time, we have to create
|
||||
// a bounce buffer and move the data a piece at a time between the
|
||||
@ -178,7 +179,8 @@ static int sddr55_read_data(struct us_data *us,
|
||||
buffer = kmalloc(len, GFP_NOIO);
|
||||
if (buffer == NULL)
|
||||
return USB_STOR_TRANSPORT_ERROR; /* out of memory */
|
||||
index = offset = 0;
|
||||
offset = 0;
|
||||
sg = NULL;
|
||||
|
||||
while (sectors>0) {
|
||||
|
||||
@ -255,7 +257,7 @@ static int sddr55_read_data(struct us_data *us,
|
||||
|
||||
// Store the data in the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&index, &offset, TO_XFER_BUF);
|
||||
&sg, &offset, TO_XFER_BUF);
|
||||
|
||||
page = 0;
|
||||
lba++;
|
||||
@ -287,7 +289,8 @@ static int sddr55_write_data(struct us_data *us,
|
||||
|
||||
unsigned short pages;
|
||||
int i;
|
||||
unsigned int len, index, offset;
|
||||
unsigned int len, offset;
|
||||
struct scatterlist *sg;
|
||||
|
||||
/* check if we are allowed to write */
|
||||
if (info->read_only || info->force_read_only) {
|
||||
@ -304,7 +307,8 @@ static int sddr55_write_data(struct us_data *us,
|
||||
buffer = kmalloc(len, GFP_NOIO);
|
||||
if (buffer == NULL)
|
||||
return USB_STOR_TRANSPORT_ERROR;
|
||||
index = offset = 0;
|
||||
offset = 0;
|
||||
sg = NULL;
|
||||
|
||||
while (sectors > 0) {
|
||||
|
||||
@ -322,7 +326,7 @@ static int sddr55_write_data(struct us_data *us,
|
||||
|
||||
// Get the data from the transfer buffer
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&index, &offset, FROM_XFER_BUF);
|
||||
&sg, &offset, FROM_XFER_BUF);
|
||||
|
||||
US_DEBUGP("Write %02X pages, to PBA %04X"
|
||||
" (LBA %04X) page %02X\n",
|
||||
|
@ -993,7 +993,8 @@ static int usbat_flash_read_data(struct us_data *us,
|
||||
unsigned char thistime;
|
||||
unsigned int totallen, alloclen;
|
||||
int len, result;
|
||||
unsigned int sg_idx = 0, sg_offset = 0;
|
||||
unsigned int sg_offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
result = usbat_flash_check_media(us, info);
|
||||
if (result != USB_STOR_TRANSPORT_GOOD)
|
||||
@ -1047,7 +1048,7 @@ static int usbat_flash_read_data(struct us_data *us,
|
||||
|
||||
/* Store the data in the transfer buffer */
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&sg_idx, &sg_offset, TO_XFER_BUF);
|
||||
&sg, &sg_offset, TO_XFER_BUF);
|
||||
|
||||
sector += thistime;
|
||||
totallen -= len;
|
||||
@ -1083,7 +1084,8 @@ static int usbat_flash_write_data(struct us_data *us,
|
||||
unsigned char thistime;
|
||||
unsigned int totallen, alloclen;
|
||||
int len, result;
|
||||
unsigned int sg_idx = 0, sg_offset = 0;
|
||||
unsigned int sg_offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
result = usbat_flash_check_media(us, info);
|
||||
if (result != USB_STOR_TRANSPORT_GOOD)
|
||||
@ -1122,7 +1124,7 @@ static int usbat_flash_write_data(struct us_data *us,
|
||||
|
||||
/* Get the data from the transfer buffer */
|
||||
usb_stor_access_xfer_buf(buffer, len, us->srb,
|
||||
&sg_idx, &sg_offset, FROM_XFER_BUF);
|
||||
&sg, &sg_offset, FROM_XFER_BUF);
|
||||
|
||||
/* ATA command 0x30 (WRITE SECTORS) */
|
||||
usbat_pack_ata_sector_cmd(command, thistime, sector, 0x30);
|
||||
@ -1162,8 +1164,8 @@ static int usbat_hp8200e_handle_read10(struct us_data *us,
|
||||
unsigned char *buffer;
|
||||
unsigned int len;
|
||||
unsigned int sector;
|
||||
unsigned int sg_segment = 0;
|
||||
unsigned int sg_offset = 0;
|
||||
struct scatterlist *sg = NULL;
|
||||
|
||||
US_DEBUGP("handle_read10: transfersize %d\n",
|
||||
srb->transfersize);
|
||||
@ -1220,9 +1222,6 @@ static int usbat_hp8200e_handle_read10(struct us_data *us,
|
||||
sector |= short_pack(data[7+5], data[7+4]);
|
||||
transferred = 0;
|
||||
|
||||
sg_segment = 0; /* for keeping track of where we are in */
|
||||
sg_offset = 0; /* the scatter/gather list */
|
||||
|
||||
while (transferred != srb->request_bufflen) {
|
||||
|
||||
if (len > srb->request_bufflen - transferred)
|
||||
@ -1255,7 +1254,7 @@ static int usbat_hp8200e_handle_read10(struct us_data *us,
|
||||
|
||||
/* Store the data in the transfer buffer */
|
||||
usb_stor_access_xfer_buf(buffer, len, srb,
|
||||
&sg_segment, &sg_offset, TO_XFER_BUF);
|
||||
&sg, &sg_offset, TO_XFER_BUF);
|
||||
|
||||
/* Update the amount transferred and the sector number */
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user