diff options
Diffstat (limited to 'drivers/of/dynamic.c')
| -rw-r--r-- | drivers/of/dynamic.c | 62 | 
1 files changed, 50 insertions, 12 deletions
| diff --git a/drivers/of/dynamic.c b/drivers/of/dynamic.c index f4f8ed9b5454..a09c1c3cf831 100644 --- a/drivers/of/dynamic.c +++ b/drivers/of/dynamic.c @@ -205,15 +205,24 @@ static void __of_attach_node(struct device_node *np)  	const __be32 *phandle;  	int sz; -	np->name = __of_get_property(np, "name", NULL) ? : "<NULL>"; -	np->type = __of_get_property(np, "device_type", NULL) ? : "<NULL>"; - -	phandle = __of_get_property(np, "phandle", &sz); -	if (!phandle) -		phandle = __of_get_property(np, "linux,phandle", &sz); -	if (IS_ENABLED(CONFIG_PPC_PSERIES) && !phandle) -		phandle = __of_get_property(np, "ibm,phandle", &sz); -	np->phandle = (phandle && (sz >= 4)) ? be32_to_cpup(phandle) : 0; +	if (!of_node_check_flag(np, OF_OVERLAY)) { +		np->name = __of_get_property(np, "name", NULL); +		np->type = __of_get_property(np, "device_type", NULL); +		if (!np->name) +			np->name = "<NULL>"; +		if (!np->type) +			np->type = "<NULL>"; + +		phandle = __of_get_property(np, "phandle", &sz); +		if (!phandle) +			phandle = __of_get_property(np, "linux,phandle", &sz); +		if (IS_ENABLED(CONFIG_PPC_PSERIES) && !phandle) +			phandle = __of_get_property(np, "ibm,phandle", &sz); +		if (phandle && (sz >= 4)) +			np->phandle = be32_to_cpup(phandle); +		else +			np->phandle = 0; +	}  	np->child = NULL;  	np->sibling = np->parent->child; @@ -268,13 +277,13 @@ void __of_detach_node(struct device_node *np)  	}  	of_node_set_flag(np, OF_DETACHED); + +	/* race with of_find_node_by_phandle() prevented by devtree_lock */ +	__of_free_phandle_cache_entry(np->phandle);  }  /**   * of_detach_node() - "Unplug" a node from the device tree. - * - * The caller must hold a reference to the node.  The memory associated with - * the node is not freed until its refcount goes to zero.   */  int of_detach_node(struct device_node *np)  { @@ -330,6 +339,25 @@ void of_node_release(struct kobject *kobj)  	if (!of_node_check_flag(node, OF_DYNAMIC))  		return; +	if (of_node_check_flag(node, OF_OVERLAY)) { + +		if (!of_node_check_flag(node, OF_OVERLAY_FREE_CSET)) { +			/* premature refcount of zero, do not free memory */ +			pr_err("ERROR: memory leak before free overlay changeset,  %pOF\n", +			       node); +			return; +		} + +		/* +		 * If node->properties non-empty then properties were added +		 * to this node either by different overlay that has not +		 * yet been removed, or by a non-overlay mechanism. +		 */ +		if (node->properties) +			pr_err("ERROR: %s(), unexpected properties in %pOF\n", +			       __func__, node); +	} +  	property_list_free(node->properties);  	property_list_free(node->deadprops); @@ -434,6 +462,16 @@ struct device_node *__of_node_dup(const struct device_node *np,  static void __of_changeset_entry_destroy(struct of_changeset_entry *ce)  { +	if (ce->action == OF_RECONFIG_ATTACH_NODE && +	    of_node_check_flag(ce->np, OF_OVERLAY)) { +		if (kref_read(&ce->np->kobj.kref) > 1) { +			pr_err("ERROR: memory leak, expected refcount 1 instead of %d, of_node_get()/of_node_put() unbalanced - destroy cset entry: attach overlay node %pOF\n", +			       kref_read(&ce->np->kobj.kref), ce->np); +		} else { +			of_node_set_flag(ce->np, OF_OVERLAY_FREE_CSET); +		} +	} +  	of_node_put(ce->np);  	list_del(&ce->node);  	kfree(ce); | 
