core.c
来自「linux 内核源代码」· C语言 代码 · 共 1,357 行 · 第 1/3 页
C
1,357 行
/* * drivers/base/core.c - core driver model code (device registration, etc) * * Copyright (c) 2002-3 Patrick Mochel * Copyright (c) 2002-3 Open Source Development Labs * Copyright (c) 2006 Greg Kroah-Hartman <gregkh@suse.de> * Copyright (c) 2006 Novell, Inc. * * This file is released under the GPLv2 * */#include <linux/device.h>#include <linux/err.h>#include <linux/init.h>#include <linux/module.h>#include <linux/slab.h>#include <linux/string.h>#include <linux/kdev_t.h>#include <linux/notifier.h>#include <asm/semaphore.h>#include "base.h"#include "power/power.h"int (*platform_notify)(struct device * dev) = NULL;int (*platform_notify_remove)(struct device * dev) = NULL;/* * sysfs bindings for devices. *//** * dev_driver_string - Return a device's driver name, if at all possible * @dev: struct device to get the name of * * Will return the device's driver's name if it is bound to a device. If * the device is not bound to a device, it will return the name of the bus * it is attached to. If it is not attached to a bus either, an empty * string will be returned. */const char *dev_driver_string(struct device *dev){ return dev->driver ? dev->driver->name : (dev->bus ? dev->bus->name : (dev->class ? dev->class->name : ""));}EXPORT_SYMBOL(dev_driver_string);#define to_dev(obj) container_of(obj, struct device, kobj)#define to_dev_attr(_attr) container_of(_attr, struct device_attribute, attr)static ssize_tdev_attr_show(struct kobject * kobj, struct attribute * attr, char * buf){ struct device_attribute * dev_attr = to_dev_attr(attr); struct device * dev = to_dev(kobj); ssize_t ret = -EIO; if (dev_attr->show) ret = dev_attr->show(dev, dev_attr, buf); return ret;}static ssize_tdev_attr_store(struct kobject * kobj, struct attribute * attr, const char * buf, size_t count){ struct device_attribute * dev_attr = to_dev_attr(attr); struct device * dev = to_dev(kobj); ssize_t ret = -EIO; if (dev_attr->store) ret = dev_attr->store(dev, dev_attr, buf, count); return ret;}static struct sysfs_ops dev_sysfs_ops = { .show = dev_attr_show, .store = dev_attr_store,};/** * device_release - free device structure. * @kobj: device's kobject. * * This is called once the reference count for the object * reaches 0. We forward the call to the device's release * method, which should handle actually freeing the structure. */static void device_release(struct kobject * kobj){ struct device * dev = to_dev(kobj); if (dev->release) dev->release(dev); else if (dev->type && dev->type->release) dev->type->release(dev); else if (dev->class && dev->class->dev_release) dev->class->dev_release(dev); else { printk(KERN_ERR "Device '%s' does not have a release() function, " "it is broken and must be fixed.\n", dev->bus_id); WARN_ON(1); }}static struct kobj_type device_ktype = { .release = device_release, .sysfs_ops = &dev_sysfs_ops,};static int dev_uevent_filter(struct kset *kset, struct kobject *kobj){ struct kobj_type *ktype = get_ktype(kobj); if (ktype == &device_ktype) { struct device *dev = to_dev(kobj); if (dev->uevent_suppress) return 0; if (dev->bus) return 1; if (dev->class) return 1; } return 0;}static const char *dev_uevent_name(struct kset *kset, struct kobject *kobj){ struct device *dev = to_dev(kobj); if (dev->bus) return dev->bus->name; if (dev->class) return dev->class->name; return NULL;}static int dev_uevent(struct kset *kset, struct kobject *kobj, struct kobj_uevent_env *env){ struct device *dev = to_dev(kobj); int retval = 0; /* add the major/minor if present */ if (MAJOR(dev->devt)) { add_uevent_var(env, "MAJOR=%u", MAJOR(dev->devt)); add_uevent_var(env, "MINOR=%u", MINOR(dev->devt)); } if (dev->type && dev->type->name) add_uevent_var(env, "DEVTYPE=%s", dev->type->name); if (dev->driver) add_uevent_var(env, "DRIVER=%s", dev->driver->name);#ifdef CONFIG_SYSFS_DEPRECATED if (dev->class) { struct device *parent = dev->parent; /* find first bus device in parent chain */ while (parent && !parent->bus) parent = parent->parent; if (parent && parent->bus) { const char *path; path = kobject_get_path(&parent->kobj, GFP_KERNEL); if (path) { add_uevent_var(env, "PHYSDEVPATH=%s", path); kfree(path); } add_uevent_var(env, "PHYSDEVBUS=%s", parent->bus->name); if (parent->driver) add_uevent_var(env, "PHYSDEVDRIVER=%s", parent->driver->name); } } else if (dev->bus) { add_uevent_var(env, "PHYSDEVBUS=%s", dev->bus->name); if (dev->driver) add_uevent_var(env, "PHYSDEVDRIVER=%s", dev->driver->name); }#endif /* have the bus specific function add its stuff */ if (dev->bus && dev->bus->uevent) { retval = dev->bus->uevent(dev, env); if (retval) pr_debug ("%s: bus uevent() returned %d\n", __FUNCTION__, retval); } /* have the class specific function add its stuff */ if (dev->class && dev->class->dev_uevent) { retval = dev->class->dev_uevent(dev, env); if (retval) pr_debug("%s: class uevent() returned %d\n", __FUNCTION__, retval); } /* have the device type specific fuction add its stuff */ if (dev->type && dev->type->uevent) { retval = dev->type->uevent(dev, env); if (retval) pr_debug("%s: dev_type uevent() returned %d\n", __FUNCTION__, retval); } return retval;}static struct kset_uevent_ops device_uevent_ops = { .filter = dev_uevent_filter, .name = dev_uevent_name, .uevent = dev_uevent,};static ssize_t show_uevent(struct device *dev, struct device_attribute *attr, char *buf){ struct kobject *top_kobj; struct kset *kset; struct kobj_uevent_env *env = NULL; int i; size_t count = 0; int retval; /* search the kset, the device belongs to */ top_kobj = &dev->kobj; while (!top_kobj->kset && top_kobj->parent) top_kobj = top_kobj->parent; if (!top_kobj->kset) goto out; kset = top_kobj->kset; if (!kset->uevent_ops || !kset->uevent_ops->uevent) goto out; /* respect filter */ if (kset->uevent_ops && kset->uevent_ops->filter) if (!kset->uevent_ops->filter(kset, &dev->kobj)) goto out; env = kzalloc(sizeof(struct kobj_uevent_env), GFP_KERNEL); if (!env) return -ENOMEM; /* let the kset specific function add its keys */ retval = kset->uevent_ops->uevent(kset, &dev->kobj, env); if (retval) goto out; /* copy keys to file */ for (i = 0; i < env->envp_idx; i++) count += sprintf(&buf[count], "%s\n", env->envp[i]);out: kfree(env); return count;}static ssize_t store_uevent(struct device *dev, struct device_attribute *attr, const char *buf, size_t count){ enum kobject_action action; if (kobject_action_type(buf, count, &action) == 0) { kobject_uevent(&dev->kobj, action); goto out; } dev_err(dev, "uevent: unsupported action-string; this will " "be ignored in a future kernel version\n"); kobject_uevent(&dev->kobj, KOBJ_ADD);out: return count;}static struct device_attribute uevent_attr = __ATTR(uevent, S_IRUGO | S_IWUSR, show_uevent, store_uevent);static int device_add_attributes(struct device *dev, struct device_attribute *attrs){ int error = 0; int i; if (attrs) { for (i = 0; attr_name(attrs[i]); i++) { error = device_create_file(dev, &attrs[i]); if (error) break; } if (error) while (--i >= 0) device_remove_file(dev, &attrs[i]); } return error;}static void device_remove_attributes(struct device *dev, struct device_attribute *attrs){ int i; if (attrs) for (i = 0; attr_name(attrs[i]); i++) device_remove_file(dev, &attrs[i]);}static int device_add_groups(struct device *dev, struct attribute_group **groups){ int error = 0; int i; if (groups) { for (i = 0; groups[i]; i++) { error = sysfs_create_group(&dev->kobj, groups[i]); if (error) { while (--i >= 0) sysfs_remove_group(&dev->kobj, groups[i]); break; } } } return error;}static void device_remove_groups(struct device *dev, struct attribute_group **groups){ int i; if (groups) for (i = 0; groups[i]; i++) sysfs_remove_group(&dev->kobj, groups[i]);}static int device_add_attrs(struct device *dev){ struct class *class = dev->class; struct device_type *type = dev->type; int error; if (class) { error = device_add_attributes(dev, class->dev_attrs); if (error) return error; } if (type) { error = device_add_groups(dev, type->groups); if (error) goto err_remove_class_attrs; } error = device_add_groups(dev, dev->groups); if (error) goto err_remove_type_groups; return 0; err_remove_type_groups: if (type) device_remove_groups(dev, type->groups); err_remove_class_attrs: if (class) device_remove_attributes(dev, class->dev_attrs); return error;}static void device_remove_attrs(struct device *dev){ struct class *class = dev->class; struct device_type *type = dev->type; device_remove_groups(dev, dev->groups); if (type) device_remove_groups(dev, type->groups); if (class) device_remove_attributes(dev, class->dev_attrs);}static ssize_t show_dev(struct device *dev, struct device_attribute *attr, char *buf){ return print_dev_t(buf, dev->devt);}static struct device_attribute devt_attr = __ATTR(dev, S_IRUGO, show_dev, NULL);/* * devices_subsys - structure to be registered with kobject core. */decl_subsys(devices, &device_ktype, &device_uevent_ops);/** * device_create_file - create sysfs attribute file for device. * @dev: device. * @attr: device attribute descriptor. */int device_create_file(struct device * dev, struct device_attribute * attr){ int error = 0; if (get_device(dev)) { error = sysfs_create_file(&dev->kobj, &attr->attr); put_device(dev); } return error;}/** * device_remove_file - remove sysfs attribute file. * @dev: device. * @attr: device attribute descriptor. */void device_remove_file(struct device * dev, struct device_attribute * attr){ if (get_device(dev)) { sysfs_remove_file(&dev->kobj, &attr->attr); put_device(dev); }}/** * device_create_bin_file - create sysfs binary attribute file for device. * @dev: device. * @attr: device binary attribute descriptor. */int device_create_bin_file(struct device *dev, struct bin_attribute *attr){ int error = -EINVAL; if (dev) error = sysfs_create_bin_file(&dev->kobj, attr); return error;}EXPORT_SYMBOL_GPL(device_create_bin_file);
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?