linux设备驱动模型之 kset原理与实例分析

1、 Kset

       kset是具有相同类型的kobject的集合,在sysfs中体现成一个目录,在内核中用kset数据结构表示,定义为:

struct kset {

struct list_head list; //连接该kset中所有kobject的链表头

spinlock_t list_lock;

struct kobject kobj; //内嵌的kobject

struct kset_uevent_ops *uevent_ops; //处理热插拔事件的操作集合

}


2、 Kset操作

       1)int kset_register(struct kset *kset)

       在内核中注册一个kset

       2)void kset_unregister(struct kset *kset)

       从内核中注销一个kset

3、 热插拔事件

       在Linux系统中,当系统配置发生变化时,如:添加kset到系统;移动kobject, 一个通知会从内核空间发送到用户空间,这就是热插拔事件。热插拔事件会导致用户空间中相应的处理程序(如udev,mdev)被调用, 这些处理程序会通过加载驱动程序, 创建设备节点等来响应热插拔事件。

4、热插拔事件操作集合

      Struct kset_uevent_ops {

int (*filter)(struct kset *kset, struct kobject *kobj);

const char *(*name)(struct kset *kset, struct kobject *kobj);

int (*uevent)(struct kset *kset, struct kobject *kobj,

struct kobj_uevent_env *env);

}

kset_uevent_ops这三个函数什么时候调用?

当该kset所管理的kobject和kset状态发生变化时(如被加入,移动),这三个函数将被调用。(例:kobject_uevent调用)

这三个函数的功能是什么?

       1)filter:决定是否将事件传递到用户空间。如果filter返回0,将不传递事件。(例: uevent_filter)

       2)name:用于将字符串传递给用户空间的热插拔处理程序。

       3)uevent:将用户空间需要的参数添加到环境变量中。(例:dev_uevent)

5、 kset实例分析

    #include <linux/device.h>
    #include <linux/module.h>
    #include <linux/kernel.h>
    #include <linux/init.h>
    #include <linux/string.h>
    #include <linux/sysfs.h>
    #include <linux/stat.h>
    #include <linux/kobject.h>

    MODULE_AUTHOR("yinjiabin");
    MODULE_LICENSE("GPL");

    struct kset *kset_p;
    struct kset kset_c;

    /* 函数声明 */
    void obj_test_release(struct kobject *);
    ssize_t kobj_test_show(struct kobject *,struct attribute *,char *);
    ssize_t kobj_test_store(struct kobject *,struct attribute *,const char *,size_t);

    static struct attribute test_attr =
    {
            .name = "kobj_config",
            .mode = S_IRWXUGO,
    };

    static struct attribute *def_attrs[] =
    {
            &test_attr,
            NULL,
    };

    static struct sysfs_ops obj_test_sysops =
    {
            .show = kobj_test_show,
            .store = kobj_test_store,
    };

    static struct kobj_type ktype =
    {
            .release = obj_test_release,
            .sysfs_ops = &obj_test_sysops,
            .default_attrs = def_attrs,
};

    void obj_test_release(struct kobject *kobject)
    {
            printk("[kobj_test: release!]\n");
    }

    ssize_t kobj_test_show(struct kobject *kobject,struct attribute *attr,char *buf)
    {
            printk("Have show -->\n");
            printk("attrname: %s.\n",attr->name);
            sprintf("buf,%s\n",attr->name);
            return strlen(attr->name) + 2;
    }

    ssize_t kobj_test_store(struct kobject *kobject,struct attribute *attr, const char *buf,size_t size)
    {
            printk("Have store -->\n");
            printk("write: %s.\n",buf);
            return size;
    }

    static int kset_filter(struct kset *kset,struct kobject *kobj)
    {
        printk("Filter: kobj %s.\n",kobj->name);
        return 1;
    }

    static const char *kset_name(struct kset *kset,struct kobject *kobj)
    {
        static char buf[20];
        printk("Name kobj %s.\n",kobj->name);
        sprintf(buf,"%s","kset_name");
        return buf;
    }

    static int kset_uevent(struct kset *kset,struct kobject *kobj, struct kobj_uevent_env *env)

