树莓派基于Linux内核驱动开发详解

2023-11-08

目录

一、驱动认知

1.1 为什么要学习写驱动

树莓派开发简单是因为有厂家提供的wiringPi库,实现超声波,实现继电器操作,做灯的点亮…都非常简单。

但未来做开发时,不一定都是用树莓派,则没有wiringPi库可以用。但只要能运行Linux,linux的标准C库一定有。

学会根据标准C库编写驱动,只要能拿到linux内核源码,拿到芯片手册,电路图…就能做开发。

用树莓派学习的目的不仅是为是体验其强大便捷的wiringPi库,更要通过树莓派学会linux内核开发,驱动编写等,做一个属于自己的库。

1.2 文件名与设备号

linux一切皆为文件,其设备管理同样是和文件系统紧密结合。在目录/dev下都能看到鼠标,键盘,屏幕,串口等设备文件,硬件要有相对应的驱动,那么open怎样区分这些硬件呢?

依靠文件名与设备号。在/devls -l可以看到

在这里插入图片描述

设备号又分为:主设备号用于区别不同种类的设备;次设备号区别同种类型的多个设备

内核中存在一个驱动链表,管理所有设备的驱动。 驱动开发无非以下两件事:

  • 编写完驱动程序,加载到内核
  • 用户空间open后,调用驱动程序(驱动程序就是操作寄存器来驱动IO口,单片机51,32就是这种操作)

驱动插入到链表的位置(顺序)由设备号检索。

1.3 open函数打通上层到底层硬件的详细过程

  • 用户空间调用open(比如open(“/dev/pin4”,O_RDWR))产生一个软中断(中断号是0x80),进入内核空间调用sys_call,这个sys_call在内核里面是汇编的,用Source Insight搜索不到。

  • sys_calll真正调用的是sys_open(属于VFS层虚拟文件系统,因为磁盘的分区和引脚分区不一样,为了实现上层统一化),根据你的设备名比如pin4去到内核的驱动链表,根据其主设备号与次设备号找到相关驱动函数

  • 调用驱动函数里面的open,这个open就是对寄存器的操作,从而设置IO口引脚电平。这件事对于单片机来说特变容易,就两句话搞定:

sbit pin4 = P1^4;
pin4=1;

(对应下图的粉色笔迹)

在这里插入图片描述

二、基于框架编写驱动代码

2.1 编写上层应用代码

目的是用简单的例子展示从用户空间到内核空间的整套流程

在上层访问一个设备跟访问普通的文件没什么区别。试写一个简单的open和write去操作设备"pin4"。
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>

int main()
{
	int fd;
	fd = open("/dev/pin4",O_RDWR);
	if(fd < 0){
		printf("open failed\n");
		perror("reson");
	}else{
		printf("open success\n");
	}
	fd = write(fd,'1',1);//写一个字符'1',写一个字节
	return 0;
}

根据上面提到的驱动认知,有个大致的概念,以open为例子:
上层opensys_callsys_open→内核驱动链表节点→执行节点里的open

当然,没有装载驱动的话这个程序执行一定会报错。只有在内核装载了驱动并且在/dev下生成了“pin4”这样一个设备才能运行。

接下来介绍最简单的字符设备驱动框架。

2.2 修改内核驱动框架代码

所谓框架,就是在往驱动链表里面加驱动的时候要符合内核规则,它是定死的东西,基本的语句必须要有,少一个都不行。

虽然有这么多的代码,但核心运行的就两个printk。

#include <linux/fs.h>		 //file_operations声明
#include <linux/module.h>    //module_init  module_exit声明
#include <linux/init.h>      //__init  __exit 宏定义声明
#include <linux/device.h>	 //class  devise声明
#include <linux/uaccess.h>   //copy_from_user 的头文件
#include <linux/types.h>     //设备号  dev_t 类型声明
#include <asm/io.h>          //ioremap iounmap的头文件

static struct class *pin4_class;  
static struct device *pin4_class_dev;

static dev_t devno;                //设备号
static int major =231;  		   //主设备号
static int minor =0;			   //次设备号
static char *module_name="pin4";   //模块名 上层的名字

//pin4_open函数
static int pin4_open(struct inode *inode,struct file *file)
{
    printk("pin4_open\n");  //内核的打印函数,和printf类似
   
    return 0;
}

//pin4_write函数  因为上层需要open和write这两个函数 
//            如果上层需要调用read等其他函数,可用SourceInsight去内核源码搜索,照着格式修改即可使用 在file_operations结构体里面
static ssize_t pin4_write(struct file *file1,const char __user *buf,size_t count, loff_t *ppos)
{
	printk("pin4_write\n");
    return 0;
}

static struct file_operations pin4_fops = {//内核定义好的结构体 内核源码里有
                                           //就是驱动的结构体 要加载到内核驱动链表
    .owner = THIS_MODULE, 
    .open  = pin4_open,  //上层有读 底层就要有open的支持
    .write = pin4_write, //上层有写 底层就要有write的支持
};




int __init pin4_drv_init(void)   //驱动的真正入口
{

    int ret;
    devno = MKDEV(major,minor);//创建设备号 
    
   //********************注册驱动 加载到内核驱动链表***********
                       //主设备号231 模块名pin4 上面的结构体
    ret   = register_chrdev(major, module_name,&pin4_fops); //注册驱动 告诉内核,把这个驱动加入到内核驱动的链表中
    pin4_class=class_create(THIS_MODULE,"myfirstdemo");  //由代码在/dev下自动生成设备  也可以手动生成设备
    pin4_class_dev =device_create(pin4_class,NULL,devno,NULL,module_name);  //创建设备文件

 
    return 0;
}

