component: add debugfs support
Currently there is no information in any vfs about which devices a master component consists of, what makes debugging hard if one of the component devices fails to register. Add 'device_component' directory to debugfs. Create a new file for each component master, when it has been added. Remove it on a master deletion. Show a list of devices required by the given master and their status (registered or not). This provides an easy way to check, which device has failed to register if the given master device is not available in the system. Signed-off-by: Maciej Purski <m.purski@samsung.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
0a006e864e
commit
59e73854b5
|
@ -14,6 +14,7 @@
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
#include <linux/debugfs.h>
|
||||||
|
|
||||||
struct component;
|
struct component;
|
||||||
|
|
||||||
|
@ -38,6 +39,7 @@ struct master {
|
||||||
const struct component_master_ops *ops;
|
const struct component_master_ops *ops;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct component_match *match;
|
struct component_match *match;
|
||||||
|
struct dentry *dentry;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct component {
|
struct component {
|
||||||
|
@ -53,6 +55,80 @@ static DEFINE_MUTEX(component_mutex);
|
||||||
static LIST_HEAD(component_list);
|
static LIST_HEAD(component_list);
|
||||||
static LIST_HEAD(masters);
|
static LIST_HEAD(masters);
|
||||||
|
|
||||||
|
#ifdef CONFIG_DEBUG_FS
|
||||||
|
|
||||||
|
static struct dentry *component_debugfs_dir;
|
||||||
|
|
||||||
|
static int component_devices_show(struct seq_file *s, void *data)
|
||||||
|
{
|
||||||
|
struct master *m = s->private;
|
||||||
|
struct component_match *match = m->match;
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
mutex_lock(&component_mutex);
|
||||||
|
seq_printf(s, "%-40s %20s\n", "master name", "status");
|
||||||
|
seq_puts(s, "-------------------------------------------------------------\n");
|
||||||
|
seq_printf(s, "%-40s %20s\n\n",
|
||||||
|
dev_name(m->dev), m->bound ? "bound" : "not bound");
|
||||||
|
|
||||||
|
seq_printf(s, "%-40s %20s\n", "device name", "status");
|
||||||
|
seq_puts(s, "-------------------------------------------------------------\n");
|
||||||
|
for (i = 0; i < match->num; i++) {
|
||||||
|
struct device *d = (struct device *)match->compare[i].data;
|
||||||
|
|
||||||
|
seq_printf(s, "%-40s %20s\n", dev_name(d),
|
||||||
|
match->compare[i].component ?
|
||||||
|
"registered" : "not registered");
|
||||||
|
}
|
||||||
|
mutex_unlock(&component_mutex);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int component_devices_open(struct inode *inode, struct file *file)
|
||||||
|
{
|
||||||
|
return single_open(file, component_devices_show, inode->i_private);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct file_operations component_devices_fops = {
|
||||||
|
.open = component_devices_open,
|
||||||
|
.read = seq_read,
|
||||||
|
.llseek = seq_lseek,
|
||||||
|
.release = single_release,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int __init component_debug_init(void)
|
||||||
|
{
|
||||||
|
component_debugfs_dir = debugfs_create_dir("device_component", NULL);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
core_initcall(component_debug_init);
|
||||||
|
|
||||||
|
static void component_master_debugfs_add(struct master *m)
|
||||||
|
{
|
||||||
|
m->dentry = debugfs_create_file(dev_name(m->dev), 0444,
|
||||||
|
component_debugfs_dir,
|
||||||
|
m, &component_devices_fops);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void component_master_debugfs_del(struct master *m)
|
||||||
|
{
|
||||||
|
debugfs_remove(m->dentry);
|
||||||
|
m->dentry = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static void component_master_debugfs_add(struct master *m)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
static void component_master_debugfs_del(struct master *m)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
static struct master *__master_find(struct device *dev,
|
static struct master *__master_find(struct device *dev,
|
||||||
const struct component_master_ops *ops)
|
const struct component_master_ops *ops)
|
||||||
{
|
{
|
||||||
|
@ -287,6 +363,7 @@ static void free_master(struct master *master)
|
||||||
struct component_match *match = master->match;
|
struct component_match *match = master->match;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
component_master_debugfs_del(master);
|
||||||
list_del(&master->node);
|
list_del(&master->node);
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
|
@ -320,6 +397,7 @@ int component_master_add_with_match(struct device *dev,
|
||||||
master->ops = ops;
|
master->ops = ops;
|
||||||
master->match = match;
|
master->match = match;
|
||||||
|
|
||||||
|
component_master_debugfs_add(master);
|
||||||
/* Add to the list of available masters. */
|
/* Add to the list of available masters. */
|
||||||
mutex_lock(&component_mutex);
|
mutex_lock(&component_mutex);
|
||||||
list_add(&master->node, &masters);
|
list_add(&master->node, &masters);
|
||||||
|
|
Loading…
Reference in New Issue