Commit 070ea018 authored by Lixin Wang's avatar Lixin Wang Committed by Rob Herring

of: dynamic: fix memory leak related to properties of __of_node_dup

If a node with no properties is dynamically added, then a property is
dynamically added to the node, then the property is dynamically removed,
the result will be node->properties == NULL and node->deadprops != NULL.

Add a separate function to release the properties in both lists.
Signed-off-by: default avatarLixin Wang <alan.1.wang@nokia-sbell.com>
Reviewed-by: default avatarFrank Rowand <frank.rowand@sony.com>
Signed-off-by: default avatarRob Herring <robh@kernel.org>
parent 4ee7c0d9
...@@ -298,6 +298,18 @@ int of_detach_node(struct device_node *np) ...@@ -298,6 +298,18 @@ int of_detach_node(struct device_node *np)
} }
EXPORT_SYMBOL_GPL(of_detach_node); EXPORT_SYMBOL_GPL(of_detach_node);
static void property_list_free(struct property *prop_list)
{
struct property *prop, *next;
for (prop = prop_list; prop != NULL; prop = next) {
next = prop->next;
kfree(prop->name);
kfree(prop->value);
kfree(prop);
}
}
/** /**
* of_node_release() - release a dynamically allocated node * of_node_release() - release a dynamically allocated node
* @kref: kref element of the node to be released * @kref: kref element of the node to be released
...@@ -307,7 +319,6 @@ EXPORT_SYMBOL_GPL(of_detach_node); ...@@ -307,7 +319,6 @@ EXPORT_SYMBOL_GPL(of_detach_node);
void of_node_release(struct kobject *kobj) void of_node_release(struct kobject *kobj)
{ {
struct device_node *node = kobj_to_device_node(kobj); struct device_node *node = kobj_to_device_node(kobj);
struct property *prop = node->properties;
/* We should never be releasing nodes that haven't been detached. */ /* We should never be releasing nodes that haven't been detached. */
if (!of_node_check_flag(node, OF_DETACHED)) { if (!of_node_check_flag(node, OF_DETACHED)) {
...@@ -318,18 +329,9 @@ void of_node_release(struct kobject *kobj) ...@@ -318,18 +329,9 @@ void of_node_release(struct kobject *kobj)
if (!of_node_check_flag(node, OF_DYNAMIC)) if (!of_node_check_flag(node, OF_DYNAMIC))
return; return;
while (prop) { property_list_free(node->properties);
struct property *next = prop->next; property_list_free(node->deadprops);
kfree(prop->name);
kfree(prop->value);
kfree(prop);
prop = next;
if (!prop) {
prop = node->deadprops;
node->deadprops = NULL;
}
}
kfree(node->full_name); kfree(node->full_name);
kfree(node->data); kfree(node->data);
kfree(node); kfree(node);
......
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