void __exit pin4_drv_exit(void)
{

    device_destroy(pin4_class,devno);       //删除设备  /dev底下的 上面也是创建了设备和类
    class_destroy(pin4_class);              //删除类
    unregister_chrdev(major, module_name);  //卸载驱动 就是删除链表节点的驱动

}

module_init(pin4_drv_init);  //入口:内核加载驱动的时候,这个宏(module_init它不是个函数)会被调用,而真正的驱动入口是它里面调用的函数
module_exit(pin4_drv_exit);
MODULE_LICENSE("GPL v2");

2.3 部分代码解读

2.3.1 static的作用

内核代码数量庞大,为了防止与其他的文件有变量命名冲突,static限定变量的作用域仅仅只在这个文件。内核源码里面运用了大量的static,因为内核源码众多,一万五千多个C文件,很容易造成代码命名的冲突。

2.3.2 结构体成员变量赋值方式

static struct file_operations pin4_fops = {

    .owner = THIS_MODULE,
    .open  = pin4_open,
    .write = pin4_write,
};

这是内核代码中常见的对结构体的操作方式,单独给指定结构体某些元素赋值。

注意:在keil的编译工具环境中不允许这样写,linux可以。

2.3.3 结构体file_operations(最终加载到内核驱动链表)

在SourceInsight中查看结构体file_operations,可以发现很多的函数指针(指向函数的指针,函数内进行一些程序的执行),这些函数名跟系统上层对文件的操作差不多。(read,write,llseek)(在课程视频9:36)

在这里插入图片描述

如果上层想要实现read,就复制过来,按照格式改一改就能使用。

上层对应底层,上层想要用read,底层就要有read的支持

2.3.4 手动生成设备

框架中有自动生成设备的代码,那么手动生成设备是怎么样的呢?(一般不这样干,麻烦,仅作为演示)

  • 进入/dev目录,查看帮助可知道创建规则

sudo mknod 设备名称 设备类型 主设备号 次设备号
在这里插入图片描述

  • 使用如下命令创建名称为zhu,主设备号为8,次设备号为1的字符设备。
sudo mknod zhu c 8 1

 
 
 
 
  • 1
ls -l可以看到已经创建成功 在这里插入图片描述

三、驱动代码编译和测试

3.1 驱动框架的模块编译并发送至树莓派

在ubuntu中,进入Linux内核源码(前一章节编译好的)字符设备驱动目录linux-rpi-4.14.y/drivers/char(IO口属于字符设备驱动)。进入源码目录下的原因是,写驱动必须要链接到源码(源码定义好了结构体等等),必须要有源码

拷贝上文分析过的驱动框架代码,拿到这个文件夹下 ,并创建成名字为pin4drive.c的文件

①Makefile内添加生成.o命令

进行配置,使得工程编译时可以编译这个文件

vi Makefile

 
 
 
 
  • 1

当然不一定要放在/char下。但注意:放在哪个文件夹下,就修改那个文件夹的Makefile即可。

Makefile:
在这里插入图片描述

模仿这些文件的编译方式,以编译成模块的形式(还有一个方式为编译进内核)编译pin4drive.c

在Makefile里面添加:

obj-m                           += pin4drive.o

 
 
 
 
  • 1

-m就是模块的形式

如图:

在这里插入图片描述

②模块编译生成.ko文件

之前编译内核镜像的时候用的是这个命令:

在这里插入图片描述现在只需进行模块编译,不需要生成zImage,dtbs文件;

  • 回到源码目录/linux-rpi-4.14.y再执行下面指令
ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make modules

注:如果说编译中途提示出错,照着错误提示去修改.c文件即可,和上层编译类似。

编译完成生成的一些文件如下:
在这里插入图片描述

③把.ko文件发送至树莓派

scp pin4drive.ko pi@192.168.101.19:/home/pi

之前犯的一个小错误是树莓派和ubuntu的ip地址一样,导致连接不上,修改树莓派的ip地址即可

3.2 上层代码交叉编译发送至树莓派

拷贝上文分析的上层代码到ubuntu中,此处我命名为pin4drivertest.c

使用交叉编译工具进行编译

arm-linux-gnueabihf-gcc pin4drivertest.c -o pin4test

发送至树莓派

scp pin4test pi@192.168.101.19:/home/pi

3.3 树莓派装载驱动并运行

①树莓派加载内核驱动(insmod)

sudo insmod pin4drive.ko
  • 查看是否已经成功添加驱动

可以去设备下查看

ls /dev/pin4 -l

看到驱动添加成功,主设备号231,次设备号0,和内核里面的代码对应上。
在这里插入图片描述

或者lsmod查看内核挂载的驱动
在这里插入图片描述

如果需要卸载驱动,就sudo rmmod pin4drive

②运行上层代码(无权限)

./pin4test

发现没有对设备pin4的访问权限
在这里插入图片描述

crw是超级用户所拥有的权限,而框中两类用户则无读写的权限(下面有详细说明)

在这里插入图片描述

③增加访问权限再运行

解决方法1:加超级用户

sudo ./pin4test

 
 
 
 
  • 1

解决方法2:增加“所有用户都可以访问的权限”(建议)

sudo chmod 666 /dev/pin4

 
 
 
 
  • 1

运行成功:
在这里插入图片描述

拓展 >> chmod 命令用于更改文件/文件夹的属性(读,写,执行)

