从零开始写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.h>

dev_t dev_no ;
static int __init  cdev_test_init(void)
{
	int ret ;
	printk("HELLO KERNEL FOR CDEV!\n");
	//1、创建设备号-->第一个是主设备号,第二个是次设备号
	//dev_no  = MKDEV(222,2);
	//2、注册设备号
	//count表示要分配多少个设备号
	//ret = register_chrdev_region(dev_no,1,"my_dev");
	//申请设备号
	ret = alloc_chrdev_region(&dev_no,1,1,"my_dev");

	if(ret < 0){
		goto register_error ;
	}

	register_error:
	return 0 ;
}

static int __exit cdev_test_exit(void)
{
	//注销驱动-->后面写1表示从dev_no开始连续一个
	unregister_chrdev_region(dev_no,1);
	return 0 ;
}

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

然后重新编译,将内核镜像下载到开发板:
cat /proc/devices查看

我们看到了,my_dev对应的主设备号是248了,上一个驱动是222,为什么这里就是248了,而不是222了呢?

因为,在这里,我们调用了#include <linux/fs.h>这个头文件下的这个函数:

extern int alloc_chrdev_region(dev_t *, unsigned, unsigned, const char *);

这个函数的作用就是,由内核给我们分配一个设备号,这个设备号是内核自动分配的,就不需要我们去使用MKDEV这个宏来进行手动分配了。这也可以称作是字符设备的动态分配方式。

函数原型如下:

int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count,
			const char *name)
{
	struct char_device_struct *cd;
	//调用__register_chrdev_region来注册字符设备
	cd = __register_chrdev_region(0, baseminor, count, name);
	//注册失败返回PTR_ERR(cd)错误码。
	if (IS_ERR(cd))
		return PTR_ERR(cd);
	//这里一样的是调用MKDEV分配设备号
	*dev = MKDEV(cd->major, cd->baseminor);
	return 0;
}
时间: 2024-09-19 23:18:56

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

从零开始写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 <li

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

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

手把手教你从零实现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设备驱动---中断(一)(基于友善之臂4412开发板)

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

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

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

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

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

手把手教你写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 这节,我们来学习一下触摸屏事件获取,然后上一个基于4412开发板ft5x0x型号的x,y坐标值读取,后面我们将从零开始实现这款触摸屏的驱动程序: 首先,我们要明白一个概念,触摸屏在input系统中是一类什么事件

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

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