diff options
Diffstat (limited to 'kernel/resource.c')
-rw-r--r-- | kernel/resource.c | 44 |
1 files changed, 39 insertions, 5 deletions
diff --git a/kernel/resource.c b/kernel/resource.c index 03c897f7935e..4e9d87fd7bc5 100644 --- a/kernel/resource.c +++ b/kernel/resource.c | |||
@@ -188,6 +188,36 @@ static int __release_resource(struct resource *old) | |||
188 | return -EINVAL; | 188 | return -EINVAL; |
189 | } | 189 | } |
190 | 190 | ||
191 | static void __release_child_resources(struct resource *r) | ||
192 | { | ||
193 | struct resource *tmp, *p; | ||
194 | resource_size_t size; | ||
195 | |||
196 | p = r->child; | ||
197 | r->child = NULL; | ||
198 | while (p) { | ||
199 | tmp = p; | ||
200 | p = p->sibling; | ||
201 | |||
202 | tmp->parent = NULL; | ||
203 | tmp->sibling = NULL; | ||
204 | __release_child_resources(tmp); | ||
205 | |||
206 | printk(KERN_DEBUG "release child resource %pR\n", tmp); | ||
207 | /* need to restore size, and keep flags */ | ||
208 | size = resource_size(tmp); | ||
209 | tmp->start = 0; | ||
210 | tmp->end = size - 1; | ||
211 | } | ||
212 | } | ||
213 | |||
214 | void release_child_resources(struct resource *r) | ||
215 | { | ||
216 | write_lock(&resource_lock); | ||
217 | __release_child_resources(r); | ||
218 | write_unlock(&resource_lock); | ||
219 | } | ||
220 | |||
191 | /** | 221 | /** |
192 | * request_resource - request and reserve an I/O or memory resource | 222 | * request_resource - request and reserve an I/O or memory resource |
193 | * @root: root resource descriptor | 223 | * @root: root resource descriptor |
@@ -316,8 +346,10 @@ int __weak page_is_ram(unsigned long pfn) | |||
316 | static int find_resource(struct resource *root, struct resource *new, | 346 | static int find_resource(struct resource *root, struct resource *new, |
317 | resource_size_t size, resource_size_t min, | 347 | resource_size_t size, resource_size_t min, |
318 | resource_size_t max, resource_size_t align, | 348 | resource_size_t max, resource_size_t align, |
319 | void (*alignf)(void *, struct resource *, | 349 | resource_size_t (*alignf)(void *, |
320 | resource_size_t, resource_size_t), | 350 | const struct resource *, |
351 | resource_size_t, | ||
352 | resource_size_t), | ||
321 | void *alignf_data) | 353 | void *alignf_data) |
322 | { | 354 | { |
323 | struct resource *this = root->child; | 355 | struct resource *this = root->child; |
@@ -343,7 +375,7 @@ static int find_resource(struct resource *root, struct resource *new, | |||
343 | tmp.end = max; | 375 | tmp.end = max; |
344 | tmp.start = ALIGN(tmp.start, align); | 376 | tmp.start = ALIGN(tmp.start, align); |
345 | if (alignf) | 377 | if (alignf) |
346 | alignf(alignf_data, &tmp, size, align); | 378 | tmp.start = alignf(alignf_data, &tmp, size, align); |
347 | if (tmp.start < tmp.end && tmp.end - tmp.start >= size - 1) { | 379 | if (tmp.start < tmp.end && tmp.end - tmp.start >= size - 1) { |
348 | new->start = tmp.start; | 380 | new->start = tmp.start; |
349 | new->end = tmp.start + size - 1; | 381 | new->end = tmp.start + size - 1; |
@@ -371,8 +403,10 @@ static int find_resource(struct resource *root, struct resource *new, | |||
371 | int allocate_resource(struct resource *root, struct resource *new, | 403 | int allocate_resource(struct resource *root, struct resource *new, |
372 | resource_size_t size, resource_size_t min, | 404 | resource_size_t size, resource_size_t min, |
373 | resource_size_t max, resource_size_t align, | 405 | resource_size_t max, resource_size_t align, |
374 | void (*alignf)(void *, struct resource *, | 406 | resource_size_t (*alignf)(void *, |
375 | resource_size_t, resource_size_t), | 407 | const struct resource *, |
408 | resource_size_t, | ||
409 | resource_size_t), | ||
376 | void *alignf_data) | 410 | void *alignf_data) |
377 | { | 411 | { |
378 | int err; | 412 | int err; |