permission to:  user(u)   group(g)   other(o)     
                /¯¯¯\      /¯¯¯\      /¯¯¯\
octal:            6          6          6
binary:         1 1 0      1 1 0      1 1 0
what to permit: r w x      r w x      r w x

what to permit - r: read, w: write, x: execute

permission to  - user: the owner that create the file/folder
                 group: the users from group that owner is member
                 other: all other users

EG: chmod 744 仅允许用户(所有者)执行所有操作,而组和其他用户只允许读。

④检查是否执行成功:demsg指令查看内核打印信息

用dmesg命令显示内核缓冲区信息,并通过管道筛选与pin4相关信息

dmesg | grep pin4

 
 
 
 
  • 1

可以看到这两个打印信息,说明内核的printk已经被成功调用,我们已经成功完成了上层对内核的调用 !

在这里插入图片描述

四、三种地址介绍

写驱动是为了操作IO口,实现自己的wiringpi库,跟硬件打交道。

首先要理解以下3个地址的概念:

4.1 总线地址

通俗来说:cpu能够访问的内存范围

现象:电脑装了32位(bit)的系统,明明内存条有8G,却只能识别3.8G左右,这是因为32位仅能表示/访问2^32^=4,294,967,296bit=4,194,304Kb=4096Mb=4G左右。只有装了64位的,才能够识别到8G。32位、64位是计算机CPU一次处理数据能力的大小。

树莓派装载32位操作系统,寻址自然是4G。

树莓派的内存

cat /proc/meminfo

 
 
 
 
  • 1

大概是927M
在这里插入图片描述

4.2 物理地址

硬件实际地址或绝对地址,就是硬盘上的排列地址

4.3 虚拟地址

又叫逻辑地址(基于算法的地址,软件层面的地址,是假地址)便称为虚拟地址

虚拟地址的作用:

以树莓派为例,总线可以访问4G,物理地址只有1G,但需要运行的程序大于1G,如果把程序全部都加载到内存是不可取的。

物理地址数据的运行真正是拿虚拟地址来操作的,虚拟地址可以比1G大,总线地址(CPU能访问的地址范围)能看到4个G,就可以把1个G的物理地址映射成4个G的虚拟地址。当物理地址不能满足程序运行空间需求时,如果没有虚拟地址,程序就不能正常运行。单片机51和STM32如果程序过大,是禁止你烧写的,而在Linux系统环境下是可以的。

树莓派3b的cpu型号是BCM2835,它是ARM-cotexA53架构

cat /proc/cpuinfo

 
 
 
 
  • 1
在这里插入图片描述

4.4 MMU内存管理单元

  • 地址框图

可以看到总线地址为FF FF FF FF,即为4G;

在这里插入图片描述

  • 内核的页表映射

在这里插入图片描述
物理地址的1M通过映射成为4M的虚拟地址(我们写的所有的代码都是在操控虚拟地址,都是假的),这中间有个设计的算法叫页表。

这个表决定了这个4M被映射到虚拟内存的哪一个段,通过MMU进行管理。单片机和ARM处理器的区别就是ARM有MMU(内存管理单元)和CACHE(高速缓存),如下图所示:

在这里插入图片描述

如果想要更多地了解Linux对内存的管理,推荐书《unix设计与实现》,类似内核设计文档,讲的是内核设计的思路,讲代码讲的不多。

五、IO口驱动编程实战

之前驱动框架的代码仅仅用来检测、走一遍整个驱动的架构。那么在这一个小节,将着手实现pin4引脚输出高 / 低电平。

驱动开发两大利器:芯片手册和电路图(电路图主要用来寻找寄存器,树莓派的芯片手册清楚地给出了各个寄存器,所以电路图很难找到)。

5.1 芯片手册导读

5.1.1 General Purpose I/O (GPIO)板块

查看芯片手册的目的性很强:做哪一块的开发,就只看那一块,现在要开发的是GPIO,熟悉控制IO口的寄存器最为重要。

如果看完这部分的文档,你对于以下几个问题(后面有解析)有清晰的答案,说明你真正读懂了这一部分的开发。

①操作逻辑:简言之就是怎么进行配置相关寄存器,这些配置步骤和思想其实都很类似。
②需要重点掌握的寄存器有哪些?例如输入 / 输出控制寄存器输出 0 / 1控制寄存器清除状态寄存器

5.1.2 Register View 导读

在新的平台也要学会捕捉类似的关键信息:选择输入还是输出,0/1,怎么清除,上升沿下降沿等。(配置过32 / 51寄存器的应该对这些很熟悉)

从下图中可以大概了解到所有的IO口被分成了0~5组。

有意思的是,下图最第一列的地址Address是树莓派总线地址,一般芯片手册给的都是真正的物理地址。第二列是寄存器的名字,第三列寄存器功能描述。

一共有41个寄存器,每个寄存器都是32位。
在这里插入图片描述

在这里插入图片描述

描述部分也很重要,大多涉及使用方法
在这里插入图片描述

5.1.3 配置引脚功能为输入/输出的寄存器

这20~29的IO口(第二列)属于分组2

在这里插入图片描述

IO编号要看好

在这里插入图片描述

5.1.4 配置引脚输出0/1的寄存器

在这里插入图片描述

5.1.5 配置引脚清除0/1状态的寄存器

在这里插入图片描述

整理关键内容

通过文档阅读,可以整理出关键的信息:

有3个最基本的要清楚:
①选择IO是 输入 / 输出控制寄存器:GPFSEL
②输出0 / 1寄存器:GPSET
③清除寄存器:GPCLR

