一、在Linux2.6内核中一个字符设备用cdev结构来描述,其定义如下:

 struct cdev {
struct kobject kobj;
struct module *owner; //所属模块
const struct file_operations *ops; //文件操作结构,在写驱动时,其结构体内的大部分函数要被实现
struct list_head list;
dev_t dev; //设备号,int 类型,高12位为主设备号,低20位为次设备号
unsigned int count;
};

可以使用如下宏调用来获得主次设备号:
MAJOR(dev_t dev) 
MINOR(dev_t dev) 
MKDEV(int major,int minor) //通过主次设备号来生成dev_t

以上宏调用在内核源码中如此定义:

 #define MINORBITS       20
#define MINORMASK ((1U << MINORBITS) - 1) //(1<<20 -1) 此操作后,MINORMASK宏的低20位为1,高12位为0
#define MAJOR(dev) ((unsigned int) ((dev) >> MINORBITS))
#define MINOR(dev) ((unsigned int) ((dev) & MINORMASK))
#define MKDEV(ma,mi) (((ma) << MINORBITS) | (mi))

二、下面一组函数用来对cdev结构体进行操作:

 void cdev_init(struct cdev *, const struct file_operations *);//初始化,建立cdev和file_operation 之间的连接
struct cdev *cdev_alloc(void); //动态申请一个cdev内存
void cdev_put(struct cdev *p); //释放
int cdev_add(struct cdev *, dev_t, unsigned); //注册设备,通常发生在驱动模块的加载函数中
void cdev_del(struct cdev *);//注销设备,通常发生在驱动模块的卸载函数中

1)如果希望在运行时动态的获得一个独立的 cdev 结构,有两种方法可以分配并初始化 cedv 结构。

 struct cdev *my_cdev = cdev_alloc();
my_cdev->ops = &my_fops; struct cdev *cdev_alloc(void)
{
struct cdev *p = kzalloc(sizeof(struct cdev), GFP_KERNEL);
if (p) {
INIT_LIST_HEAD(&p->list);
kobject_init(&p->kobj, &ktype_cdev_dynamic);
}
return p;
}

2)有时可能希望就把 cdev 结构内嵌在自己的特定设备结构里,那么在分配好 cdev 结构后,就用 cdev_init() 函数对其初始化:

 void cdev_init(struct cdev *cdev, const struct file_operations *fops)
{
memset(cdev, , sizeof *cdev);
INIT_LIST_HEAD(&cdev->list);
kobject_init(&cdev->kobj, &ktype_cdev_default);
cdev->ops = fops;
}

3)注册与注销:像cdev 中的owner 要设置为 THIS_MOULE ,在注册时应该先调用:
int register_chrdev_region(dev_t from,unsigned count,const char *name)函数为其分配设备号,此函数可用:

int alloc_chrdev_region(dev_t *dev,unsigned baseminor,unsigned count,const char *name)函数代替,他们之间的区别在于:register_chrdev_region()用于已知设备号时,另一个用于动态申请,其优点在于不会造成设备号重复的冲突。 
在注销之后,应调用:void unregister_chrdev_region(dev_t from,unsigned count)函数释放原先申请的设备号。 
他们之间的顺序关系如下: 
register_chrdev_region()-->cdev_add()     //此过程在加载模块中 
cdev_del()-->unregister_chrdev_region()     //此过程在卸载模块中

参考原文:http://bbs.ednchina.com/BLOG_ARTICLE_1897586.HTM

最新文章

  1. iOS—如何申请苹果公司开发者账号流程详细图文介绍(包括邓白氏编码的申请方法详细介绍)
  2. javascript_获取浏览器属性
  3. JavaScript访问修改css样式表
  4. 64位python安装MySQL-python 1.2.5
  5. Android 动态改变布局属性RelativeLayout.LayoutParams.addRule()
  6. Contoso 大学 - 7 – 处理并发
  7. python27读书笔记0.1
  8. vmware产品
  9. AFNnetworking快速教程,官方入门教程译
  10. javaWeb RSA加密使用
  11. TOGAF企业连续体和工具之架构资源库及架构工具的选择
  12. iOS中定时器NSTimer的使用/开启与关闭
  13. javascript 函数的基础知识
  14. Visual Studio2017中如何让Entity Framework工具【ADO.NET实体数据模型】支持MYSQL数据源
  15. swift3.0 coreData的使用-日记本demo
  16. spring mvc 中自定义404页面在IE中无法显示favicon.ico问题的解决方法。
  17. CCF CSP 201503-1 图像旋转 (降维)
  18. Android滑动冲突解决
  19. 【XSY1081】随机存储器 网络流
  20. linux中按照指定内容查找文件

热门文章

  1. AtCoder Regular Contest 092 C D E F
  2. linux基础编程 套接字socket 完整的服务器端多线程socket程序【转】
  3. iOS推送原理和证书生成简介
  4. Linux内核态抢占机制分析【转】
  5. python Nosql-redis 连接、管道
  6. flask+gunicorn中文文件下载报错问题及解决
  7. 【C++】默认构造函数
  8. ZOJ-3822
  9. Jquery学习之路(三) 实现弹出层插件
  10. 只用120行Java代码写一个自己的区块链-4实现真正的p2p网络