diff options
author | FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp> | 2008-12-18 00:49:38 -0500 |
---|---|---|
committer | James Bottomley <James.Bottomley@HansenPartnership.com> | 2009-01-02 12:10:35 -0500 |
commit | 97ae77a1cd332c7b011d71315c8faabce6840c72 (patch) | |
tree | fc243ca28ea474eaf81729079eeb3b259cd1b81b /block/blk-map.c | |
parent | 56c451f4b583ccdf80c9e676179c9cb49de86745 (diff) |
[SCSI] block: make blk_rq_map_user take a NULL user-space buffer for WRITE
The commit 818827669d85b84241696ffef2de485db46b0b5e (block: make
blk_rq_map_user take a NULL user-space buffer) extended
blk_rq_map_user to accept a NULL user-space buffer with a READ
command. It was necessary to convert sg to use the block layer mapping
API.
This patch extends blk_rq_map_user again for a WRITE command. It is
necessary to convert st and osst drivers to use the block layer
apping API.
Signed-off-by: FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp>
Acked-by: Jens Axboe <jens.axboe@oracle.com>
Signed-off-by: James Bottomley <James.Bottomley@HansenPartnership.com>
Diffstat (limited to 'block/blk-map.c')
-rw-r--r-- | block/blk-map.c | 16 |
1 files changed, 7 insertions, 9 deletions
diff --git a/block/blk-map.c b/block/blk-map.c index c7e55b23a2bc..f103729b462f 100644 --- a/block/blk-map.c +++ b/block/blk-map.c | |||
@@ -42,7 +42,7 @@ static int __blk_rq_unmap_user(struct bio *bio) | |||
42 | 42 | ||
43 | static int __blk_rq_map_user(struct request_queue *q, struct request *rq, | 43 | static int __blk_rq_map_user(struct request_queue *q, struct request *rq, |
44 | struct rq_map_data *map_data, void __user *ubuf, | 44 | struct rq_map_data *map_data, void __user *ubuf, |
45 | unsigned int len, int null_mapped, gfp_t gfp_mask) | 45 | unsigned int len, gfp_t gfp_mask) |
46 | { | 46 | { |
47 | unsigned long uaddr; | 47 | unsigned long uaddr; |
48 | struct bio *bio, *orig_bio; | 48 | struct bio *bio, *orig_bio; |
@@ -63,7 +63,7 @@ static int __blk_rq_map_user(struct request_queue *q, struct request *rq, | |||
63 | if (IS_ERR(bio)) | 63 | if (IS_ERR(bio)) |
64 | return PTR_ERR(bio); | 64 | return PTR_ERR(bio); |
65 | 65 | ||
66 | if (null_mapped) | 66 | if (map_data && map_data->null_mapped) |
67 | bio->bi_flags |= (1 << BIO_NULL_MAPPED); | 67 | bio->bi_flags |= (1 << BIO_NULL_MAPPED); |
68 | 68 | ||
69 | orig_bio = bio; | 69 | orig_bio = bio; |
@@ -114,17 +114,15 @@ int blk_rq_map_user(struct request_queue *q, struct request *rq, | |||
114 | { | 114 | { |
115 | unsigned long bytes_read = 0; | 115 | unsigned long bytes_read = 0; |
116 | struct bio *bio = NULL; | 116 | struct bio *bio = NULL; |
117 | int ret, null_mapped = 0; | 117 | int ret; |
118 | 118 | ||
119 | if (len > (q->max_hw_sectors << 9)) | 119 | if (len > (q->max_hw_sectors << 9)) |
120 | return -EINVAL; | 120 | return -EINVAL; |
121 | if (!len) | 121 | if (!len) |
122 | return -EINVAL; | 122 | return -EINVAL; |
123 | if (!ubuf) { | 123 | |
124 | if (!map_data || rq_data_dir(rq) != READ) | 124 | if (!ubuf && (!map_data || !map_data->null_mapped)) |
125 | return -EINVAL; | 125 | return -EINVAL; |
126 | null_mapped = 1; | ||
127 | } | ||
128 | 126 | ||
129 | while (bytes_read != len) { | 127 | while (bytes_read != len) { |
130 | unsigned long map_len, end, start; | 128 | unsigned long map_len, end, start; |
@@ -143,7 +141,7 @@ int blk_rq_map_user(struct request_queue *q, struct request *rq, | |||
143 | map_len -= PAGE_SIZE; | 141 | map_len -= PAGE_SIZE; |
144 | 142 | ||
145 | ret = __blk_rq_map_user(q, rq, map_data, ubuf, map_len, | 143 | ret = __blk_rq_map_user(q, rq, map_data, ubuf, map_len, |
146 | null_mapped, gfp_mask); | 144 | gfp_mask); |
147 | if (ret < 0) | 145 | if (ret < 0) |
148 | goto unmap_rq; | 146 | goto unmap_rq; |
149 | if (!bio) | 147 | if (!bio) |