操作逻辑:
GPFSEL0寄存器举例,引脚pin4对应的分组就是第0组(51单片机引脚也是分成第0组、第1组、第2组、第3组)。只要在这个分组下,把14-12位设置为001,就能配置pin4引脚为输出。

总之还是要自己多看多翻阅,这里仅为简单的导读。

5.2 寄存器地址配置(ioremap、volatile物理地址映射成虚拟地址)

①在原来框架的基础上,添加寄存器的定义

volatile unsigned int* GPFSEL0 = NULL;
volatile unsigned int* GPSET0  = NULL;
volatile unsigned int* GPCLR0  = NULL;

要想写出上面的代码,要掌握以下几点:

弄清楚寄存器的分组

其中寄存器的0表示的是分组,目标操作的IO是pin4,由文档可知,属于寄存器分组0。

volatile的使用

volatile作用是 : 1、防止编译器优化(你给的这个地址编译器可能认为不好,可能会省略,也可能会进行更改)这些寄存器变量;2、要求每次直接从寄存器里读值。由于随着程序的执行,会改变寄存器当中的数据,而读取的都是内存里面的备份数据,数据的时效性没有那么强,读的可能是一个老数据。在内核中对IO口进行操作都要有volatile。

②配置寄存器的地址

在①的基础上,在驱动的初始化pin4_drv_init中添加寄存器地址配置

GPFSEL0 = (volatile unsigned int *)ioremap(0x3f200000,4);
GPSET0  = (volatile unsigned int *)ioremap(0x3f20001C,4);
GPCLR0  = (volatile unsigned int *)ioremap(0x3f200028,4);

要想写出上面的代码,要掌握以下几点:

分别找到几个IO寄存器的物理地址

弄清楚GPIO的物理地址(真实地址)

并不是用下图这个地址来对应GPIO功能选择寄存器0的地址,否则编译后运行会有段错误在这里插入图片描述
IO口的起始地址是0x3f000000(从网上找的,树莓派手册第一列是总线地址),加上GPIO的偏移量0x200000,所以GPIO的实际物理地址应该是从0x3f200000开始的,然后在这个基础上进行Linux系统的MMU内存虚拟化管理,映射到虚拟地址上,编程都是操作虚拟地址。

继续顺着操作手册找到GPSET0和GPCLR0寄存器的偏移量,见下图。其实寄存器的名字是人为的根据功能命名的,本质是一串客观存在的物理地址。
在这里插入图片描述

特别注意:BCM2708 和BCM2709 IO起始地址不同,BCM2708是0x20000000,BCM2709是0x3f000000

根据偏移值,弄清楚寄存器的物理地址(真实地址)

可以看到寄存器GPSET0相对于GPIO物理地址的偏移值为1C。在这里插入图片描述

同样的方法,寄存器GPCLR0的偏移值为28。

物理地址转换为虚拟地址:ioremap函数

因为无论内核代码还是上层代码操作的都是虚拟地址,代码中直接用物理地址肯定不行,需要进行转换,将IO口寄存器映射成普通内存单元进行访问。

使用函数ioremap:

函数原型:void *ioremap(unsigned long phys_addr, unsigned long size)
phys_addr:要映射的物理地址的基地址;
size:要映射的空间的大小(一个寄存器4个字节);

5.3 寄存器功能配置

①在函数pin4_open中配置pin4为输出引脚

可以看到只要32位寄存器GPFSEL014-12位配置为001,其它位不管,即可配置pin4为输出引脚
在这里插入图片描述
当然直接暴力赋值(0000…001…0000)是不可取的,会把其他的IO口给影响。最好的结果是只改变了14-12位。

运用与(&) / 或(|)运算进行位操作

*GPFSEL0 &= ~(0x6 << 12);//110左移12位 取反 与运算
*GPFSEL0 |= (0x1 << 12); //001左移12位      或运算

想要写出以上代码,必须清楚下面两个步骤

1)、与运算给指定位(14bit、13bit)赋值0,其他不变

为了方便描述,这里把需要“与”上的数称为 “辅助数”。(寄存器中的数是假设的)

在这里插入图片描述

但为了方便(1越少,用计算器换算就越简单)得到这个第13、14位为0的数,选择对辅助数“取两次反“。

第一次取反为: 00000…110…00000

用计算器在二进制BIN中输入110(方便就在这,你要是直接在计算器中输入目标辅助数进行换算,数有多少个1都很吃力!!)

在这里插入图片描述

0110,向左移12位,低位自动补0,则1 1正好对上14、13位。

再取反(~),得到一开始想要的让寄存器的数14、13位与上0的辅助数。

2)、或运算给指定位(12bit)赋值1

是同样的思路,不再赘述

②在函数pin4_write中配置pin4输出 0 / 1(高低电平)

获取上层write函数的值:copy_from_user函数

函数介绍
unsigned long copy_from_user(void * to, const void __user * from, unsigned long n)

此函数将from指针指向的用户空间地址开始的连续n个字节的数据送到to指针指向的内核空间地址,简言之是用于将用户空间的数据传送到内核空间

第一个参数to是内核空间的数据目标地址指针,
第二个参数from是用户空间的数据源地址指针,
第三个参数n是数据长度

如果数据拷贝成功,则返回;否则,返回没有拷贝成功的数据字节数。

根据值来操作IO口

int userCmd;上层写的是整型数1,底层就要对应起来用int.如果是字符则用char

