从零开始写linux字符设备驱动程序(三)(基于友善之臂tiny4412开发板)

这一节,我们再来看看新的知识点,这一次,我们将进一步完善这个字符设备的驱动程序。

首先,将上一节的代码做下修改:

#include <linux/init.h>
#include <linux/module.h>
#include <linux/sched.h>
#include <linux/kernel.h>
#include <linux/cdev.h>
#include <linux/fs.h>
#include <linux/slab.h>

//创建一个字符设备
struct char_dev
{
    struct cdev c_dev ;
    dev_t dev_num ;
    char buf[1024];
};

int my_open()
{
    printk("cdev open");
}

int my_close()
{
    printk("cdev del");
}

struct file_operations my_ops = {
	.open = my_open,
	.release = my_close ,
};

struct char_dev *test_dev ;
static int __init  cdev_test_init(void)
{
	int ret ;
	//1、给字符设备结构分配内存
	test_dev = kmalloc(sizeof(*test_dev),GFP_KERNEL);
	if(!test_dev){
	   ret = -ENOMEM ;
	   goto malloc_dev_fair;
	}
	//2、申请设备号并注册字符设备
	ret = alloc_chrdev_region(&test_dev->dev_num,1,1,"test_dev");
	if(ret < 0){
	   goto alloc_chrdev_fair ;
	}
	//3、初始化字符设备
	cdev_init(&test_dev->dev_num , &my_ops);
	//4、添加一个字符设备
	ret = cdev_add(&test_dev->c_dev,test_dev->dev_num,1);
	if(ret < 0){
	   goto cdev_add_fair;
	}
	my_open();
	return 0 ;
	cdev_add_fair:
	return ret ;
	malloc_dev_fair :
	return ret  ;
	alloc_chrdev_fair :
	return ret ;
}

static int __exit cdev_test_exit(void)
{
	//删除设备
	cdev_del(&test_dev->c_dev);
	//注销驱动-->后面写1表示从dev_no开始连续一个
	unregister_chrdev_region(test_dev->dev_num,1);
	return 0 ;
}

module_init(cdev_test_init);
module_exit(cdev_test_exit);
MODULE_LICENSE("GPL");

在代码中,我们要实现一个虚拟的字符设备,这个设备很简单,只不过更加丰富了。
我们首先创建一个字符设备,用一个结构体char_dev来表示。

对结构体分配内存,然后申请设备号并注册,最后初始化,再将这个字符设备加到内核里去,一旦这些操作成功后,将调用my_open函数。

这就是一个字符设备的最基本构成。

上节我们已经说过alloc_chrdev_region这个函数的作用。

那么这节多了file_operations这个结构体,它的功能是什么?

当一个字符设备被注册后,我们随即就要来操作这个字符设备,open  , read , write , close等操作。

如下代码:

struct file_operations {
	struct module *owner;
	loff_t (*llseek) (struct file *, loff_t, int);
	ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
	ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
	ssize_t (*aio_read) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
	ssize_t (*aio_write) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
	int (*readdir) (struct file *, void *, filldir_t);
	unsigned int (*poll) (struct file *, struct poll_table_struct *);
	long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
	long (*compat_ioctl) (struct file *, unsigned int, unsigned long);
	int (*mmap) (struct file *, struct vm_area_struct *);
	int (*open) (struct inode *, struct file *);
	int (*flush) (struct file *, fl_owner_t id);
	int (*release) (struct inode *, struct file *);
	int (*fsync) (struct file *, loff_t, loff_t, int datasync);
	int (*aio_fsync) (struct kiocb *, int datasync);
	int (*fasync) (int, struct file *, int);
	int (*lock) (struct file *, int, struct file_lock *);
	ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
	unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
	int (*check_flags)(int);
	int (*flock) (struct file *, int, struct file_lock *);
	ssize_t (*splice_write)(struct pipe_inode_info *, struct file *, loff_t *, size_t, unsigned int);
	ssize_t (*splice_read)(struct file *, loff_t *, struct pipe_inode_info *, size_t, unsigned int);
	int (*setlease)(struct file *, long, struct file_lock **);
	long (*fallocate)(struct file *file, int mode, loff_t offset,
			  loff_t len);
};

那么内核是如何去识别相应的函数呢?
是通过系统调用

在上层应用程序,打个比方。

通过open()打印相应的设备,那么syscall函数就会通过系统调用号识别到内核态里的函数,进而调用到我们这里实现的my_open,这就是内核态和用户态相互沟通的方式。

这里我就不去写相应的应用程序了,以前也写过了,我就直接将open函数调用放在init函数,随着字符设备注册并执行。

这样将zImage下载到开发板上,串口上也是可以打印cdev_open的。

不知道怎么用应用程序去读写设备的可以参考以下文章:

http://blog.csdn.net/morixinguan/article/details/50619675

接下来看看本节使用的函数:

void cdev_init(struct cdev *, const struct file_operations *);

int cdev_add(struct cdev *, dev_t, unsigned);
void cdev_del(struct cdev *);

static __always_inline void *kmalloc(size_t size, gfp_t flags);

留心的小伙伴会发现,在exit函数中,我没有对内存进行释放,这里是故意这么做的,为了提醒粗心的伙伴,在内核中,分配的内存一定要释放的。

释放调用函数:

void kfree(const void *objp)

时间: 2025-01-25 04:22:35

从零开始写linux字符设备驱动程序(三)(基于友善之臂tiny4412开发板)的相关文章

