一、驱动认知:参考链接https://blog.csdn.net/weixin_41679960/article/details/116209261
用户态
App:cp指令,ftp的项目等等这就是App的开发。
App开发需要C的基础和C库,C库讲到文件,进程,进程间通信,线程,网络,界面(GTk)。
C库(是linux标准库一定有):就是Clibary,提供了APP支配内核干活的接口,调用的open,read,write,fork,pthread,socket由此处封装实现,由写的应用程序调用,C库中的各种API调用的是内核态,支配内核干活
内核态
进程,内存,线程,网络,设备驱动不需要上层应用开发者关心,驱动在前面的学习是调用wringpi库实现,接下来的学习就是要自己实现wringpi库,因为wringpi库由厂家提供,可能有可能没有,这样当我们拿到另一种类型的板子时,同样也可以完成开发。
驱动链表:管理所有设备的驱动,添加或查找,添加是发生在我们编写完驱动程序,加载到内核。查找发生在调用驱动程序,由应用层用户空间去open。
驱动插入链表的顺序由设备号检索,就是说主设备号和次设备号除了除了能区分不同种类的设备和不同类型的设备,还能起到将驱动程序创到链表的某个位置。
硬件
linux经典的一句话一切皆文件:包括文件,设备(鼠标,键盘,LED,屏幕,flash,内存,网卡等),普通的IO口,串口等等。
ls -l 查看设备号
驱动代码的开发无非就是添加驱动和调用驱动。
–>用户态调用引脚四open(“/de/pin4”,O_RDWR)
–>调用System call interface(这个函数是在内核中的)
–>这个函数会根据设备名找到设备号
–>调用VFS中的sys_open
–>sys_open会找到引脚4中的open函数
–>引脚4中的open函数是对寄存器的操作(实操)
–>用户态进入内核态会发生一次软中断 ,中断号是0x80。
二、代码框架编写即编译
要进入到源码树目录底下进行编译
cd ~/SYSTEM/linux-rpi-4.14.y/drivers/char
驱动代码:pin4driver2.c
#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; //主设备号 创建不同的驱动时这边需要修改值 231
static int minor =0; //次设备号 创建不同的驱动时这边需要修改值 0
static char *module_name="pin4"; //模块名
//是个寄存器变量不希望编译器优化而省略,且要求每次直接读值用volatile
//因为是个地址希望是无符号的用unsigned
volatile unsigned int *GPFSEL0 = NULL; //功能选择 输出/输入
volatile unsigned int *GPSET0 = NULL; //输出
volatile unsigned int *GPCLR0 = NULL; //清除
//led_open函数
static int pin4_open(struct inode *inode,struct file *file)
{
printk("pin4_open\n"); //内核的打印函数和printf类似
//配置Pin4引脚为输出引脚 ,bit 12--14 配置成001
*GPFSEL0 &= ~(0x6 << 12); //把bit14 bit13 配置成0 创建不同的驱动时这边需要修改值 12
*GPFSEL0 |= (0x1 << 12); //把bit12 配置成1 创建不同的驱动时这边需要修改值 12
return 0;
}
//led_write函数
static ssize_t pin4_write(struct file *file,const char __user *buf,size_t count, loff_t *ppos)
{
int userCmd;
printk("pin4_write\n");
//获取上层write函数的值
copy_from_user(&userCmd,buf,count);
//根据值来操作io口,高电平,或者低电平
//printk("get value\n");
if(userCmd == 1){
printk("set 1\n");
*GPSET0 |= 0x1 << 4; //4代表第四个引脚 创建不同的驱动时这边需要修改值 4
}else if(userCmd == 0){
printk("set 0\n");
*GPCLR0 |= 0x1 << 4;
}else{
printk("undo\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"); //创建不同的驱动时这边需要修改值 "myfirstdemo"
pin4_class_dev =device_create(pin4_class,NULL,devno,NULL,module_name); //创建设备文件
GPFSEL0 = (volatile unsigned int *)ioremap(0x3f200000,4); //用ioremap将物理地址转换成虚拟地址 4是4个字节 io口寄存器映射成普通内存单元进行访问
GPSET0 = (volatile unsigned int *)ioremap(0x3f20001C,4);
GPCLR0 = (volatile unsigned int *)ioremap(0x3f200028,4);
return 0;
}
void __exit pin4_drv_exit(void)
{
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");
上层调用代码pin4test.c
#include <stdio.h>
#include <sys/types.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <unistd.h>
int main()
{
int fd;
char buf[128] = {0};
fd = open("/dev/pin4",O_RDWR);
if(fd < 0){
printf("open failed\n");
perror("reason");
}else{
printf("open success\n");
}
fd = write(fd,buf,1);
return 0;
}
要让工程能编译到pin4driver2.c 要对Makefile进行修改
ps:文件放在哪个路径就要修改该路径下的vi Makefile
vi Makefile
在第二行加入
obj-m += pin4driver2.o
# SPDX-License-Identifier: GPL-2.0
#
# Makefile for the kernel character device drivers.
#
obj-y += mem.o random.o
obj-m += pin4driver2.o //编译成模块的方式(这句话自己添加)
obj-$(CONFIG_TTY_PRINTK) += ttyprintk.o
obj-y += misc.o
obj-$(CONFIG_ATARI_DSP56K) += dsp56k.o
obj-$(CONFIG_VIRTIO_CONSOLE) += virtio_console.o
obj-$(CONFIG_RAW_DRIVER) += raw.o
obj-$(CONFIG_SGI_SNSC) += snsc.o snsc_event.o
obj-$(CONFIG_MSPEC) += mspec.o
obj-$(CONFIG_UV_MMTIMER) += uv_mmtimer.o
obj-$(CONFIG_IBM_BSR) += bsr.o
obj-$(CONFIG_SGI_MBCS) += mbcs.o
obj-$(CONFIG_BFIN_OTP) += bfin-otp.o
obj-$(CONFIG_PRINTER) += lp.o
obj-$(CONFIG_APM_EMULATION) += apm-emulation.o
obj-$(CONFIG_DTLK) += dtlk.o
obj-$(CONFIG_APPLICOM) += applicom.o
obj-$(CONFIG_SONYPI) += sonypi.o
obj-$(CONFIG_RTC) += rtc.o
obj-$(CONFIG_HPET) += hpet.o
obj-$(CONFIG_EFI_RTC) += efirtc.o
obj-$(CONFIG_DS1302) += ds1302.o
obj-$(CONFIG_XILINX_HWICAP) += xilinx_hwicap/
ifeq ($(CONFIG_GENERIC_NVRAM),y)
obj-$(CONFIG_NVRAM) += generic_nvram.o
else
obj-$(CONFIG_NVRAM) += nvram.o
endif
obj-$(CONFIG_TOSHIBA) += toshiba.o
回到模块源码进行内核编译
cd ~/SYSTEM/linux-rpi-4.14.y
输入一下代码进行编译
ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make modules
成功后会生成 .ko文件
CHK include/config/kernel.release
CHK include/generated/uapi/linux/version.h
CHK include/generated/utsrelease.h
CHK include/generated/bounds.h
CHK include/generated/timeconst.h
CHK include/generated/asm-offsets.h
CALL scripts/checksyscalls.sh
CHK scripts/mod/devicetable-offsets.h
CHK kernel/config_data.h
CC [M] drivers/char/pin4driver2.o
Building modules, stage 2.
MODPOST 1523 modules
CC drivers/char/pin4driver2.mod.o
LD [M] drivers/char/pin4driver2.ko
在对pin4test.c进行编译
arm-linux-gnueabihf-gcc -o pin4test
拷贝到树莓派上面进行测试
scp drivers/char/pin4driver2.ko pi@192.168.0.105:/home/pi
scp drivers/char/pin4test pi@192.168.0.105:/home/pi
在树莓派上面首先加载内核驱动
sudo insmod pin4driver2.ko
ps:如果要卸载的话指令是:
sudo rmmod pin4driver2
查看是否生成/dev/pin4的驱动
ls /dev/pin4
接着修改pin4test的权限为:
sudo chmod 666 /dev/pin4
运行./pin4test发现看不到现象
原因:看到的界面是在上层,要在内核态才能查看到数据
解决方法:运行指令
dmesg
会在最后一行看到添加成功
pin4_open