copy_from_user(&userCmd,buf,count);

if(userCmd == 1){
	printk("set 1\n");//内核调试信息
	*GPSET0 |= 0x1 << 4;
}else if(userCmd == 0){
	printk("set 0\n");
	*GPCLR0 |= 0x1 << 4;
}else{
	printk("cmd error\n");
}

说明(这也是操作逻辑的一部分啦):
①这个GPSET0,0指的是分组,不是设置成低电平。
②左移4位,是因为GPSET0寄存器的第4位对应pin4,只要把第4位设置为1,表示这个寄存器就对pin4发挥作用,设置成高电平,如果是0则 no effct(手册内容)。

5.4 解除虚拟地址映射(iounmap)

退出程序卸载驱动的时候,解除映射:iounmap函数

void iounmap(void* addr)//取消ioremap所映射的IO地址

iounmap(GPFSEL0);  //init是相反的执行顺序
iounmap(GPSET0);
iounmap(GPCLR0);

5.5 完整代码

内核驱动框架

#include <linux/fs.h>		 //file_operations声明
#include <linux/module.h>    //module_init  module_exit声明
#include <linux/init.h>      //__init  __exit 宏定义声明
#include <linux/device.h>	 //class  devise声明
#include <linux/uaccess.h>   //copy_from_user 的头文件
#include <linux/types.h>     //设备号  dev_t 类型声明
#include <asm/io.h>          //ioremap iounmap的头文件

static struct class *pin4_class;  
static struct device *pin4_class_dev;

static dev_t devno;                //设备号
static int major =231;  		   //主设备号
static int minor =0;			   //次设备号
static char *module_name="pin4";   //模块名

volatile unsigned int* GPFSEL0 = NULL;
volatile unsigned int* GPSET0  = NULL;
volatile unsigned int* GPCLR0  = NULL;

//pin4_open函数
static int pin4_open(struct inode *inode,struct file *file)
{
    printk("pin4_open\n");  //内核的打印函数,和printf类似
    //open的时候配置pin4为输出引脚
    *GPFSEL0 &= ~(0x6 << 12);
	*GPFSEL0 |= (0x1 << 12);
    
    return 0;
}

//pin4_write函数
static ssize_t pin4_write(struct file *file,const char __user *buf,size_t count, loff_t *ppos)
{
	int userCmd;//上层写的是整型数1,底层就要对应起来用int.如果是字符则用char

	printk("pin4_write\n");
	//获取上层write的值
	copy_from_user(&userCmd,buf,count);//用户空间向内核空间传输数据
	
	//根据值来执行操作
	if(userCmd == 1){
		printk("set 1\n");
		*GPSET0 |= 0x1 << 4;
	}else if(userCmd == 0){
		printk("set 0\n");
		*GPCLR0 |= 0x1 << 4;
	}else{
		printk("cmd error\n");//加入调试信息,方便通过查看内核信息进行修改
	}
	
    return 0;
}

static struct file_operations pin4_fops = {

    .owner = THIS_MODULE,
    .open  = pin4_open,
    .write = pin4_write,
};

int __init pin4_drv_init(void)   //驱动的真正入口
{

    int ret;
    printk("insmod driver pin4 success\n");
    devno = MKDEV(major,minor);  //创建设备号
    ret   = register_chrdev(major, module_name,&pin4_fops);  //注册驱动  告诉内核,把这个驱动加入到内核驱动的链表中

    pin4_class=class_create(THIS_MODULE,"myfirstdemo");  //由代码在/dev下自动生成设备
    pin4_class_dev =device_create(pin4_class,NULL,devno,NULL,module_name);  //创建设备文件

	GPFSEL0 = (volatile unsigned int *)ioremap(0x3f200000,4);
	GPSET0  = (volatile unsigned int *)ioremap(0x3f20001C,4);
	GPCLR0  = (volatile unsigned int *)ioremap(0x3f200028,4);
 	
 	return 0;
}

void __exit pin4_drv_exit(void)//可以发现和init刚好是相反的执行顺序。
{
	iounmap(GPFSEL0);
	iounmap(GPSET0);
	iounmap(GPCLR0);
	
    device_destroy(pin4_class,devno);
    class_destroy(pin4_class);
    unregister_chrdev(major, module_name);  //卸载驱动

}

module_init(pin4_drv_init);  //入口:内核加载驱动的时候,这个宏会被调用,而真正的驱动入口是它调用的函数
module_exit(pin4_drv_exit);
MODULE_LICENSE("GPL v2");

上层应用程序

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>

int main()
{
        int fd;
        int cmd;

        fd = open("/dev/pin4",O_RDWR);
        if(fd < 0){
                printf("open failed\n");
                perror("reson");
        }else{
                printf("open success\n");
        }

        printf("请输入0 / 1\n 0:设置pin4为低电平\n 1:设置pin4为高电平\n");
        scanf("%d",&cmd);

        if(cmd == 0){
                printf("pin4设置成低电平\n");
        }else if(cmd == 1){
                printf("pin4设置成高电平\n");
        }

        fd = write(fd,&cmd,1);//写一个字符'1',写一个字节
        return 0;
}

5.6 交叉编译并发送至树莓派

①树莓派上卸载之前的pin4驱动、删除树莓派上层可执行程序pin4testpin4drive.ko文件

sudo rmmod pin4drive

 
 
 
 
  • 1

用lsmod查看是否卸载成功。

基本上都会自动卸载驱动的,因为上一节框架代码最后有卸载驱动的代码操作。