从零开始写linux字符设备驱动程序(二)(基于友善之臂tiny4412开发板)

上节,我们讲解了如何写第一个linux字符设备驱动程序,这节,我们将代码做一下修改. 如下: #include <linux/init.h> #include <linux/module.h> #include <linux/sched.h> #include <linux/kernel.h> #include <linux/cdev.h> #include <linux/kdev_t.h> #include <linux/fs

手把手教你从零实现Linux misc设备驱动一(基于友善之臂4412开发板)

关于如何来写一个misc设备,在前面有篇文章已经介绍了大致的流程,现在就让我们来实现一个最简单的misc设备驱动. http://blog.csdn.net/morixinguan/article/details/52700146 关于前面的字符设备有以下四篇文章,可以做参考: http://blog.csdn.net/morixinguan/article/details/55002774 http://blog.csdn.net/morixinguan/article/details/550

手把手教你从零实现Linux misc设备驱动二(基于友善之臂4412开发板)

上一节,我教大家实现了一个最简单的MISC设备驱动,那么这节,我们将用一个实例来驱动蜂鸣器,这里为了方便,我就不再写应用程序进行测试,直接在驱动里调用open函数,这个程序是在Android系统里跑起来,后面我会教大家如何在Android下写应用测试程序. 我们参考以前写的蜂鸣器驱动程序,将它移植到我们这个程序里,让它成为一个MISC设备. 参考以前写的文章: http://blog.csdn.net/morixinguan/article/details/50628588 接下来,看看代码:

手把手教你写Linux设备驱动---中断(一)(基于友善之臂4412开发板)

今天,我们要来实现一个基于tiny4412开发板上的最简本的按键中断驱动程序,那么,写这个程序之前,我们先来了解下Linux中断的基本知识. 在Linux内核中,每一个能够发出中断请求的硬件设备控制器都有一条名为IRQ的输出线.所有现在存在的IRQ线都与一个名为可编程中断控制器的硬件电路的输入引脚相连,我们可以来看下4412上与板子上相连的按键. 下面这张电路图,也就是4412板子上按键的电路图和CPU的连接关系图: 我们明显可以看到,4个按键分别接在GPX3这几个引脚上,对应着引脚,接下来我们

手把手教你写Linux设备驱动---input子系统(三)--电容屏事件坐标读取(基于友善之臂4412开发板)

前面我们学习了鼠标是如何如何通过应用程序来读取事件和坐标值的,后面也写了一个简单的input系统的按键驱动程序. 博文如下,讲的内容非常清楚,给小白来入手当然是非常容易的: http://blog.csdn.net/morixinguan/article/details/69808832 这节,我们来学习一下触摸屏事件获取,然后上一个基于4412开发板ft5x0x型号的x,y坐标值读取,后面我们将从零开始实现这款触摸屏的驱动程序: 首先,我们要明白一个概念,触摸屏在input系统中是一类什么事件

手把手教你写Linux设备驱动---input子系统(四)--电容屏驱动ft5x06编写(一)(基于友善之臂4412开发板)

这一节,我们将从零开始写tiny4412的触摸屏驱动ft5x06,写这节博客之前,先了解下需要什么知识: 1.i2c驱动相关的知识 2.输入子系统 3.中断 4.工作队列 关于i2c驱动相关的知识,在后期的博文里会专门写几篇博文来进行总结,这里就不再说i2c相关的知识,我们先知道怎么用就行了. 首先,我在ts.h构造了一个ts_info_st结构体,用来存放触摸屏的中断线,x坐标,y坐标,压力值. 用ts_st构造了该触摸屏的设备结构体. 我们还是直接看点实际的东西,上代码: ts.h #ifn

手把手教你写Linux设备驱动---input子系统(二)--按键驱动实现(一)(基于友善之臂4412开发板)

在上一节里,我们用一个应用程序实现了鼠标的控制,并控制鼠标用相对位移不断的画一个正方形,感觉非常有意思,这一节,我们将通过一个简单按键实例来真正的实现一个input设备驱动程序. http://blog.csdn.net/morixinguan/article/details/69808832 在写Input驱动之前,我们要了解下这个结构体,在此,我们要包含相应的头文件: #include <linux/input.h> 我们在这个头文件中找到了以下结构体,它就是input设备的核心: //用

手把手教你写Linux设备驱动---定时器(一)(基于友善之臂4412开发板)

这个专题我们来说下Linux中的定时器. 在Linux内核中,有这样的一个定时器,叫做内核定时器,内核定时器用于控制某个函数,也就是定时器将要处理的函数在未来的某个特定的时间内执行.内核定时器注册的处理函数只执行一次,即不是循环执行的. 如果对延迟的精度要求不高的话,最简单的实现方法如下---忙等待: Unsigned long j = jiffies + jit_delay * HZ; While(jiffies < j) { -- } 下面来说下具体的参数代表的含义: jiffies:全局变

手把手教你写Linux设备驱动---中断(二)--tasklet实现(基于友善之臂4412开发板)

上节:http://blog.csdn.net/morixinguan/article/details/68958185 在上一节博文中,教会了大家如何来写一个Linux设备的中断程序,实现也非常简单,我们来回顾一下具体的操作流程,只要遵循以下几个步骤即可实现最简单的中断处理程序: 使用中断相关的API和定义时要包含以下头文件: #include <linux/interrupt.h> 然后写中断需要以下步骤 1.申请中断号 使用gpio_to_irq函数,可以从返回值获取到对应的中断号 2.