《Linux那些事儿之我是USB》我是U盘(28)彼岸花的传说(七)

来源:岁月联盟 编辑:exp 时间:2011-11-22

 

很显然,我们是把为INQUIRY命令准备的数据保存到了我们自己定义的一个结构体中,即structdata_ptr[36],但是我们是为了回应一个SCSI命令,最终需要知道答案的是SCSI核心层。正是它们传递了一个scsi_cmnd结构体下来,即srb。struct scsi_cmnd中有两个成员,即unsigned request_bufflen和void *request_buffer,应该把data数组中的数据传送到request_buffer中去,这样,SCSI核心层就知道去哪里获取结果。没错,当时就是这样!

 

usb_stor_set_xfer_buf()这个函数来自,drivers/usb/storage/protocol.c中:

 

243 /* Store the contents of buffer into srb'stransfer buffer and set the

 

244  *SCSI residue. */

 

245 void usb_stor_set_xfer_buf(unsigned char*buffer,

 

246         unsigned intbuflen, struct scsi_cmnd *srb)

 

247 {

 

248   unsigned int index = 0, offset =0;

 

249

 

250    usb_stor_access_xfer_buf(buffer,buflen, srb, &index, &offset,

 

251                        TO_XFER_BUF);

 

252   if (buflen <srb->request_bufflen)

 

253        srb->resid = srb->request_bufflen - buflen;

 

254 }

 

主要调用的又是usb_stor_access_xfer_buf()函数,这个函数也来自同一个文件:drivers/usb/storage/protocol.c:

 

159 unsigned int usb_stor_access_xfer_buf(unsignedchar *buffer,

 

160         unsigned intbuflen, struct scsi_cmnd *srb, unsigned int *index,

 

161         unsigned int*offset, enum xfer_buf_dir dir)

 

162 {

 

163     unsignedint cnt;

 

164

 

165    /* If notusing scatter-gather, just transfer the data directly.

 

166    * Make certain it will fit in the available buffer space. */

 

167    if(srb->use_sg == 0) {

 

168       if (*offset >= srb->request_bufflen)

 

169            return 0;

 

170         cnt = min(buflen, srb->request_bufflen -*offset);

 

171         if (dir == TO_XFER_BUF)

 

172            memcpy((unsigned char *) srb->request_buffer+ *offset,

 

173                                        buffer,cnt);

 

174          else

 

175        memcpy(buffer, (unsigned char*) srb->request_buffer +

 

176                                        *offset, cnt);

 

177        *offset += cnt;

 

178

 

179     /*Using scatter-gather.  We have togo through the list one entry

 

180     * at a time.  Each s-g entry contains some number of pages, and

 

181     * each page has to be kmap()'edseparately.  If the page is already

 

182     * in kernel-addressable memory thenkmap() will return its address.

 

183      * If the page is not directly accessible-- such as a user buffer

 

184      * located in high memory -- then kmap()will map it to a temporary

 

185     * position in the kernel's virtualaddress space. */

 

186    } else {

 

187        struct scatterlist *sg =

 

188                         (struct scatterlist *)srb->request_buffer

 

189                           + *index;

 

190

 

191         /* This loop handles a single s-g list entry,which may

 

192          *include multiple pages.  Find theinitial page structure

 

193           *and the starting offset within the page, and update

 

194          *the *offset and *index values for the next loop. */

 

195         cnt = 0;

 

196         while (cnt < buflen && *index <srb->use_sg) {

 

197             struct page *page = sg->page +

 

198                               ((sg->offset + *offset)>> PAGE_SHIFT);

 

199              unsigned int poff =

 

200                         (sg->offset+ *offset) & (PAGE_SIZE-1);

 

201              unsigned int sglen = sg->length- *offset;

 

202

 

203             if (sglen > buflen - cnt) {

 

204

 

205                  /* Transfer ends within this s-gentry */

 

206                   sglen = buflen - cnt;

 

207                   *offset += sglen;

 

208             } else {

 

209

 

210                    /* Transfer continues to next s-g entry */

 

211                   *offset = 0;

 

212                   ++*index;

 

213                   ++sg;

 

214              }

 

215

 

216             /* Transfer the data for all thepages in this

 

217               * s-g entry.  For each page: call kmap(), do the

 

218               * transfer, and call kunmap() immediately after. */

 

219             while (sglen > 0) {

 

220                   unsigned int plen = min(sglen,(unsigned int)

 

221                                                PAGE_SIZE- poff);

 

222                   unsigned char *ptr = kmap(page);

 

223

 

224                 if (dir == TO_XFER_BUF)

 

225                       memcpy(ptr + poff, buffer + cnt,plen);

 

226                  else

 

227                         memcpy(buffer + cnt, ptr + poff, plen);

 

228                  kunmap(page);

 

229

 

230                   /* Start at the beginning of thenext page */

 

231                   poff = 0;

 

232                   ++page;

 

233                   cnt += plen;

 

234                   sglen -= plen;

 

235             }

 

236          }

 

237   }

 

238

 

239    /* Returnthe amount actually transferred */

 

240    returncnt;

 

241 }

 

