diff options
-rw-r--r-- | arch/powerpc/kernel/prom.c | 11 | ||||
-rw-r--r-- | include/asm-powerpc/prom.h | 1 |
2 files changed, 12 insertions, 0 deletions
diff --git a/arch/powerpc/kernel/prom.c b/arch/powerpc/kernel/prom.c index bcd1c5ed44a..6d5e601097a 100644 --- a/arch/powerpc/kernel/prom.c +++ b/arch/powerpc/kernel/prom.c | |||
@@ -1375,8 +1375,17 @@ static void of_node_release(struct kref *kref) | |||
1375 | struct device_node *node = kref_to_device_node(kref); | 1375 | struct device_node *node = kref_to_device_node(kref); |
1376 | struct property *prop = node->properties; | 1376 | struct property *prop = node->properties; |
1377 | 1377 | ||
1378 | /* We should never be releasing nodes that haven't been detached. */ | ||
1379 | if (!of_node_check_flag(node, OF_DETACHED)) { | ||
1380 | printk("WARNING: Bad of_node_put() on %s\n", node->full_name); | ||
1381 | dump_stack(); | ||
1382 | kref_init(&node->kref); | ||
1383 | return; | ||
1384 | } | ||
1385 | |||
1378 | if (!of_node_check_flag(node, OF_DYNAMIC)) | 1386 | if (!of_node_check_flag(node, OF_DYNAMIC)) |
1379 | return; | 1387 | return; |
1388 | |||
1380 | while (prop) { | 1389 | while (prop) { |
1381 | struct property *next = prop->next; | 1390 | struct property *next = prop->next; |
1382 | kfree(prop->name); | 1391 | kfree(prop->name); |
@@ -1457,6 +1466,8 @@ void of_detach_node(const struct device_node *np) | |||
1457 | prevsib->sibling = np->sibling; | 1466 | prevsib->sibling = np->sibling; |
1458 | } | 1467 | } |
1459 | 1468 | ||
1469 | of_node_set_flag(np, OF_DETACHED); | ||
1470 | |||
1460 | out_unlock: | 1471 | out_unlock: |
1461 | write_unlock(&devtree_lock); | 1472 | write_unlock(&devtree_lock); |
1462 | } | 1473 | } |
diff --git a/include/asm-powerpc/prom.h b/include/asm-powerpc/prom.h index f1006b91bd1..1632baa17dc 100644 --- a/include/asm-powerpc/prom.h +++ b/include/asm-powerpc/prom.h | |||
@@ -99,6 +99,7 @@ extern struct device_node *of_chosen; | |||
99 | 99 | ||
100 | /* flag descriptions */ | 100 | /* flag descriptions */ |
101 | #define OF_DYNAMIC 1 /* node and properties were allocated via kmalloc */ | 101 | #define OF_DYNAMIC 1 /* node and properties were allocated via kmalloc */ |
102 | #define OF_DETACHED 2 /* node has been detached from the device tree */ | ||
102 | 103 | ||
103 | static inline int of_node_check_flag(struct device_node *n, unsigned long flag) | 104 | static inline int of_node_check_flag(struct device_node *n, unsigned long flag) |
104 | { | 105 | { |