resources: when allocate_resource() fails, leave resource untouched

When "allocate_resource(root, new, size, ...)" fails, we currently
clobber "new".  This is inconvenient for the caller, who might care
about the original contents of the resource.

For example, when pci_bus_alloc_resource() fails, the "can't allocate
mem resource %pR" message from pci_assign_resources() currently contains
junk for the resource start/end.

This patch delays the "new" update until we're about to return success.

Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Acked-by: Yinghai Lu <yhlu.kernel@gmail.com>
Signed-off-by: Bjorn Helgaas <bjorn.helgaas@hp.com>
Signed-off-by: Jesse Barnes <jbarnes@virtuousgeek.org>
This commit is contained in:
Bjorn Helgaas 2009-11-02 10:45:36 -07:00 committed by Jesse Barnes
parent 03db42adfe
commit 1e5ad96790

View file

@ -308,35 +308,37 @@ static int find_resource(struct resource *root, struct resource *new,
void *alignf_data) void *alignf_data)
{ {
struct resource *this = root->child; struct resource *this = root->child;
resource_size_t start, end;
new->start = root->start; start = root->start;
/* /*
* Skip past an allocated resource that starts at 0, since the assignment * Skip past an allocated resource that starts at 0, since the assignment
* of this->start - 1 to new->end below would cause an underflow. * of this->start - 1 to new->end below would cause an underflow.
*/ */
if (this && this->start == 0) { if (this && this->start == 0) {
new->start = this->end + 1; start = this->end + 1;
this = this->sibling; this = this->sibling;
} }
for(;;) { for(;;) {
if (this) if (this)
new->end = this->start - 1; end = this->start - 1;
else else
new->end = root->end; end = root->end;
if (new->start < min) if (start < min)
new->start = min; start = min;
if (new->end > max) if (end > max)
new->end = max; end = max;
new->start = ALIGN(new->start, align); start = ALIGN(start, align);
if (alignf) if (alignf)
alignf(alignf_data, new, size, align); alignf(alignf_data, new, size, align);
if (new->start < new->end && new->end - new->start >= size - 1) { if (start < end && end - start >= size - 1) {
new->end = new->start + size - 1; new->start = start;
new->end = start + size - 1;
return 0; return 0;
} }
if (!this) if (!this)
break; break;
new->start = this->end + 1; start = this->end + 1;
this = this->sibling; this = this->sibling;
} }
return -EBUSY; return -EBUSY;