②驱动框架模块化方式编译和上层应用程序在Ubuntu中进行交叉编译并发送至树莓派

上文有涉及,一样的操作不再赘述。Makefile文件之前修改过了,这里不用再次进行修改。

注意:

在Ubuntu的/char目录下因为之前的模块编译生成了.ko ,.mod等文件

没关系,直接复制新的驱动框架、新的上层代码到原来的2个.c文件覆盖保存。然后进行交叉编译,新生成的文件会覆盖掉原来的文件。

框架交叉编译后:
红色框表示编译生成了需要的模块
而蓝色框的警告可以不用理会(copy_form_user前两个参数传入的是空类型的指针,在框架代码中没有进行强制转换)
在这里插入图片描述

5.7 树莓派装载驱动

sudo insmod pin4drive.ko

 
 
 
 
  • 1

用dmesg可以看到内核打印出“驱动装载成功”(打印信息来自框架代码)

在这里插入图片描述

给权限

sudo chmod 666 /dev/pin4

或者

sudo chmod +x /dev/pin4

5.8 运行上层应用文件

./pin4test

 
 
 
 
  • 1

运行成功!
在这里插入图片描述

5.8.1 驱动运行成功

输入1时,用命令gpio readall查看pin4引脚变化,应为OUT 1

在这里插入图片描述


输入0时,再用命令gpio readall查看pin4引脚变化,应为OUT 0

dmesg打开内核打印界面,可以看到内核的printk已经被调用,配置执行。

在这里插入图片描述

这样就实现了类似WiringPi的库。

5.8.2 驱动运行失败:学会调试

当然,很多情况下不是直接就能够成功运行的,学会调试很重要。

出问题先看上层(因为上层简单,好修改),后看底层。

Mode我们配置的是输出模式,如果是IN,或者ALT2等,说明底层的模式配置出了问题,大多是寄存器的移位没搞明白。

多看看内核的打印信息,write打印信息,变量的值等等。

奇怪的问题

上层代码中,在加入红框中的printf调试信息之前

在这里插入图片描述
会出现这样的错误,内核接收不到由上层传输过来的 0 / 1,或者接收到的不是0 / 1?

在这里插入图片描述gpio readall查看,置为高低电平也失效。

但是凭什么??我加入的只是调试信息罢了,关这个什么事儿?我又没有动cmd的值。

我还以为是偶然性问题,去掉调试信息编译后重新运行,果然还是出现了cmd error

printf有毒?于是在上层随便加了一条printf
在这里插入图片描述运行,功能正常,没有出现cmd error??

啊哈,error了个啥?我很想知道

于是在内核框架代码加了一句

在这里插入图片描述
于是在error时就出现了:

在这里插入图片描述
又注意到一个可能的问题:上层代码中write写入的是一个int型,4个字节,而这里只写了一个,会不会是这个原因呢?
在这里插入图片描述于是去掉有毒的printf,把1改成4,问题解决了。

这时候我又好奇了,想看看write的返回值:
在这里插入图片描述

但发现不管是error,还是成功,写入的返回值都是0。为什么写入成功了,返回值会是0呢?不应该是写入的字节数吗?

在这里插入图片描述
本打算把命令传输由整型数换成字符,但想到还要改底层

会不会是那个用户态向内核态传输指令(copy_from_user)搞的鬼(有一个类型未转换的警告)?

对这个问题的探讨先告一段落,说不定后面会有好的例子,这个问题也就很好解决,不能花太多时间纠结于这个上面,毕竟这是我的第一个驱动程序,以后见多了,可能就理解了。

六、其他

简单了解DMA

通过阅读树莓派博通BCM2835芯片手册发现有DMA(direct memory access)(直接存储器访问)的一些介绍,多阅读这本英文芯片手册不仅能够帮助你了解芯片的一些裸机知识结构,同时也利于快速阅读其他型号的英文芯片手册。

在这里插入图片描述

大数据的快速拷贝单元。

使用cp指令拷贝大文件会很大程度占用CPU资源,DMA是专门用来做协助数据拷贝的微控制器,CPU可以发动DMA使其进行数据的拷贝。

md5sum检查两个文件是否完全一样

多用于检测原来的“同名”文件是否已经被新的替代,也可以用作检查拷贝过程中是否损坏。

md5sum file.c

 
 
 
 
  • 1

唯一标识,相同则两个文件为同一文件
在这里插入图片描述

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

