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");

  • 4
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值