On Mon, Apr 14 2008 at 17:41 +0300, Pete Wyckoff <pw@xxxxxxx> wrote: > fujita.tomonori@xxxxxxxxxxxxx wrote on Mon, 14 Apr 2008 19:50 +0900: >> This patch changes rq->cmd from the static array to a pointer to >> support large commands. >> >> We rarely handle large commands. So for optimization, a struct request >> still has a static array for a command. rq_init sets rq->cmd pointer >> to the static array. >> >> Signed-off-by: FUJITA Tomonori <fujita.tomonori@xxxxxxxxxxxxx> >> Cc: Jens Axboe <jens.axboe@xxxxxxxxxx> > [..] >> diff --git a/include/linux/blkdev.h b/include/linux/blkdev.h >> index b3a58ad..5710ae4 100644 >> --- a/include/linux/blkdev.h >> +++ b/include/linux/blkdev.h >> @@ -215,8 +215,9 @@ struct request { >> /* >> * when request is used as a packet command carrier >> */ >> - unsigned int cmd_len; >> - unsigned char cmd[BLK_MAX_CDB]; >> + unsigned short cmd_len; >> + unsigned char __cmd[BLK_MAX_CDB]; >> + unsigned char *cmd; >> >> unsigned int data_len; >> unsigned int extra_len; /* length of alignment and padding */ >> @@ -812,6 +813,13 @@ static inline void put_dev_sector(Sector p) >> page_cache_release(p.v); >> } >> >> +static inline void rq_set_cmd(struct request *rq, unsigned char *cmd, >> + unsigned short cmd_len) >> +{ >> + rq->cmd = cmd; >> + rq->cmd_len = cmd_len; >> +} > > Here's one way this will be used, in a patched bsg that understands > large commands. Complication is the need to copy and hold onto the > big command across the duration of the request. > > Submit time is fairly clean: > > /* buf, len from user request */ > rq = blk_get_request(..); > rq->cmd_len = len; > if (len > BLK_MAX_CDB) { > rq->cmd = kmalloc(len); > if (rq->cmd == NULL) > goto out; > } > copy_from_user(rq->cmd, buf, len); > > Completion time needs to know when to free rq->cmd: > > if (rq->cmd_len > BLK_MAX_CDB) > kfree(rq->cmd); > blk_put_request(rq); > > Could use (rq->cmd != rq->__cmd) instead, but nothing had better > ever touch rq->cmd_len. > Don't do any of that please. The 16 bytes buffer at request is eventually going away. Just always allocate and call rq_set_cmd(). This way we are free to change in the future, and users code need not change. And better then kmalloc the space for the command, is if you have a structure that governs your request, just allocate a SCSI_MAX_VARLEN_CDB_SIZE (260) bytes array at your structure and always use that. > I don't think the helper rq_set_cmd() will be very useful, as the > caller (bsg) must think about allocation of the command buffer if it > is big. > > One option would be to handle allocation/freeing of the big command > in rq_set_... functions, but I don't think you want to constrain the > interface like that. > > Boaz's concern about big rq->cmd_len still worries me, although I > think this approach is better and worth solving bugs in drivers as > they arise. It only matters in the case that someone adds, say, a > bsg interface to all block devices though. The queuecommand of ub > shows a good example of how this will break. > > In sum, this is a cleaner approach, and a bit easier for callers > with long commands to deal with. And you could get rid of the > trivial helper. > Don't. Please use the set helper it will give us future compatibility. > -- Pete > -- Boaz -- To unsubscribe from this list: send the line "unsubscribe linux-scsi" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html