class.c 添加中文注释(3)
int class_device_register(struct class_device *class_dev)
{
/* [cgw]: 初始化一个struct class_device */
class_device_initialize(class_dev);
/* [cgw]: 添加一个struct class_device */
return class_device_add(class_dev);
} /**
* class_device_create - creates a class device and registers it with sysfs
* @cls: pointer to the struct class that this device should be registered to.
* @parent: pointer to the parent struct class_device of this new device, if any.
* @devt: the dev_t for the char device to be added.
* @device: a pointer to a struct device that is assiociated with this class device.
* @fmt: string for the class device's name
*
* This function can be used by char device classes. A struct
* class_device will be created in sysfs, registered to the specified
* class.
* A "dev" file will be created, showing the dev_t for the device, if
* the dev_t is not 0,0.
* If a pointer to a parent struct class_device is passed in, the newly
* created struct class_device will be a child of that device in sysfs.
* The pointer to the struct class_device will be returned from the
* call. Any further sysfs files that might be required can be created
* using this pointer.
*
* Note: the struct class passed to this function must have previously
* been created with a call to class_create().
*/
struct class_device *class_device_create(struct class *cls,
struct class_device *parent,
dev_t devt,
struct device *device,
const char *fmt, ...)
{
va_list args;
struct class_device *class_dev = NULL;
int retval = -ENODEV; /* [cgw]: cls为空或错误 */
if (cls == NULL || IS_ERR(cls))
goto error; /* [cgw]: 分配sizeof(*class_dev)字节大小的内存空间 */
class_dev = kzalloc(sizeof(*class_dev), GFP_KERNEL);
/* [cgw]: 分配失败 */
if (!class_dev) {
retval = -ENOMEM;
goto error;
} /* [cgw]: 分配设备号 */
class_dev->devt = devt;
/* [cgw]: 分配一个struct device */
class_dev->dev = device;
/* [cgw]: 分配一个struct class */
class_dev->class = cls;
/* [cgw]: 分配一个parent struct class_device */
class_dev->parent = parent;
/* [cgw]: 分配release方法 */
class_dev->release = class_device_create_release;
/* [cgw]: 分配uevent方法 */
class_dev->uevent = class_device_create_uevent; va_start(args, fmt);
/* [cgw]: 把args格式化到字符串fmt中,填装到class_dev->class_id */
vsnprintf(class_dev->class_id, BUS_ID_SIZE, fmt, args);
va_end(args);
/* [cgw]: 注册一个class_dev */
retval = class_device_register(class_dev);
if (retval)
goto error; return class_dev; error:
/* [cgw]: 释放class_dev的内存空间 */
kfree(class_dev);
return ERR_PTR(retval);
} void class_device_del(struct class_device *class_dev)
{
struct class *parent_class = class_dev->class;
struct class_device *parent_device = class_dev->parent;
struct class_interface *class_intf; /* [cgw]: parent_class不为空 */
if (parent_class) {
/* [cgw]: 获取信号量 */
down(&parent_class->sem);
/* [cgw]: 删除class_dev->node节点 */
list_del_init(&class_dev->node);
/* [cgw]: 历遍class_intf->node链表,直到回到parent_class->interfaces节点 */
list_for_each_entry(class_intf, &parent_class->interfaces, node)
/* [cgw]: class_intf->remove指针不为空 */
if (class_intf->remove)
/* [cgw]: 调用remove方法 */
class_intf->remove(class_dev, class_intf);
/* [cgw]: 释放信号量 */
up(&parent_class->sem);
} /* [cgw]: class_dev->dev指针不为空 */
if (class_dev->dev) {
/* [cgw] : 从class_dev->dev->kobj对象目录下删除一个名字为
* class_dev->class->name+class_dev->kobj->k_name的链表
*/
remove_deprecated_class_device_links(class_dev);
/* [cgw]: class_dev->kobj目录下,删除名为device的链表 */
sysfs_remove_link(&class_dev->kobj, "device");
}
/* [cgw]: class_dev->kobj目录下,删除名为subsystem的链表 */
sysfs_remove_link(&class_dev->kobj, "subsystem"); /* [cgw]: 删除class_dev->kobj对象对应的一个class_dev->uevent_attr->attr
* 属性文件
*/
class_device_remove_file(class_dev, &class_dev->uevent_attr);
/* [cgw]: class_dev->devt_attr指针不为空 */
if (class_dev->devt_attr)
/* [cgw]: 删除class_dev->kobj对象对应的一个class_dev->devt_attr->attr
* 属性文件
*/
class_device_remove_file(class_dev, class_dev->devt_attr);
/* [cgw]: 历遍class_dev->class->class_dev_attrs[]数组,如果该属性名字不为空,
* 则对应地删除一个属性文件
*/
class_device_remove_attrs(class_dev);
/* [cgw]: 历遍class_dev->groups[],删除所有属性组 */
class_device_remove_groups(class_dev); /* [cgw]: 通知用户空间,产生一个KOBJ_REMOVE事件 */
kobject_uevent(&class_dev->kobj, KOBJ_REMOVE);
/* [cgw]: 删除kobj对象目录,从kset链表中删除class_dev->kobj */
kobject_del(&class_dev->kobj); /* [cgw]: parent_device->kobj引用计数-1 */
class_device_put(parent_device);
/* [cgw]: parent_class->kobj引用计数-1 */
class_put(parent_class);
} void class_device_unregister(struct class_device *class_dev)
{
pr_debug("CLASS: Unregistering class device. ID = '%s'\n",
class_dev->class_id);
/* [cgw]: 删除struct class_device */
class_device_del(class_dev);
/* [cgw]: class_dev->kobj引用计数-1 */
class_device_put(class_dev);
} /**
* class_device_destroy - removes a class device that was created with class_device_create()
* @cls: the pointer to the struct class that this device was registered * with.
* @devt: the dev_t of the device that was previously registered.
*
* This call unregisters and cleans up a class device that was created with a
* call to class_device_create()
*/
void class_device_destroy(struct class *cls, dev_t devt)
{
struct class_device *class_dev = NULL;
struct class_device *class_dev_tmp; /* [cgw]: 获得信号量 */
down(&cls->sem);
/* [cgw]: 返回class_dev_tmp指针,历遍class_dev_tmp链表,
* 查找与cls->children相等的节点class_dev_tmp->node
*/
list_for_each_entry(class_dev_tmp, &cls->children, node) {
/* [cgw]: 找到该节点,设备号为devt */
if (class_dev_tmp->devt == devt) {
/* [cgw]: class_dev指向class_dev_tmp */
class_dev = class_dev_tmp;
break;
}
}
/* [cgw]: 释放信号量 */
up(&cls->sem); /* [cgw]: class_dev指针不为空 */
if (class_dev)
/* [cgw]: 注销这个class_dev */
class_device_unregister(class_dev);
} struct class_device * class_device_get(struct class_device *class_dev)
{
/* [cgw]: class_dev指针不为空 */
if (class_dev)
/* [cgw]: class_dev->kobj引用计数+1
* 并根据kobj找到包含这个kobj的结构体指针
*/
return to_class_dev(kobject_get(&class_dev->kobj));
return NULL;
} void class_device_put(struct class_device *class_dev)
{
/* [cgw]: class_dev指针不为空 */
if (class_dev)
/* [cgw]: class_dev->kobj引用计数-1 */
kobject_put(&class_dev->kobj);
} int class_interface_register(struct class_interface *class_intf)
{
struct class *parent;
struct class_device *class_dev;
struct device *dev; /* [cgw]: class_intf或class_intf->class指针为空 */
if (!class_intf || !class_intf->class)
return -ENODEV; /* [cgw]: class_intf->class->subsys->kset.kobj引用计数+1,
* parent指向class_intf->class
*/
parent = class_get(class_intf->class);
/* [cgw]: parent指针为空 */
if (!parent)
return -EINVAL; /* [cgw]: 获取信号量 */
down(&parent->sem);
/* [cgw]: 添加一个新节点class_intf->node,位于parent->interfaces节点前 */
list_add_tail(&class_intf->node, &parent->interfaces); /* [cgw]: class_intf->add指针不为空 */
if (class_intf->add) {
/* [cgw]: 返回class_dev指针,历遍class_dev链表,
* 调用每个节点node对应的add方法
*/
list_for_each_entry(class_dev, &parent->children, node)
/* [cgw]: 调用与node (class_dev)对应的add方法 */
class_intf->add(class_dev, class_intf);
} /* [cgw]: class_intf->add_dev指针不为空 */
if (class_intf->add_dev) {
/* [cgw]: 返回dev指针,历遍dev链表,
* 调用每个节点node对应的add_dev方法
*/
list_for_each_entry(dev, &parent->devices, node)
/* [cgw]: 调用与node (dev)对应的add_dev方法 */
class_intf->add_dev(dev, class_intf);
}
/* [cgw]: 释放信号量 */
up(&parent->sem); return ;
} void class_interface_unregister(struct class_interface *class_intf)
{
/* [cgw]: parent指针指向class_intf->class */
struct class * parent = class_intf->class;
struct class_device *class_dev;
struct device *dev; /* [cgw]: parent指针为空 */
if (!parent)
return; /* [cgw]: 获取信号量 */
down(&parent->sem);
/* [cgw]: 删除一个节点class_intf->node */
list_del_init(&class_intf->node);
/* [cgw]: class_intf->remove指针不为空 */
if (class_intf->remove) {
/* [cgw]: 返回class_dev指针,历遍class_dev链表,
* 调用每个节点node对应的remove方法
*/
list_for_each_entry(class_dev, &parent->children, node)
/* [cgw]: 调用与node (class_dev)对应的remove方法 */
class_intf->remove(class_dev, class_intf);
} /* [cgw]: class_intf->remove_dev指针不为空 */
if (class_intf->remove_dev) {
/* [cgw]: 返回dev指针,历遍dev链表,
* 调用每个节点node对应的remove_dev方法
*/
list_for_each_entry(dev, &parent->devices, node)
/* [cgw]: 调用与node (class_dev)对应的remove_dev方法 */
class_intf->remove_dev(dev, class_intf);
}
/* [cgw]: 释放信号量 */
up(&parent->sem); /* [cgw]: parent->subsys->kobj引用计数-1 */
class_put(parent);
} int __init classes_init(void)
{
int retval; retval = subsystem_register(&class_subsys);
if (retval)
return retval; /* ick, this is ugly, the things we go through to keep from showing up
* in sysfs... */
subsystem_init(&class_obj_subsys);
if (!class_obj_subsys.kobj.parent)
class_obj_subsys.kobj.parent = &class_obj_subsys.kobj;
return ;
} EXPORT_SYMBOL_GPL(class_create_file);
EXPORT_SYMBOL_GPL(class_remove_file);
EXPORT_SYMBOL_GPL(class_register);
EXPORT_SYMBOL_GPL(class_unregister);
EXPORT_SYMBOL_GPL(class_create);
EXPORT_SYMBOL_GPL(class_destroy); EXPORT_SYMBOL_GPL(class_device_register);
EXPORT_SYMBOL_GPL(class_device_unregister);
EXPORT_SYMBOL_GPL(class_device_initialize);
EXPORT_SYMBOL_GPL(class_device_add);
EXPORT_SYMBOL_GPL(class_device_del);
EXPORT_SYMBOL_GPL(class_device_get);
EXPORT_SYMBOL_GPL(class_device_put);
EXPORT_SYMBOL_GPL(class_device_create);
EXPORT_SYMBOL_GPL(class_device_destroy);
EXPORT_SYMBOL_GPL(class_device_create_file);
EXPORT_SYMBOL_GPL(class_device_remove_file);
EXPORT_SYMBOL_GPL(class_device_create_bin_file);
EXPORT_SYMBOL_GPL(class_device_remove_bin_file); EXPORT_SYMBOL_GPL(class_interface_register);
EXPORT_SYMBOL_GPL(class_interface_unregister);
class.c 添加中文注释(3)的更多相关文章
- Python中添加中文注释报错SyntaxError: Non-UTF-8 code starting with '\xc1'
问题:在文本编辑器中编辑Python文件时添加中文注释,运行python文件时报错.SyntaxError: Non-UTF-8 code starting with '\xc1' 解决方法:在文本开 ...
- char_dev.c 添加中文注释
char_dev.c里的中文注释,仅代表个人理解,仅供参考.如有错误之处,请指出,谢谢! /* * linux/fs/char_dev.c * * Copyright (C) 1991, 1992 L ...
- 在mapper.xml映射文件中添加中文注释报错
问题描述: 在写mapper.xml文件时,想给操作数据库语句添加一些中文注释,添加后运行报如下错误: 思考 可能是写了中文注释,编译器在解析xml文件时,未能成功转码,从而导致乱码.但是文件开头也采 ...
- class.c 添加中文注释(1)
注释仅代表个人理解,仅供参考. /* * class.c - basic device class management * * Copyright (c) 2002-3 Patrick Mochel ...
- class.c 添加中文注释(2)
/* Class Device Stuff */ int class_device_create_file(struct class_device * class_dev, const struct ...
- sass中文注释的解决方法和一些简单用法
最近用sass来编写项目中的css,发现不能添加中文注释,报错如下 于是查阅了一下发现需要在scss文件顶部加上@charset "utf-8"即可解决. 在此顺便记录一些sass ...
- 关于 Keil uVision2 中文注释会显示不完整,字体不正常的问题
在Keil中添加中文注释经常出现这样情况: ,注释文字不正常! 解决方案:Edit---->Option----->选择color&fonts选项卡中的Editor c Files ...
- Eclipse MyBatis generator 1.3.7插件的核心包(中文注释)
一.最近刚搭建一个项目框架,使用springboot + mybatis,但是在使用Eclipse开发时发现开发mybatis的Dao.mapper.xml和entity时特别不方便,手工去写肯定是不 ...
- mysql数据库导出模型到powerdesigner,PDM图形窗口中显示数据列的中文注释
1,mysql数据库导出模型到powerdesigner 2,CRL+Shift+X 3,复制以下内容,执行 '******************************************** ...
随机推荐
- Android笔记——Bundle类的作用
Bundle类用作携带数据,它类似于Map,用于存放key-value键值对形式的值.相对于Map,它提供了各种常用类型的putXxx()/getXxx()方法,如:putString()/getSt ...
- 编程之美—烙饼排序问题(JAVA)
一.问题描述 星期五的晚上,一帮同事在希格玛大厦附近的"硬盘酒吧"多喝了几杯.程序员多喝了几杯之后谈什么呢?自然是算法问题.有个同事说:"我以前在餐 馆打工,顾 ...
- ASP.NET Core管道深度剖析(3):管道是如何处理HTTP请求的?
我们知道ASP.NET Core请求处理管道由一个服务器和一组有序的中间件组成,所以从总体设计来讲是非常简单的,但是就具体的实现来说,由于其中涉及很多对象的交互,我想很少人能够地把它弄清楚.为了让读者 ...
- YYModel 源码解读(二)之YYClassInfo.h (1)
NS_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_END 为了兼容Swift 中的 ? 和 ! oc 在6.3引入了两个新的类型注释:__nullable和__non ...
- 【NLP】Python NLTK 走进大秦帝国
Python NLTK 走进大秦帝国 作者:白宁超 2016年10月17日18:54:10 摘要:NLTK是由宾夕法尼亚大学计算机和信息科学使用python语言实现的一种自然语言工具包,其收集的大量公 ...
- 探寻<a>中的href和onclick
一.知识点: onclick的方法参数必须加引号 href跳转参数有长度限制 href中执行js会把encodeURIComponent()编码之后的东西自动解码,有时会影响参数传递 a标签中的onc ...
- Events基本概念----Beginning Visual C#
span.kw { color: #007020; font-weight: bold; } code > span.dt { color: #902000; } code > span. ...
- jQuery中取消后续执行的内容
<html xmlns="http://www.w3.org/1999/xhtml"><head> <title></title&g ...
- ChatSecure
ChatSecure是一个免费和开源消息传递应用程序的特性 OTR加密在 XMPP. 你可以连接到现有的谷歌帐户或创建新帐户 公共XMPP服务器(包括通过 Tor),甚至 连接到您自己的服务器额外的安 ...
- .NET文件并发与RabbitMQ(初探RabbitMQ)
本文版权归博客园和作者吴双本人共同所有.欢迎转载,转载和爬虫请注明原文地址:http://www.cnblogs.com/tdws/p/5860668.html 想必MQ这两个字母对于各位前辈们和老司 ...