【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第三十六章 Linux驱动初探

i.MX8MM处理器采用了先进的14LPCFinFET工艺,提供更快的速度和更高的电源效率;四核Cortex-A53,单核Cortex-M4,多达五个内核 ,主频高达1.8GHz,2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT、4G模块、CAN、RS485等接口一应俱全。H264、VP8视频硬编码,H.264、H.265、VP8、VP9视频硬解码,并提供相关历程,支持8路PDM接口、5路SAI接口、2路Speaker。系统支持Android9.0(支持获取root限)Linux4.14.78+Qt5.10.1、Yocto、Ubuntu20、Debian9系统。适用于智能充电桩,物联网,工业控制,医疗,智能交通等,可用于任何通用工业和物联网应用、

【公众号】迅为电子

【粉丝群】258811263


第三篇 嵌入式Linux驱动开发篇

第一部分 Linux驱动初探

三十六 章 Linux驱动 初探

本章导读

本章将介绍初步探索linux驱动开发。

36.1章节讲解了linux设备驱动开发基本概念

36.2章节讲解了linux三大设备驱动的特点

36.3章节以最简单的驱动-helloworld为例,编写入门驱动开发的第一个驱动例程。

本章内容对应视频讲解链接(在线观看):

什么是linux驱动 → https://www.bilibili.com/video/BV1Vy4y1B7ta?p=3

Linux第一个驱动Helloworld → https://www.bilibili.com/video/BV1Vy4y1B7ta?p=4

36.1 linux设备驱动开发基本概念

任何一个计算机系统的运转都是系统中软硬件共同努力的结果,没有硬件的软件是空中楼阁,而没有

软件的硬件则只是一堆废铁。硬件是底层基础,是所有软件得以运行的平台,代码最终会落实为硬件上的

组合逻辑与时序逻辑;软件则实现了具体应用,它按照各种不同的业务需求而设计,并完成用户的最终诉

求。硬件较固定,软件则很灵活,可以适应各种复杂多变的应用。因此,计算机系统的软硬件相互成就了

对方。驱动程序负责硬件和应用软件之间的沟通,而驱动工程师则负责硬件工程师和应用软件工程师之间的沟通,那么从字面意思来看,设备驱动最通俗的解释就是"驱使硬件设备行动"。在学习驱动之前,我们先了解一些基础概念。

概念一 裸机编程或单片机开发

裸机编程,顾名思义,就是直接在硬件上编程写代码,或者说编写直接在硬件上运行的程序,没有操作系统的支持。一般我们把没有操作系统的编程环境,称为裸机编程环境,比如在单片机上编程。通过串口直接将程序下载到单片机芯片内部的Flash中,单片机运行时,直接调用我们编程的程序。这时,我们编写的程序一般都有一个while 1的死循环存在,这样程序才能一直保持运行。裸机编程现在主要是正对低端的嵌入式系统,如SCM(single chip machine)、各式MCU、DSP等。当然,编写PC的bootloader肯定也属于裸机编程。

单片机是一种集成电路芯片,是用大规模集成电路技术通过编程数据处理能力的中央处理器CPU、随机存储器RAM、只读存储器ROM、多种I/O口和中断系统、定时器/计时器等功能,这其中还包括显示驱动电路、脉宽调制电路、模拟多路转换器、A/D转换器等功能等集成到一块小芯片上。

单片机开发包括:程序设计(PC端软件编程),程序送往执行(将编程好的软件下载到单片机,需要编程器或者下载线),单片机系统的设计(硬件上设计你需要的任务的执行机构,如控制开关,温度检测,红外传输等等,都是根据你的所需,然后在选择对应的硬件器件)。

概念二 linux系统开发