{
        int i = 0;
        printk("uevent: kobj %s.\n",kobj->name);

        while(i < env->envp_idx)
        {
            printk("%s.\n",env->envp[i]);
            i ++;
        }

        return 0;
    }

    static struct kset_uevent_ops uevent_ops =
    {
        .filter = kset_filter,
        .name = kset_name,
        .uevent = kset_uevent,
    };


    static int __init kset_test_init(void)
    {
        int ret = 0;

        printk("kset test init!\n");

        /* 创建并注册 kset_p */
        kset_p = kset_create_and_add("kset_p",&uevent_ops,NULL);

        /* 添加 kset_c 到 kset_p */
        kobject_set_name(&kset_c.kobj,"kset_c");
        kset_c.kobj.kset = kset_p;

        /* 对于较新版本的内核,在注册 kset 之前,需要  
             * 填充 kset.kobj 的 ktype 成员,否则注册不会成功 */
        kset_c.kobj.ktype = &ktype;
        ret = kset_register(&kset_c);

        if(ret)
            kset_unregister(kset_p);

        return 0;
    }

    static void __exit kset_test_exit(void)
    {
        printk("kset test exit!\n");
        kset_unregister(kset_p);
        kset_unregister(&kset_c);

    }

    module_init(kset_test_init);
    module_exit(kset_test_exit);





  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
LINUX设备驱动第三版_ 前言 第一章 设备驱动程序简介 设备驱动程序的作用 内核功能划分 设备和模块的分类 安全问题 版本编号 许可证条款 加入内核开发社团 本书概要 第二章 构造和运行模块 设置测试系统 Hello World模块 核心模块与应用程序的对比 编译和装载 内核符号表 预备知识 初始化和关闭 模块参数 在用户空间编写驱动程序 快速参考 第三章 字符设备驱动程序 scull的设计 主设备号和次设备号 一些重要的数据结构 字符设备的注册 open和release scull的内存使用 read和write 试试新设备 快速参考 第四章 调试技术 内核中的调试支持 通过打印调试 通过查询调试 通过监视调试 调试系统故障 调试器和相关工具 第五章 并发和竞态 scull的缺陷 并发及其管理 信号量和互斥体 completion 自旋锁 锁陷阱 除了锁之外的办法 快速参考 第六章 高级字符驱动程序操作 ioctl 阻塞型I/O poll和select 异步通知 定位设备 设备文件的访问控制 快速参考 第七章 时间、延迟及延缓操作 度量时间差 获取当前时间 延迟执行 内核定时器 tasklet 工作队列 快速参考 第八章 分配内存 kmalloc函数的内幕 后备高速缓存 get_free_page和相关函数 vmalloc及其辅助函数 per-CPU变量 获取大的缓冲区 快速参考 第九章 与硬件通信 I/O端口和I/O内存 使用I/O端口 I/O端口示例 使用I/O内存 快速参考 第十章 中断处理 准备并口 安装中断处理例程 实现中断处理例程 顶半部和底半部 中断共享 中断驱动的I/O 快速参考 第十一章 内核的数据类型 使用标准C语言类型 为数据项分配确定的空间大小 接口特定的类型 其他有关移植性的问题 链表 快速参考 第十二章 PCI驱动程序 PCI接口 ISA回顾 PC/104和PC/104+ 其他的PC总线 SBus NuBus 外部总线 快速参考 第十三章 USB驱动程序 USB设备基础 USB和Sysfs USB urb 编写USB驱动程序 不使用urb的USB传输 快速参考 第十四章 Linux设备模型 kobject、kset和子系统 低层sysfs操作 热插拔事件的产生 总线、设备驱动程序 类 各环节的整合 热插拔 处理固件 快速索引 第十五章 内存映射和DMA Linux的内存管理 mmap设备操作 执行直接I/O访问 直接内存访问 快速参考 第十六章 块设备驱动程序 注册 块设备操作 请求处理 其他一些细节 快速参考 第十七章 网络驱动程序 snull设计 连接到内核 net_device结构细节 打开和关闭 数据包传输 数据包的接收 中断处理例程 不使用接收中断 链路状态的改变 套接字缓冲区 MAC 地址解析 定制 ioctl 命令 统计信息 组播 其他知识点详解 快速参考 第十八章 TTY驱动程序 小型TTY驱动程序 tty_driver函数指针 TTY线路设置 ioctls proc和sysfs对TTY设备的处理 tty_driver结构详解 tty_operations结构详解 tty_struct结构详解 快速参考 参考书目 9112405-1_o.jpg (85.53 KB, 下载次数: 50)

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值