树莓派基于Linux内核驱动开发详解 的相关文章

  • Snap-confine 具有提升的权限,并且不受限制,但应该受到限制。拒绝继续避免权限升级攻击

    我已经使用 snap 一段时间了 但最近升级后 当我尝试打开任何应用程序时 出现此错误 Snap confine has elevated permissions and is not confined but should be Refu
  • Mono 和 WebRequest 速度 - 测试

    在 mono 4 6 2 linux 中 我注意到 wget 下载文件的速度与webclient DownloadString 所以我做了一个小测试来调查 为什么 wget 明显比 C 快 根据我自己的实验 使用 wget 下载 手动读取文
  • Alsa 带有来自调制解调器的 PCM 接口

    我有一个基于 imx28 CPU 的定制板 CPU 的串行端口连接到调制解调器的 PCM 输出 我必须为调制解调器的 PCM 接口开发一个驱动程序 使其成为 ALSA SoC 的一部分 您能指出内核树 中与我的设置重新组合的一些驱动程序吗
  • jpackage linux 创建的桌面文件不足

    我刚刚开始使用 jpackage 它是一个非常棒的工具 只要迈出一步 我的肩上的工作就减轻了很多 我对看起来硬编码且无法定制的东西越感到惊讶 JPackage 自动生成启动器 lib
  • 对 sf:: 的未定义引用

    我想用 C 制作 GUI 应用程序 发现 SFML 是一个不错的选择 幸运的是 我使用的是 Linux 所以 SFML 2 4 已经安装在我的系统上 所以我开始搜索一些教程并找到了一个制作简单窗口的教程 但是当我运行代码时 出现错误 提示未
  • xdotool 类型需要很长时间并导致整个桌面冻结

    我一直在使用xdotool type过去只能在快捷方式上输入耸肩xdotool type 这可行 但总是需要相当长的时间 并导致整个桌面冻结 完全冻结 而不仅仅是输入 几秒钟 不过并没有太打扰我 现在我需要一种方法来从文件中读取内容 对其进
  • 变量作为 bash 数组索引?

    bin bash set x array counter 0 array value 1 array 0 0 0 for number in array do array array counter array value array co
  • 将 stdout 作为命令行 util 的文件名传递?

    我正在使用一个命令行实用程序 该实用程序需要传递文件名以将输出写入 例如 foo o output txt 它唯一写入的东西stdout是一条消息 表明它运行成功 我希望能够通过管道传输写入的所有内容output txt到另一个命令行实用程
  • 如何使用ffmpeg重叠和合并多个音频文件?

    我正在尝试将多个音频文件合并到一个文件中 但我可以使用以下命令来连接 而不是连接 ffmpeg v debug i file1 wav i file2 wav i file3 wav filter complex 0 0 concat n
  • 如何仅将整个嵌套目录中的头文件复制到另一个目录,在复制到新文件夹后保持相同的层次结构

    我有一个目录 其中有很多头文件 h 和其他 o 和 c 文件以及其他文件 这个目录里面有很多嵌套的目录 我只想将头文件复制到一个单独的目录 并在新目录中保留相同的结构 cp rf oldDirectory newDirectory将复制所有
  • 如何在 Linux x86_64 上模拟 iret

    我正在编写一个基于 Intel VT 的调试器 由于当 NMI Exiting 1 时 iret 指令在 vmx guest 中的性能发生了变化 所以我应该自己处理vmx主机中的NMI 否则 guest会出现nmi可重入错误 我查了英特尔手
  • 是否有可能在linux中找到包含特定文本的文件?

    考虑这种情况 我在文件夹 Example 下有很多文件 如果我需要找到一个包含特定短语 如 Class Example 的文件 我该如何使用 Linux shell 来做到这一点 linux中有类似 定位 的函数可以做到这一点吗 Thank
  • Linux 中的电源管理通知

    在基于 Linux 的系统中 我们可以使用哪些方法 最简单的方法 来获取电源状态更改的通知 例如 当计算机进入睡眠 休眠状态等时 我需要这个主要是为了在睡眠前保留某些状态 当然 在计算机唤醒后恢复该状态 您只需配置即可获得所有这些事件acp
  • 为什么 ld 无法从 /etc/ld.so.conf 中的路径找到库?

    我想添加 opt vertica lib64进入系统库路径 所以我执行以下步骤 1 添加 opt vertica lib64 into etc ld so conf 然后运行ldconfig 2 检查 bash ldconfig p gre
  • 无关的库链接

    我有一个可能有点愚蠢的问题 因为我很确定我可能已经知道答案了 假设你有静态库A 动态共享库B和你的linux下的程序C 假设库 A 调用库 B 中的函数 并且您的程序调用库 A 中的函数 现在假设 C 在 A 中调用的所有函数都不使用 B
  • 如何确保 numpy BLAS 库可用作动态加载库?

    The theano安装文档 http www deeplearning net software theano install html troubleshooting make sure you have a blas library指
  • CentOS目录结构是树形的吗?

    CentOS 上有相当于树的东西吗 如果你的 Centos 系统上没有安装 tree 无论如何我通常建议服务器设置使用最小安装磁盘 你应该在命令行中输入以下内容 yum install tree y 如果没有安装 那是因为您没有正确的存储库
  • Linux 上的 Python 3.6 tkinter 窗口图标错误

    我正在从 Python GUI 编程手册 学习 Python GUI 某项任务要求我通过将以下代码添加到我的配方中来更改窗口图标 Change the main windows icon win iconbitmap r C Python3
  • 如何从linux命令行运行.exe可执行文件? [关闭]

    Closed 这个问题不符合堆栈溢出指南 help closed questions 目前不接受答案 我在 Windows 中有一个 abc exe 可执行文件 我可以使用 DOS 命令提示来执行此应用程序 并为其提供一些运行时变量 我想从
  • Scrapy FakeUserAgentError:获取浏览器时发生错误

    我使用 Scrapy FakeUserAgent 并在我的 Linux 服务器上不断收到此错误 Traceback most recent call last File usr local lib64 python2 7 site pack