基于linux操作系统来开发我们的产品叫linux系统开发。此开发的编程方式和裸机开发的编程方式是截然不同的。裸机开发的编程方式是直接运行在硬件之上,不与任何操作系统关联。操作系统的存在势必要求设备驱动附加更多的代码和功能,把单一的"驱使硬件设备行动"变成了操作系统内与硬件交互的模块,它对外呈现为操作系统的API,不再给应用软件工程师直接提供接口。由此可见,当系统中存在操作系统的时候,驱动变成了连接硬件和内核的桥梁。

linux系统开发框架如下图所示,最上层的是应用软件,下面是操作系统,再下面是驱动程序,最后是我们的硬件。如果在硬件上跑操作系统,驱动程序是位于硬件和操作系统中间的,是连接操作系统和硬件之间的桥梁。

这里拿linux操作系统给大家举个例子,linux系统可以跑到不同的硬件上面如pc机或者arm开发板上面。如果linux操作系统跑到PC机上,那么驱动就要适配PC机; 如果linux操作系统跑到arm开发板上面,那么驱动就要适配arm开发板;所以说不同的硬件架构都可以跑linux,但是它的驱动程序是不同的,那么linux操作系统源码都是一样的,唯一不一样的就是驱动程序了。也就是说同一个操作系统可以跑到不同的硬件上面,但是驱动程序是有差异的,因为驱动程序是操作系统和硬件连接的一个桥梁。

概念三 系统移植 linux驱动移植

移植是说同样的一个linux操作系统 ,我们可以跑到不同的硬件上面,我们把操作系统移植到不同的硬件上面,这个过程叫做移植。设备驱动移植步骤,如下图所示:

概念四 应用软件

在操作系统上面有应用软件,应用软件程序的执行是依赖于操作系统的,应用程序需要调用linux操作系统的库函数来实现,也就是说,应用软件的程序会调用linux操作系统的函数来完成对硬件的操作,那么应用程序是不能对硬件直接进行操作的。

概念五 linux系统架构优点

linux系统开发架构和我们裸机的架构是不同的,架构相比于裸机架构是非常复杂了,那么我们使用这个架构都有什么好处呢?

1 有了系统的架构后,开发起来就非常容易了,因为它有自己的框架,这种框架都是非常成熟的框架,我们直接按照框架开发就可以了,框架给我们提供了很多现成的功能。

2 这个框架让我们的系统变得更加安全,因为我们的应用软件不能直接对硬件进行操作,它要借助操作系统来对硬件进行操作。如果应用软件有好几个,假如其中的一个应用软件崩溃了,它不会影响我们整个系统的运行,不会造成系统的死机,这样就会让系统更加安全,出问题的概率变得更小了。

36.2 linux设备驱动分类及特点

计算机系统的硬件主要由CPU、存储器和外设组成。随着IC制作工艺的发展,目前,芯片的集成度越

