详细到吐血 —— 树莓派驱动开发入门:从读懂框架到自己写驱动

本文详述了树莓派驱动开发的全过程,从驱动的认知到编写GPIO驱动的步骤,包括理解设备号的作用、内核驱动框架的代码流程、如何在Ubuntu上交叉编译并部署到树莓派。此外,还介绍了3种地址的含义,并通过实战操作GPIO口输出高/低电平,展示了如何读取芯片手册、配置寄存器地址和进行功能配置。
摘要由CSDN通过智能技术生成

师承陈立臣

目录

README

emmm一不小心写了这么长的篇幅,建议配合目录一起看,从目录点击对应知识点,对知识体系和结构有整体的认识,阅读的时候才不会感到吃力。
在这里插入图片描述

一、驱动初步认知

为什么要学会写驱动?

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

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

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

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

设备号的两个作用?

区分硬件

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

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

在这里插入图片描述

索引驱动在驱动链表中的位置

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

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

  • 编写完驱动程序,加载到内核
  • 用户空间open后,调用驱动程序

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

从open到设备,从上层到底层,经历了什么?

  • 用户层调用open产生一个软中断(中断号是0x80),进入内核空间调用sys_callsys_call。
  • sys_callsys_call真正调用的是sys_open,去内核的驱动链表根据主设备号与次设备号找到相关驱动函数。
  • 调用驱动函数里面的open,去设置IO口引脚电平。

(对应下图的粉色笔迹)

在这里插入图片描述

二、基于内核驱动框架编写驱动代码流程

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

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.根据上层需求修改内核驱动框架代码

所谓框架,就是定死的东西,基本的语句必须要有,少一个都不行。

虽然有这么多的代码,但核心运行的就两个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函数
static ssize_t pin4_write(struct file *file,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,
    .write = pin4_write,
};

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

    int ret;
    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);  //创建设备文件

 
    return 0;
}

void __exit pin4_drv_exit(void)
{
   

    device_destroy(pin4_class,devno);
    class_destroy(pin4_class);
    unregister_chrdev(major, module_name);  //卸载驱动

}

module_init(pin4_drv_init);  //入口:内核加
C 64位读驱动是一种用于操作64位操作系统的设备驱动程序。64位操作系统是现代计算机系统中广泛使用的操作系统,与32位操作系统相比,它具有更大的寻址空间和更高的内存管理能力。 一个C 64位读驱动程序可以通过系统调用接口与操作系统内核进行通信,以在硬件设备上执行读取和入操作。该驱动程序通常由硬件设备的制造商提供,并在操作系统启动时加载并初始化。 驱动程序的主要功能是将数据从硬件设备读取到操作系统内存中,或将数据从内存入硬件设备。为了实现这一功能,驱动程序需要实现适当的读取和入函数,并与操作系统内核的设备驱动接口进行交互。 在编C 64位读驱动程序时,开发人员需要了解硬件设备的规范和特性,并使用适当的技术和接口进行编程。他们还需要确保驱动程序与操作系统内核正确地交互,并遵守操作系统内核的编程约定。 C 64位读驱动程序的编需要严格的错误处理和调试过程,以确保驱动程序在各种情况下都能正常工作并保证系统的稳定性和安全性。 总之,C 64位读驱动程序是用于与64位操作系统配合使用的设备驱动程序,其主要功能是实现数据在硬件设备和操作系统内存之间的读取和入操作。编这样的驱动程序需要适当的技术和接口知识,并要进行严格的错误处理和调试过程。
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值