devres.c

来自「linux 内核源代码」· C语言 代码 · 共 647 行 · 第 1/2 页

C
647
字号
/* * drivers/base/devres.c - device resource management * * Copyright (c) 2006  SUSE Linux Products GmbH * Copyright (c) 2006  Tejun Heo <teheo@suse.de> * * This file is released under the GPLv2. */#include <linux/device.h>#include <linux/module.h>#include "base.h"struct devres_node {	struct list_head		entry;	dr_release_t			release;#ifdef CONFIG_DEBUG_DEVRES	const char			*name;	size_t				size;#endif};struct devres {	struct devres_node		node;	/* -- 3 pointers */	unsigned long long		data[];	/* guarantee ull alignment */};struct devres_group {	struct devres_node		node[2];	void				*id;	int				color;	/* -- 8 pointers */};#ifdef CONFIG_DEBUG_DEVRESstatic int log_devres = 0;module_param_named(log, log_devres, int, S_IRUGO | S_IWUSR);static void set_node_dbginfo(struct devres_node *node, const char *name,			     size_t size){	node->name = name;	node->size = size;}static void devres_log(struct device *dev, struct devres_node *node,		       const char *op){	if (unlikely(log_devres))		dev_printk(KERN_ERR, dev, "DEVRES %3s %p %s (%lu bytes)\n",			   op, node, node->name, (unsigned long)node->size);}#else /* CONFIG_DEBUG_DEVRES */#define set_node_dbginfo(node, n, s)	do {} while (0)#define devres_log(dev, node, op)	do {} while (0)#endif /* CONFIG_DEBUG_DEVRES *//* * Release functions for devres group.  These callbacks are used only * for identification. */static void group_open_release(struct device *dev, void *res){	/* noop */}static void group_close_release(struct device *dev, void *res){	/* noop */}static struct devres_group * node_to_group(struct devres_node *node){	if (node->release == &group_open_release)		return container_of(node, struct devres_group, node[0]);	if (node->release == &group_close_release)		return container_of(node, struct devres_group, node[1]);	return NULL;}static __always_inline struct devres * alloc_dr(dr_release_t release,						size_t size, gfp_t gfp){	size_t tot_size = sizeof(struct devres) + size;	struct devres *dr;	dr = kmalloc_track_caller(tot_size, gfp);	if (unlikely(!dr))		return NULL;	memset(dr, 0, tot_size);	INIT_LIST_HEAD(&dr->node.entry);	dr->node.release = release;	return dr;}static void add_dr(struct device *dev, struct devres_node *node){	devres_log(dev, node, "ADD");	BUG_ON(!list_empty(&node->entry));	list_add_tail(&node->entry, &dev->devres_head);}#ifdef CONFIG_DEBUG_DEVRESvoid * __devres_alloc(dr_release_t release, size_t size, gfp_t gfp,		      const char *name){	struct devres *dr;	dr = alloc_dr(release, size, gfp);	if (unlikely(!dr))		return NULL;	set_node_dbginfo(&dr->node, name, size);	return dr->data;}EXPORT_SYMBOL_GPL(__devres_alloc);#else/** * devres_alloc - Allocate device resource data * @release: Release function devres will be associated with * @size: Allocation size * @gfp: Allocation flags * * Allocate devres of @size bytes.  The allocated area is zeroed, then * associated with @release.  The returned pointer can be passed to * other devres_*() functions. * * RETURNS: * Pointer to allocated devres on success, NULL on failure. */void * devres_alloc(dr_release_t release, size_t size, gfp_t gfp){	struct devres *dr;	dr = alloc_dr(release, size, gfp);	if (unlikely(!dr))		return NULL;	return dr->data;}EXPORT_SYMBOL_GPL(devres_alloc);#endif/** * devres_free - Free device resource data * @res: Pointer to devres data to free * * Free devres created with devres_alloc(). */void devres_free(void *res){	if (res) {		struct devres *dr = container_of(res, struct devres, data);		BUG_ON(!list_empty(&dr->node.entry));		kfree(dr);	}}EXPORT_SYMBOL_GPL(devres_free);/** * devres_add - Register device resource * @dev: Device to add resource to * @res: Resource to register * * Register devres @res to @dev.  @res should have been allocated * using devres_alloc().  On driver detach, the associated release * function will be invoked and devres will be freed automatically. */void devres_add(struct device *dev, void *res){	struct devres *dr = container_of(res, struct devres, data);	unsigned long flags;	spin_lock_irqsave(&dev->devres_lock, flags);	add_dr(dev, &dr->node);	spin_unlock_irqrestore(&dev->devres_lock, flags);}EXPORT_SYMBOL_GPL(devres_add);static struct devres *find_dr(struct device *dev, dr_release_t release,			      dr_match_t match, void *match_data){	struct devres_node *node;	list_for_each_entry_reverse(node, &dev->devres_head, entry) {		struct devres *dr = container_of(node, struct devres, node);		if (node->release != release)			continue;		if (match && !match(dev, dr->data, match_data))			continue;		return dr;	}	return NULL;}/** * devres_find - Find device resource * @dev: Device to lookup resource from * @release: Look for resources associated with this release function * @match: Match function (optional) * @match_data: Data for the match function * * Find the latest devres of @dev which is associated with @release * and for which @match returns 1.  If @match is NULL, it's considered * to match all. * * RETURNS: * Pointer to found devres, NULL if not found. */void * devres_find(struct device *dev, dr_release_t release,		   dr_match_t match, void *match_data){	struct devres *dr;	unsigned long flags;	spin_lock_irqsave(&dev->devres_lock, flags);	dr = find_dr(dev, release, match, match_data);	spin_unlock_irqrestore(&dev->devres_lock, flags);	if (dr)		return dr->data;	return NULL;}EXPORT_SYMBOL_GPL(devres_find);/** * devres_get - Find devres, if non-existent, add one atomically * @dev: Device to lookup or add devres for * @new_res: Pointer to new initialized devres to add if not found * @match: Match function (optional) * @match_data: Data for the match function * * Find the latest devres of @dev which has the same release function * as @new_res and for which @match return 1.  If found, @new_res is * freed; otherwise, @new_res is added atomically. * * RETURNS: * Pointer to found or added devres. */void * devres_get(struct device *dev, void *new_res,		  dr_match_t match, void *match_data){	struct devres *new_dr = container_of(new_res, struct devres, data);	struct devres *dr;	unsigned long flags;	spin_lock_irqsave(&dev->devres_lock, flags);	dr = find_dr(dev, new_dr->node.release, match, match_data);	if (!dr) {		add_dr(dev, &new_dr->node);		dr = new_dr;		new_dr = NULL;	}	spin_unlock_irqrestore(&dev->devres_lock, flags);	devres_free(new_dr);	return dr->data;}EXPORT_SYMBOL_GPL(devres_get);/** * devres_remove - Find a device resource and remove it * @dev: Device to find resource from * @release: Look for resources associated with this release function * @match: Match function (optional) * @match_data: Data for the match function * * Find the latest devres of @dev associated with @release and for * which @match returns 1.  If @match is NULL, it's considered to * match all.  If found, the resource is removed atomically and * returned. * * RETURNS: * Pointer to removed devres on success, NULL if not found. */void * devres_remove(struct device *dev, dr_release_t release,		     dr_match_t match, void *match_data){	struct devres *dr;	unsigned long flags;	spin_lock_irqsave(&dev->devres_lock, flags);	dr = find_dr(dev, release, match, match_data);	if (dr) {		list_del_init(&dr->node.entry);		devres_log(dev, &dr->node, "REM");	}	spin_unlock_irqrestore(&dev->devres_lock, flags);	if (dr)		return dr->data;	return NULL;}EXPORT_SYMBOL_GPL(devres_remove);/** * devres_destroy - Find a device resource and destroy it * @dev: Device to find resource from * @release: Look for resources associated with this release function * @match: Match function (optional) * @match_data: Data for the match function * * Find the latest devres of @dev associated with @release and for * which @match returns 1.  If @match is NULL, it's considered to * match all.  If found, the resource is removed atomically and freed. * * RETURNS: * 0 if devres is found and freed, -ENOENT if not found. */int devres_destroy(struct device *dev, dr_release_t release,		   dr_match_t match, void *match_data){	void *res;	res = devres_remove(dev, release, match, match_data);	if (unlikely(!res))		return -ENOENT;	devres_free(res);	return 0;

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?