随机推荐

  • Linux的介绍

    简介 主要介绍Linux的概念 Linux是一款操作系统 类似与Windows 开源 免费 安全 高校 稳定 非常擅长处理高并发 现在大多数企业级项目都是部署在Linux系统的服务器中运行 Linux的创始人是Linus 吉祥物是一只叫Tu
  • 深入理解Mysql索引底层数据结构与算法

    索引是帮助MySQL高效获取数据的排好序的数据结构 索引数据结构对比 二叉树 左边子节点的数据小于父节点数据 右边子节点的数据大于父节点数据 如果col2是索引 查找索引为89的行元素 那么只需要查找两次 就可以获取到行元素所在的磁盘指针地
  • [labtools 27-2269]no devices detected on target localhost问题解决

    小白刚学FPGA 以流水灯为例入门 在连接板子的时候遇到了这个问题 记录一下 板子型号 xc7z045ffg900 2 解决办法之一 按照table 1 11更改图1 3里34位置处拨码
  • Linux系统中 systemd-journaldCPU占用异常的解决方法

    一 待解决问题 先贴几张图 问题解决之前最头疼的问题 因打印日志的高占用 以致CPU占用高达96 已经无法满足日常使用 从图中可见systemd journald占用了1 4的CPU资源 注 我是用的是Deepin系统 二 解决办法 因为要
  • SpringBoot2.x 集成Hadoop3.0.3 实现HDFS文件系统管理

    任务要求 搭建SpringBoot 2 x 集成Hadoop3 0 3环境 实现Hadoop 重要组成部分HDFS 文件系统管理的封装 核心pom xml 文件
  • vSphere之vCLS

    vCLS vSphere Cluster Services 是在vSphere7 0U1引入的集群服务 它使用代理虚拟机维护集群服务的运行状况 当主机添加到集群时 将创建 vCLS 代理虚拟机 vCLS vm 每个 vSphere 集群中最
  • Dell工作站8T硬盘安装ubuntu 16.04

    Dell工作站8T硬盘安装ubuntu 16 04 MBR文件系统仅支撑2T磁盘 因此在2T以上磁盘上安装ubuntu时 如果想利用全部磁盘空间 需要采用GPT分区 文件系统 模型 这需要重新分区 制作Ubuntu 16 04启动U盘 一
  • js-语言基础进阶-变换按钮的实现

    作者 芝士小熊饼干 系列专栏 数据结构 蓝桥杯 算法 坚持天数 16天
  • 从零搭建Maven私有仓库

    前言 主要使用到的技术 linux docker sonatype nexus maven 1 nexus3介绍 世界上第一个也是唯一一个免费使用的通用工件存储库 2 使用docker安装nexus3 1 下载 使用命令 docker pu
  • 20201105枚举课后总结

    文章目录 枚举 210733 奶牛碑文 http wikioi cn problem 210733 题目描述 输入格式 输出格式 样例输入 样例输出 思路 1 2 代码 210792 分解质因数 http wikioi cn problem
  • java会话技术--03--Session覆盖问题

    java会话技术 03 Session覆盖问题 代码位置 https gitee com DanShenGuiZu learnDemo tree master sessionCookie learn 1 现象 同一域名 同一个服务 不同的端
  • 如何在 vue 项目中引入高德地图

    文章目录 前言 一 申请 地图api开发者key 二 在vue项目安装高德地图的包 三 使用 1 在自己的组件中引入高德地图类 2 编写初始化函数 3 添加插件 前言 相信在 web 开发中有不少项目都用到过地图 那么我们怎么在自己的项目中
  • 程序删除自己

    void DeleteApplicationSelf std string strFileName this char szCommandLine MAX PATH 10 设置本进程为实时执行 快速退出 SetPriorityClass G
  • RabbitMQ消息队列实战(1)—— RabbitMQ的体系

    RabbitMQ是一个开源的消息代理和队列服务器 用来在不同的应用之间共享数据 1983年 被认为是RabbitMQ的雏形的Teknekron创建 首次提出了消息总线的概念 中间经历过数个阶段的发展 一直到2004年 AMQP Advanc
  • Makefile中的$@ $^等常见的符号解析

    之前学过一些Makefile 但是长时间不看 里面的符号又不少 慢慢就忘记了 这次在看Makefile文件 就顺带整理一些常用的符号 以后查询起来也方便 表示目标 表示所有的依赖 lt 表示第一个依赖 即时赋值 延时赋值 附加 例如 CC
  • bluestore中lru和2q缓存和onode_map的联系

    collection对应于某个pg 里面有OnodeSpace结构的onode map变量 OnodeSpace结构的变量内又有Cache结构的cache指针变量和unordered map的onode map变量 这个onode map变
  • 数据结构——>稀疏数组

    一 稀疏数组 1 定义 稀疏数组也叫稀疏矩阵 是普通数组的压缩 在这里我们可以说普通数组的无效数据量远远大于有效数据量 有效数据 在下方的例子中 非0数字就是有效数据 普通数组 其稀疏数组表现形式为 如果第一个例子没看懂 那我们再来举个例子
  • el-input @focus设置光标位置

    关于el input获取焦点失去焦点时光标位置可以使用 focus方法 随便写个el input 设置ref focus blur属性和方法
  • 『HTML5实现人工智能』小游戏《井字棋》发布,据说IQ上200才能赢【算法&代码讲解+资源打包下载】

    一 什么是TicTacToe 井字棋 本游戏为在下用lufylegend开发的第二款小游戏 此游戏是大家想必大家小时候都玩过 因为玩它很简单 只需要一张草稿纸和一只笔就能开始游戏 所以广受儿童欢迎 可能我说了半天 对它名字不熟悉的朋友也不懂
  • 树莓派基于Linux内核驱动开发详解

    目录 一 驱动认知 1 1 为什么要学习写驱动 1 2 文件名与设备号 1 3 open函数打通上层到底层硬件的详细过程 二 基于框架编写驱动代码 2 1 编写上层应用代码 2 2 修改内核驱动框架代码 2 3 部分代码解读 2 3 1 s