来越高,往往在CPU内部就集成了存储器和外设适配器。譬如,相当多的ARM、PowerPC、MIPS等处理器都集成了UART、I2C控制器、SPI控制器、USB控制器、SDRAM控制器等,有的处理器还集成了GPU(图形处理器)、视频编解码器等。驱动针对的对象是存储器和外设(包括CPU内部集成的存储器和外设,而不是针对CPU内核。Linux将存储器和外设分为3个基础大类:字符设备驱动,块设备驱动,网络设备驱动。

其中,理解和掌握字符设备驱动的概念最重要,因为在后续工作中我们遇到的大部分是字符设备。为什么会这么说呢?比如说我们选了一个cpu,不管它是哪个厂家的,比如它是三星的或者恩智浦的或者TI(德州仪器),那么他们都会给你提供一个bsp包,在这个开发包里面,像块设备驱动和网络设备驱动已经做好了,我们要做的事情是配置一下就可以用了。平常开发的时候用的最多的就是字符设备驱动,我们掌握了字符设备的开发,那么我们开发产品基本上就没有什么问题了。

字符设备指那些必须以串行顺序依次进行访问的设备,如触摸屏、磁带驱动器、鼠标等。块设备可以按任意顺序进行访问,以块为单位进行操作,如硬盘、eMMC等。字符设备和块设备的驱动设计有出很大的差异,但是对于用户而言,它们都要使用文件系统的操作接口open()、close()、read()、write()等进行访问。网络设备主要有哪些呢?从字面意思可以看出,和网络相关的都是网络设备,比如WiFi,以太网。在Linux系统中,网络设备面向数据包的接收和发送而设计,它并不倾向于对应于文件系统的节点。内核与网络设备的通信与内核和字符设备、网络设备的通信方式完全不同,网络设备主要还是使用套接字接口。

36.3 最简单的设备驱动- helloworld

程序源码在网盘资料"iTOP-i.MX8MM开发板\02-i.MX8MM开发板网盘资料汇总(不含光盘内容)\嵌入式Linux开发指南(iTOP-i.MX8MM)手册配套资料\2.驱动程序例程\01-第一个驱动helloworld"路径下。

经过前面的学习,我们了解了驱动开发的框架,本章节将带领大家实验操作,写最简单的驱动-helloworld。

Linux设备驱动会以内核模块的形式出现,因为linux内核的整体架构就非常庞大,包含的组件也非常多,如果把所有的功能都编译到linux内核中会使得内核非常臃肿,为了解决这个问题,更方便地新增和删除功能,linux提供了这样的机制,这种机制被称为模块。为了大家对模块有一个感性的认识,我们先来看一个最简单的驱动-helloworld。

驱动分为四个部分:

  • 头文件
  • 驱动模块的入口函数和出口函数
  • 声明信息
  • 功能实现

我们在windows上面新建一个helloworld.c文件,这里使用sourceinsight来编辑文件,大家也可以用其他编译器来编写程序。

第一步 包含头文件

#include <linux/init.h> //包含宏定义的头文件

#include <linux/module.h> //包含初始化加载模块的头文件

第二步 驱动模块的入口函数和出口函数

module_init();

module_exit();

第三步 声明模块拥有开源许可证

MODULE_LICENSE("GPL");

第四步 功能实现:内核模块加载的时候打印hello world! ,内核模块卸载的时候打印gooodbye!

注意:内核打印函数不能用printf,因为内核没有办法使用C语言库。

cpp 复制代码
static int hello_init(void){
    printk("hello world! \n");
    return 0;
} 
static void hello_exit(void){
    printk("gooodbye! \n");
}

完整的一个最简单的Linux内核模块,如下图所示:

此时,我们需要有一个感性认识,代码中的某些陌生元素都是linux内核为了字符设备定义的,以实现驱动与内核接口而定义的。Linux对各类设备的驱动都定义了类似的数据结构和函数。

相关推荐
S+叮当猫1 小时前
第五部分:2---信号的介绍、产生、处理
linux·运维·服务器
东城绝神2 小时前
《Linux运维总结:基于ARM64+X86_64架构CPU使用docker-compose一键离线部署mongodb 7.0.14容器版副本集群》
linux·运维·mongodb·架构
WZF-Sang4 小时前
Linux权限理解【Shell的理解】【linux权限的概念、管理、切换】【粘滞位理解】
linux·运维·服务器·开发语言·学习
小橞5 小时前
Linux搭建简易路由转发
linux·运维·服务器
robot_大菜鸟5 小时前
linux-L7-linux 查看json文件
linux·运维
Flying_Fish_roe5 小时前
linux-网络管理-网络抓包
linux·网络·php
小技与小术7 小时前
lvs-nat模式实验详解
linux·运维·服务器·网络·lvs
梅双单片机7 小时前
【源代码+仿真+原理图+技术文档+演示录屏+软件】基于单片机的厨房监控系统的设计与实现
单片机·嵌入式硬件
py.鸽鸽7 小时前
Linux2-mkdir,touch,cat,more
linux
aabbcc456aa7 小时前
ubuntu安装mysql 8.0忘记root初始密码,如何重新修改密码
linux·mysql·ubuntu