Commit 83df3be7 authored by Brian King's avatar Brian King Committed by Linus Torvalds

[PATCH] handle blk_queue_tags_resize() allocation failures

Fixes blk_queue_resize_tags to properly handle allocation failures.

Currently, if a memory allocation failure occurs during
blk_queue_resize_tags, the tag map ends up getting freed, which should
not happen.  The old tag map should be preserved and only the resize
should fail.
Signed-off-by: default avatarJens Axboe <axboe@suse.de>
Signed-off-by: default avatarBrian King <brking@us.ibm.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 1111a1db
...@@ -573,6 +573,8 @@ static int ...@@ -573,6 +573,8 @@ static int
init_tag_map(request_queue_t *q, struct blk_queue_tag *tags, int depth) init_tag_map(request_queue_t *q, struct blk_queue_tag *tags, int depth)
{ {
int bits, i; int bits, i;
struct request **tag_index;
unsigned long *tag_map;
if (depth > q->nr_requests * 2) { if (depth > q->nr_requests * 2) {
depth = q->nr_requests * 2; depth = q->nr_requests * 2;
...@@ -580,29 +582,31 @@ init_tag_map(request_queue_t *q, struct blk_queue_tag *tags, int depth) ...@@ -580,29 +582,31 @@ init_tag_map(request_queue_t *q, struct blk_queue_tag *tags, int depth)
__FUNCTION__, depth); __FUNCTION__, depth);
} }
tags->tag_index = kmalloc(depth * sizeof(struct request *), GFP_ATOMIC); tag_index = kmalloc(depth * sizeof(struct request *), GFP_ATOMIC);
if (!tags->tag_index) if (!tag_index)
goto fail; goto fail;
bits = (depth / BLK_TAGS_PER_LONG) + 1; bits = (depth / BLK_TAGS_PER_LONG) + 1;
tags->tag_map = kmalloc(bits * sizeof(unsigned long), GFP_ATOMIC); tag_map = kmalloc(bits * sizeof(unsigned long), GFP_ATOMIC);
if (!tags->tag_map) if (!tag_map)
goto fail; goto fail;
memset(tags->tag_index, 0, depth * sizeof(struct request *)); memset(tag_index, 0, depth * sizeof(struct request *));
memset(tags->tag_map, 0, bits * sizeof(unsigned long)); memset(tag_map, 0, bits * sizeof(unsigned long));
tags->max_depth = depth; tags->max_depth = depth;
tags->real_max_depth = bits * BITS_PER_LONG; tags->real_max_depth = bits * BITS_PER_LONG;
tags->tag_index = tag_index;
tags->tag_map = tag_map;
/* /*
* set the upper bits if the depth isn't a multiple of the word size * set the upper bits if the depth isn't a multiple of the word size
*/ */
for (i = depth; i < bits * BLK_TAGS_PER_LONG; i++) for (i = depth; i < bits * BLK_TAGS_PER_LONG; i++)
__set_bit(i, tags->tag_map); __set_bit(i, tag_map);
return 0; return 0;
fail: fail:
kfree(tags->tag_index); kfree(tag_index);
return -ENOMEM; return -ENOMEM;
} }
......
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