当前位置: 首页 > news >正文

Linux2.6设备驱动开发

一:Linux2.6驱动设备开发的特点

1:首先是属于字符型设备注册的方法之一

        这种开发接口是在Linux2.6引入的,之前的版本不支持这种开发方式,也是目前最标准的开发方式。

2:Linux2.6的设备开发

        不再去限制设备号,与其对应的就是需要去申请可用的设备号。

3:Linux2.6注册完毕设备且成功之后并不会生成设备文件

对于这种情况有两种方法:

        通过指令手动生成

        通过内核其他接口函数生成

4:Linux2.6与杂项的区别

杂项:

        优点:简单

        缺点:没有分类。能注册的设备有限

Linux2.6:

        有点:主设备号不限制。2^12个主设备号,设备分类;2^20个次设备号

        缺点:函数流程相对于杂项复杂

总结:Linux2.6相对于杂项而言复杂,Linux2.6设备号范围比杂项大。

二:Linux2.6 驱动开发架构

1、先去申请设备号->alloc_chrdev_region();

2、初始化Linux2.6核心结构体 cdev->cdev_init();

3、添加注册设备->cdev_add

4、生成一个与之相对应的设备文件:class_create device_create

总结:在使用Linux2.6驱动时的开发步骤,需要先申请设备号,然后进行初始化,然后添加注册的设备,最后生成设备文件。

三:linux2.6驱动开发接口

现在进入到具体的开发流程

1:如何在内核中申请一个可以用的设备号

头文件:

#include "linux/fs.h"

对应的接口函数:

int alloc_chrdev_region(dev_t *dev, //uint32_t类型数字,也就是申请的设备号unsigned baseminor,//申请的起始次设备号unsigned count,//连续申请的设备数量const char *name)//任意填写名字

函数功能:向内核申请目前可以用的设备号,可以申请多个设备号

函数返回值:

成功返回0,失败返回非0

释放设备号接口:

void unregister_chrdev_region(dev_t from, //填写的申请的首设备号unsigned count//释放几个信号
)

2:linux2.6的设备注册

头文件:

#include "linux/cdev.h"

初始化函数:

void cdev_init(struct cdev *cdev, //要初始化的cdev核心结构体const struct file_operations *fops//与cdev做绑定,注册的设备都会用此内核接口
)

动态开辟空间的方法:kzalloc();

添加函数:

int cdev_add(struct cdev *p, //cdev核心结构体dev_t dev, //要注册的设备的首设备号unsigned count//要注册的设备的数量)

删除函数:

void cdev_del(struct cdev *p//核心结构体
)

3:设备文件的生成

手动生成:

mknod /dev/led c 234(主设备号) 0(次设备号)

自动生成:

内核接口:

想要生成设备文件必须先创建类结构体->class

struct class * cls = class_create(THIS_MODULE,"led_class");

销毁一个类:

class_destroy(struct class * cls)
有了类结构体你就可以创建设备文件了struct device *device_create(struct class *class,struct device *parent,dev_t devt,void *drvdata,const char *fmt, ...
) *
class:刚才得到的类结构体
*parent:设备的父设备->没有父设备->NULL
*devt设备的设备号
*drvdata创建设备附带的私有数据->NULL
*fmt,...:跟 printf 一样的
直接当作字符串传递也是一样的
他就是你创建设备文件的名字!

如何销毁一个设备文件:

void device_destroy(struct class *class, dev_t devt)
class:类结构体
devt:设备号

4:事例:linux2.6下的LED驱动

#include"linux/fs.h"
#include"linux/module.h"
#include"linux/kernel.h"
#include"linux/cdev.h"
#include"linux/gpio.h"
#include"device.h"dev_t  mydevnum;//设备号
struct cdev mycdev;//Cdev核心结构体
struct file_operations ops;//内核层的文件操集合结构体
struct class * cls;//类设备文件结构体
//开灯回调函数int led_on (struct inode *i, struct file *f)
{gpio_set_value(21,1);gpio_set_value(22,0);return 0;
}
//关灯回调函数
int led_off (struct inode *i, struct file *f)
{gpio_set_value(21,0);gpio_set_value(22,1);return 0;
}
//入口函数
static int __init myled_init(void)
{//申请设备号int ret = alloc_chrdev_region(&mydevnum,0,1,"myled");if(ret < 0){return -EINVAL;//加载失败}printk("主设备:%d\t\t次设备:%d\r\n",mydevnum>>20,mydevnum&0xFFFFF);//初始化Linux2.6 cdev结构体ops.owner = THIS_MODULE;ops.open = led_on;ops.release =led_off;cdev_init(&mycdev,&ops);//添加cdev设备cdev_add(&mycdev,mydevnum,1);//生成一个类设备文件cls=class_create(THIS_MODULE, "led_class");//创建设备文件device_create(cls,NULL,mydevnum,NULL,"led");//初始化硬件gpio_request(21,"led1");gpio_request(22,"led2");gpio_direction_output(21,0);gpio_direction_output(22,0);return 0;
}
//出口函数
static void __exit myled_exit(void)
{//卸载程序只有一个原则 ->倒序gpio_free(21);gpio_free(22);device_destroy(cls,mydevnum);class_destroy(cls);cdev_del(&mycdev);unregister_chrdev_region(mydevnum,1);
}
module_init(myled_init);
module_exit(myled_exit);
MODULE_LICENSE("GPL");


http://www.mrgr.cn/news/3025.html

相关文章:

  • Jmeter请求发送加密参数详解
  • Renumber程序——后面的想法没有实现
  • linux使用nginx部署springboot + vue分离项目
  • 【微信小程序】基础连接微信云数据库
  • 音视频相关知识
  • Nginx优化与防盗链
  • NoClassDefFoundError
  • 《区块链:能源市场变革的引领者》
  • 盒马鲜生源码开发
  • 杭州百腾教育科技 TiDB 6.5 to 7.5 升级记录
  • Day44 | 图论理论基础 98. 所有可达路径
  • Mycat分片-垂直拆分
  • 雨云美国二区E5v2服务器测评(非广告)
  • MySQL中处理JSON数据一文即可入门
  • 【图解秒杀系列】秒杀技术点——秒杀按钮点亮、削峰
  • 【无人驾驶】坐标变换和点云配准
  • Hadoop之DataNode启动源码解析
  • vue3 组合式 API:setup()
  • Java语言程序设计——篇十五(3)
  • CSS的:current伪类:精准定位当前活动元素