在编写Linux设备驱动时,总是要涉及内存管理。内存管理毫无疑问是Linux内核中最复杂的一部分,能不涉及我们都希望别去涉及。但生活中总是充满了无奈,该来的还是会来。

 

所以,usb_stor_access_xfer_buf()函数映入了我们的眼帘。 www.2cto.com

 

首先判断srb->use_sg是否为0。IT玩家们创建了有一个词,即scatter/gather,它是一种用于高性能IO的标准技术。它通常意味着一种DMA传输方式,对于一个给定的数据块,它可能在内存中存在于一些离散的缓冲区。换而言之,就是一些不连续的内存缓冲区一起保存一个数据块。如果没有scatter/gather,那么当我们要建立一个从内存到磁盘的传输,那么操作系统通常会为每一个buffer做一次传输,或者干脆就是把这些不连续的buffer里边的东西全都移动到另一个很大的buffer里面,再开始传输。那么这两种方法显然都是效率不高的。

 

毫无疑问,如果操作系统、驱动程序或硬件能够把这些来自内存中离散位置的数据收集起来(gatherup)并转移它们到适当位置整个这个步骤是一个单一的操作的话,效率肯定就会更高。反之,如果要从磁盘向内存中传输,而有一个单一的操作能够把数据块直接分散开来(scatter)到达内存中需要的位置,而不再需要中间的那个块移动,或者别的方法,那么显然,效率总会更高。

 

在structscsi_cmnd中,有一个成员unsignedshort use_sg,上头传下来的scsi_cmnd,其use_sg是设好了的,这里判断一下。如果它为0,那么说明没有使用scatter/gather。struct scsi_cmnd中还有两个成员,unsigned request_bufflen和void *request_buffer,它们和use_sg是什么关系呢?

 

事实上,要使用scatter/gather,就需要一个scatterlist数组,有人称它为散列表数组。对于不同的硬件平台,定义了不同的structscatterlist结构体,它们来自include/asm/scatterlist.h中。(如果是硬件平台i386的,那么就是include/asm-i386/scatterlist.h,如果是x86_64的平台,那么就在include/asm-x86_64/scatterlist.h中),然后所谓的scatter/gather就是一次把整个scatterlist数组给传送掉。而use_sg为0就表示没有scatter gather list,或者说scatterlist,对于这种情况,数据将直接传送给request_buffer或者直接从request_buffer中取得数据。而如果use_sg大于0,那么表示scatter gather list这么一个数组就在request_buffer中,而数组元素个数正是use_sg个。也就是说,srb->request_buffer里边的数据有两种可能,一种是包含了数据本身,另一种是包含了scattergather list。具体是哪种情况通过判断use_sg来决定。而接下来即将要讲到的srb->request_bufflen顾名思义,就是buffer的长度,但对于use_sg大于0的情况,换言之,对于使用scatter gather list的情况,request_bufflen没有意义,将被忽略。

 

对这些原理有了基本的了解之后,我们可以从下节开始看代码了。这里先提醒一下,要注意我们这个函数虽然看似是传输数据,可它实际上并没有和USB真正发生关系,我们只是从软件上来fix一个硬件的bug,这个bug就是我们已经说过了的,不能响应基本的SCSI命令INQUIRY。

 

所以对于那些不能响应INQUIRY命令的设备,当上层的驱动程序去INQUIRY时,实际上是调用我们的queuecommand,那么我们根本就不用和下面的硬件去打交道,就直接回复上层,即我们从软件上来准备这个一段INQUIRY数据给上层,这才是我们这个函数的目的。真正的和硬件打交道的代码在后面,我们还没走到那一步。