软件版本:Vivado2021.1
操作系统:WIN10 64bit、Ubuntu18.04
硬件平台:ZYNQ UltraScale
文章目录
- 1.1系统框图
- 1.2介绍
- 1.2.1寄存器查询手册
- 1.2.2物理地址与虚拟地址
- 1.2.3MIO介绍
- 1.2.4PS的LED 引脚介绍
- 1.3搭建工程
- 1.4程序分析
- 1.4.1驱动程序分析
- 1:内存映射
- 2:设置 MIO 功能
- 3:设置 MIO 输出电流的大小
- 4:设置引脚是 Schmitt Trigger 还是 CMOS Input
- 5:设置引脚的上下拉及使能
- 6:设置 MIO 的速率
- 7:设置 GPIO 的输入输出
- 8:通过 Data 的方式熄灭 LED
- 9:内存释放
- 1.5交叉编译
- 1.6演示
1.1系统框图
在 PS 端接有两个 LED 灯,这些灯接在了 MIO 上,而 MIO 可以复用成为 GPIO,因此控制 GPIO 其实就是控制了 LED 等器件。本文通过读写寄存器来实现对 GPIO 的控制。
1.2介绍
1.2.1寄存器查询手册
使用 GPIO 需要设置两个寄存器,一个是设置 GPIO 的管脚复用的 IOU_SLCR Module 寄存器,一个是设置GPIO 的管脚功能的 GPIO Module 寄存器。查询寄存器地址需要用到一个官方文档,编号的 ug1087。Xilinx 官方网站:https://docs.xilinx.com/r/en-US/ug1087-zynq-ultrascale-registers/MIO_PIN_41-IOU_SLCR-Register。
1.2.2物理地址与虚拟地址
提到寄存器,那就不得不说说地址了。寄存器作为系统中存储数据的器件,每个数据都有其自身的物理地址。而与物理地址相对的则是 Linux 系统下的虚拟地址。在裸机等通常程序里,是不存在虚拟地址一说的,对地址的操作就是对物理地址的操作。但是在 Linux 下,对物理地址直接访问是行不通的,需要通过虚拟地址间接访问,因此所有对寄存器的操作都需要通过地址映射。
之所以要使用虚拟地址是因为以前的电脑内存非常小,程序也是非常小,因此靠手动便可以管理内存。但是日新月异,今天的程序已经非常庞大了,虽然内存也在飞速发展,但是无法跟上日益臃肿的应用程序。众所周知要想运行一个程序,就必须将其完整地读入内存,如果程序本身就大于内存,那除了加内存可以运行程序,那剩下的唯一办法就是使用虚拟地址了。
虚拟地址可以模拟出几倍于物理地址的内存空间,当程序运行时,被运行的部分读入内存,其他部分依旧存放于外存。物理地址于虚拟地址的关系如下图。
虚拟内存与物理内存是对应的,但是没有顺序。这个顺序是由 MMU 管理的,MMU 全称 Memory Management Unit,即内存管理单元。它的功能包括虚拟地址到物理地址的转换(即虚拟内存管理)、内存保护、中央处理器高速缓存的控制等等。
在本章驱动编写时,会使用 ioremap、iounmap 这两个函数,是用来申请物理地址对应的虚拟地址的。因为在Linux 内核启动之后,会初始化 MMU,将物理地址映射为虚拟地址。这个时候,不能向寄存器的物理基地址写入
数据,而是应该用 ioremap 向系统申请的虚拟地址,然后在虚拟地址上进行寄存器的写入和读取操作。
1.2.3MIO介绍
ZYNQ Ultrascale+ MPSOC PS 部分的 IO 包括 PS-MIO 和 PS-EMIO,MIO 分布在 PS 的 Bank0 、BANK1 和Bank2,EMIO 分布在 PS 的 Bank3、BANK4 和 Bank5。PS-MIO 具有 78 个 GPIO ,PS-EMIO 具有最多 96 个。PS-MIO的 IO 位置是固定好的,功能也是预先定义好了,而 PS-EMIO 是通过把芯片内部 PS 的 PS-EMIO 引线接到了 PL 部分的 FPGA Pin 脚上,所以 PS-MIO 满足必要常用的外设 IO 需求,比如串口、SDIO、以太网等,而 PS-EMIO 按需配置,用多少,配置多少。
芯片有 78 个 MIO(multiuse I/O),它们派发在 GPIO 的,隶属于 PS 部分。这些 IO 与 PS 直接相连。不需要添加引脚约束,MIO 信号对 PL 部分是透明的,不可见。所以对 MIO 的操作可以看作是纯 PS 的操作。GPIO 控制和状态寄存器是从基地址 0xFF0A_0000 开始的存储器映射,并受 XPPU 保护。
PS-MIO:
Bank0:MIO[0 :25] GPIO PIN 脚号:0~25
Bank1:MIO[26:51] GPIO PIN 脚号:26~51
Bank2:MIO[52:77] GPIO PIN 脚号:52~77
PS-EMIO:
Bank3:EMIO[ 0:31] 可以分配到任意的 FPGA IO
Bank4:EMIO[32:63] 可以分配到任意的 FPGA IO
Bank5:EMIO[64:95] 可以分配到任意的 FPGA IO
注意:GPIO [92:95]四个输出可以用作 PL 中用户定义逻辑的复位信号。 GPIO EMIO 信号的数量取决于在 Vivado PS
配置向导(PCW)中选择的 PL 结构复位的数量。 例如,如果选择一个复位,则将 GPIO [95]分配为复位信号。如果选择两个,则分配
GPIO [95:94]。
MIO 内部构造:
DATA_RO:
此寄存器使能软件观察 PIN 脚,当 GPIO 被配置成输出的时候,这个寄存器的值会反应输出的 PIN 脚情况。
DATA:
此寄存器控制输出到 GPIO 的值,读这个寄存器的值可以读到最后一次写入该寄存器的值。
MASK_DATA_LSW:
位操作寄存器,写入 GPIO 低 16bit 其他没有改变的位置保存原先的状态
MASK_DATA_MSW:
位操作寄存器,写入 GPIO 高 16bit 其他没有改变的位置保存原先的状态
DIRM:
此寄存器控制输出的开关,当 DIRM[x]==0 时候,禁止输出
OEN:
输出使能,当 OEN[x]==0 的时候输出关闭,PIN 脚处于三态。
因 此 , 如果要读 IO 状态就得读DATA_RO 的值 , 如果是对某一位 进行操作就是写
MASK_DATA_LSW/MASK_DATA_MSW
1.2.4PS的LED 引脚介绍
从原理图可以看出,当 MIO 高电平的时候,LED 可以点亮。低电平的时候,MIO 变暗。
还可以查到如下数据:
- LED1:MIO42
- LED2:MIO40
查看 1.2.1寄存器手册,这四个管脚均属于 bank1,如果有 MIO 在其他 bank 上,我们需要将一样的代码再写一遍以控制全部LED,这里我们只控制 bank1 的 LED1 和 LED2。
1.3搭建工程
在IP核中勾选 GPIO1 MIO:
制作开发板系统.
1.4程序分析
1.4.1驱动程序分析
//添加头文件
#include <linux/init.h>
#include <linux/module.h>
#include <linux/gpio.h>
static int led_major;
static struct class *led_cls;
/* gpio 内存映射地址 */
static unsigned long *gpio_addr = NULL;
static unsigned long *iou_slcr = NULL;
/* gpio 寄存器物理基地址 */
#define GPIO_BASE_ADDR 0xFF0A0000
#define IOU_SLCR_ADDR 0xFF180000
/* gpio 寄存器所占空间大小 */
#define GPIO_BASE_SIZE 0x368
#define IOU_SLCR_SIZE 0x714
/* gpio 引脚设置GPIO功能*/
#define GPIO_PIN_40 (unsigned int *)((unsigned long)iou_slcr + 0x000000A0)
#define GPIO_PIN_42 (unsigned int *)((unsigned long)iou_slcr + 0x000000A8)
/* gpio 控制电流0*/
#define IOU_SLCR_BANK1_CTRL0 (unsigned int *)((unsigned long)iou_slcr + 0x00000154)
/* gpio 控制电流1*/
#define IOU_SLCR_BANK1_CTRL1 (unsigned int *)((unsigned long)iou_slcr + 0x00000158)
/* gpio 输入引脚选择Schmitt或CMOS*/
#define IOU_SLCR_BANK1_CTRL3 (unsigned int *)((unsigned long)iou_slcr + 0x0000015C)
/* gpio 设置上下拉*/
#define IOU_SLCR_BANK1_CTRL4 (unsigned int *)((unsigned long)iou_slcr + 0x00000160)
/* gpio 使能上下拉*/
#define IOU_SLCR_BANK1_CTRL5 (unsigned int *)((unsigned long)iou_slcr + 0x00000164)
/* gpio 引脚速率选择*/
#define IOU_SLCR_BANK1_CTRL6 (unsigned int *)((unsigned long)iou_slcr + 0x00000168)
/* gpio 方向寄存器 */
#define GPIO_DIRM_1 (unsigned int *)((unsigned long)gpio_addr + 0x00000244)
/* gpio 使能寄存器 */
#define GPIO_OEN_1 (unsigned int *)((unsigned long)gpio_addr + 0x00000248)
/* gpio 输出数据寄存器 */
#define GPIO_DATA_1 (unsigned int *)((unsigned long)gpio_addr + 0x00000044)
/* gpio 输出数据控制寄存器1 */
#define GPIO_MASK_DATA_1_LSW (unsigned int *)((unsigned long)gpio_addr + 0x00000008)
/* gpio 输出数据控制寄存器2 */
#define GPIO_MASK_DATA_1_MSW (unsigned int *)((unsigned long)gpio_addr + 0x0000000C)
const struct file_operations led_fops = {};
//实现装载入口函数和卸载入口函数
static __init int led_drv_v1_init(void)
{
printk("-------^v^-------\n");
printk("-led drv v1 init-\n");
//申请主设备号
led_major = register_chrdev(0, "led_drv_v1", &led_fops);
if (led_major < 0)
{
printk("register chrdev faile!\n");
return led_major;
}
printk("register chrdev ok!\n");
//创建设备节点
//创建设备的类别
led_cls = class_create(THIS_MODULE, "led_class");
printk("class create ok!\n");
//创建设备
device_create(led_cls, NULL, MKDEV(led_major, 0), NULL, "Myled%d", 0);
printk("device create ok!\n");
//1.内存映射
gpio_addr = ioremap(GPIO_BASE_ADDR, GPIO_BASE_SIZE);
iou_slcr = ioremap(IOU_SLCR_ADDR, IOU_SLCR_SIZE);
printk("gpio_addr init over!\n");
//2.MIO40 MIO42 设置成GPIO,参考手册设置
iowrite32((ioread32(GPIO_PIN_40) & 0x0), GPIO_PIN_40);
printk("gpio MIO40 init over!\n");
iowrite32((ioread32(GPIO_PIN_42) & 0x0), GPIO_PIN_42);
printk("gpio MIO42 init over!\n");
// //3.MIO40 MIO42设置输出驱动电流大小
// // 将 0x3FFFFFF 和 0x0 转换为二进制。
// // 0011 1111 1111 1111 1111 1111 1111
// // 0000 0000 0000 0000 0000 0000 0000
// // 参考手册,可以知道每一位的地址都是由 CTRL0 和 CTRL1 控制的,在这里都是 10, 查看手册得 8 mA。
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL0) | 0x3FFFFFF), IOU_SLCR_BANK1_CTRL0);
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL1) & 0x0), IOU_SLCR_BANK1_CTRL1);
//4.选择引脚是Schmitt还是CMOS
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL3) & 0x0), IOU_SLCR_BANK1_CTRL3);
//5.输出管脚上下拉及使能
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL4) | 0x3FFFFFF), IOU_SLCR_BANK1_CTRL4);
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL5) | 0x3FFFFFF), IOU_SLCR_BANK1_CTRL5);
//6.MIO速率的选择
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL6) & 0x0), IOU_SLCR_BANK1_CTRL6);
//7.MIO40 MIO42 设置成输出
// 40 - 26 = 14, 42 - 26 = 16
// 0000 0000 000|0 0|000 0000 0000 0000
// |16 |14
// 0000 0000 000 1 0 100 0000 0000 0000
// 0000 0000 0001 0100 0000 0000 0000
// 0x00014000
iowrite32((ioread32(GPIO_DIRM_1) | 0x00014000), GPIO_DIRM_1);
//MIO40 MIO42 使能
iowrite32((ioread32(GPIO_OEN_1) | 0x00014000), GPIO_OEN_1);
/* MASK_DATA方式按灭LED1,LED2,这个需要自己修改相关参数 */
//iowrite32((ioread32(GPIO_MASK_DATA_0_LSW ) & 0xFEFFEFFF), GPIO_MASK_DATA_0_LSW );
//printk("GPIO_MASK_DATA_0_LSW = 0x%x\n", *GPIO_MASK_DATA_0_LSW);
//iowrite32((ioread32(GPIO_MASK_DATA_0_MSW ) & 0xFEFFEFFF), GPIO_MASK_DATA_0_MSW );
//printk("GPIO_MASK_DATA_0_MSW = 0x%x\n", *GPIO_MASK_DATA_0_MSW);
//8.DATA方式按灭LED1,LED2
// 1111 1111 1111 111|1 1|111 1111 1111 1111
// |0 |0
// 1111 1111 1111 1110 1011 1111 1111 1111
// 0xFFFEBFFF
// iowrite32((ioread32(GPIO_DATA_1) & 0xFFFEBFFF), GPIO_DATA_1);
iowrite32((ioread32(GPIO_DATA_1) & 0xFFFEFFFF), GPIO_DATA_1);
printk("GPIO_DATA_1 = 0x%x\n", *GPIO_DATA_1);
return 0;
}
static __exit void led_drv_v1_exit(void)
{
iowrite32((ioread32(GPIO_DATA_1) | 0xFFFFFFFF), GPIO_DATA_1);
//9.内存释放
iounmap(gpio_addr);
iounmap(iou_slcr);
//删除设备
device_destroy(led_cls, MKDEV(led_major, 0));
//删除类
class_destroy(led_cls);
//注销主设备号
unregister_chrdev(led_major, "led_drv_v1");
printk("-------^v^-------\n");
printk("-led drv v1 exit-\n");
}
//申明装载入口函数和卸载入口函数
module_init(led_drv_v1_init);
module_exit(led_drv_v1_exit);
//添加GPL协议
MODULE_LICENSE("GPL");
MODULE_AUTHOR("msxbo");
行 9~11,用来储存映射后虚拟地址的基地址。87、88 行赋值。
行 13~15,是我们要控制的两个寄存器物理地址的基地址。
行 16~18,是两个寄存器的完整大小,在地址转换的时候直接映射整个寄存器,方便理解。
行 20~22,定义设置复用功能的寄存器。
行 24~35,六个控制电气特性的寄存器。
行 37~46,GPIO 的控制寄存器。
行 113~117,是使用 mask_data 的方法熄灭 led,和 data 方法熄灭效果一样。
行 119~126,使用 data 方法熄灭 led。
1:内存映射
static __init int led_drv_v1_init(void)
{
//1.内存映射
gpio_addr = ioremap(GPIO_BASE_ADDR, GPIO_BASE_SIZE);
iou_slcr = ioremap(IOU_SLCR_ADDR, IOU_SLCR_SIZE);
printk("gpio_addr init over!\n");
return 0;
}
含义:内存映射的目的是为了把物理地址映射为虚拟地址,方便对寄存器进行操作。
- GPIO_BASE_ADDR:这是寄存器手册中查询到的“GPIO Module”的“Base Address”。
- GPIO_BASE_SIZE:这是该寄存器的容量,也就是“GPIO Module”的容量。从“GPIO Module”的最后一行可以看到,它的最后一位是“0x0000000364”,位宽是 32 位。简单的算下,就是 4 字节,所以“GPIO Module”的容量为“0x0000000368”。
2:设置 MIO 功能
static __init int led_drv_v1_init(void)
{
//2.MIO40 MIO42 设置成GPIO,参考手册设置
iowrite32((ioread32(GPIO_PIN_40) & 0x0), GPIO_PIN_40);
printk("gpio MIO40 init over!\n");
iowrite32((ioread32(GPIO_PIN_42) & 0x0), GPIO_PIN_42);
printk("gpio MIO42 init over!\n");
return 0;
}
含义:将 MIO 的功能设置为 GPIO。
具体分析:
• iowrite32:这个函数会把前面的值写入到后面的目的地址。
• ioread32:这个函数则是读取目的地址的值。
• 0x0:这个部分需要查询寄存器手册的“IOU_SLCR Module”部分,找到需要设置的 MIO 引脚的序号,然后根据自己的需要设置手册中的相应的值。
•GPIO_PIN_40:这是在之前确定好的“iou_slcr”进行偏移的地址。偏移地址可以在寄存器手册中查询到
3:设置 MIO 输出电流的大小
static __init int led_drv_v1_init(void)
{
// //3.MIO40 MIO42设置输出驱动电流大小
// // 将 0x3FFFFFF 和 0x0 转换为二进制。
// // 0011 1111 1111 1111 1111 1111 1111
// // 0000 0000 0000 0000 0000 0000 0000
// // 参考手册,可以知道每一位的地址都是由 CTRL0 和 CTRL1 控制的,在这里都是 10, 查看手册得 8 mA。
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL0) | 0x3FFFFFF), IOU_SLCR_BANK1_CTRL0);
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL1) & 0x0), IOU_SLCR_BANK1_CTRL1);
return 0;
}
含义:将 Bank1 的电流都设置为 8mA,Bank1 控制的是 MIO 引脚 [26:51]。
具体分析:
寄存器部分的操作都是按位计算的,想要什么功能,在手册中查到之后,就需要通过寄存器把想要控制的引脚的地址写入对应功能所需的设置。这里以控制电流为例。因为需要控制的引脚为 MIO40 和 MIO42,40 和 42 都是 bank1 所负责的区域。因此,在寄存器手册中找到对应的bank1 部分。
这个部分的寄存器都是控制 bank1 部分功能的,我们先介绍 ctrl0 和 ctrl1 两个部分。这两个部分是控制电流的。
通过查看“bank1_crtl0 (IOU_SLCR) Register”的“Description”可以知道,ctrl0 得和 ctrl1 两个部分才能控制电流的大小。
我们需要控制的是 MIO40 和 MIO42,这里为了方便起见,将 ctrl0 全部设置为 1,将 ctrl1 全部设置为 0,查表可以知道是 8mA。
- &:按位与,可以理解为交集,交集是指 A 与 B 共有的。
- | :按位或,可以理解为并集,并集是指 A 和 B 所有的。
4:设置引脚是 Schmitt Trigger 还是 CMOS Input
static __init int led_drv_v1_init(void)
{
//4.选择引脚是Schmitt还是CMOS
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL3) & 0x0), IOU_SLCR_BANK1_CTRL3);
return 0;
}
含义:设置 MIO 引脚的功能,这里选择 CMOS Input。
5:设置引脚的上下拉及使能
static __init int led_drv_v1_init(void)
{
//5.输出管脚上下拉及使能
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL4) | 0x3FFFFFF), IOU_SLCR_BANK1_CTRL4);
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL5) | 0x3FFFFFF), IOU_SLCR_BANK1_CTRL5);
return 0;
}
含义:设置 MIO 引脚的上拉或者下拉,并且使能。
具体分析:在 ctrl4 部分将其全部设置为 1,对照寄存器手册可以知道是上拉,并且 bank1 上所有的都是上拉。同理在 ctrl5部分也将其全部设置为 1,对照寄存器手册可以看到 bank1 上 [26:51] 的 MIO 都被启动了。
6:设置 MIO 的速率
static __init int led_drv_v1_init(void)
{
//6.MIO速率的选择
iowrite32((ioread32(IOU_SLCR_BANK1_CTRL6) & 0x0), IOU_SLCR_BANK1_CTRL6);
return 0;
}
Ctrl6 可以设置 MIO 的速率,0 为低速,1 为高速。
7:设置 GPIO 的输入输出
static __init int led_drv_v1_init(void)
{
//7.MIO40 MIO42 设置成输出
// 40 - 26 = 14, 42 - 26 = 16
// 0000 0000 000|0 0|000 0000 0000 0000
// |16 |14
// 0000 0000 000 1 0 100 0000 0000 0000
// 0000 0000 0001 0100 0000 0000 0000
// 0x00014000
iowrite32((ioread32(GPIO_DIRM_1) | 0x00014000), GPIO_DIRM_1);
//MIO40 MIO42 使能
iowrite32((ioread32(GPIO_OEN_1) | 0x00014000), GPIO_OEN_1);
return 0;
}
含义:将 GPIO 的输入输出部分,MIO40 和 MIO42 的地方设置为输出。
具体分析:
这个部分设置的是 GPIO 的相关功能,所以要去“GPIO Module”部分去找相关的寄存器的功能。
- GPIO_DIRM_1:是指控制 GPIO 的输入输出方向。
- GPIO_OEN_1:则是用来控制使能,也就是 enable 的意思。
8:通过 Data 的方式熄灭 LED
static __init int led_drv_v1_init(void)
{
/* MASK_DATA方式按灭LED1,LED2,这个需要自己修改相关参数 */
//iowrite32((ioread32(GPIO_MASK_DATA_0_LSW ) & 0xFEFFEFFF), GPIO_MASK_DATA_0_LSW );
//printk("GPIO_MASK_DATA_0_LSW = 0x%x\n", *GPIO_MASK_DATA_0_LSW);
//iowrite32((ioread32(GPIO_MASK_DATA_0_MSW ) & 0xFEFFEFFF), GPIO_MASK_DATA_0_MSW );
//printk("GPIO_MASK_DATA_0_MSW = 0x%x\n", *GPIO_MASK_DATA_0_MSW);
//8.DATA方式按灭LED1,LED2
// 1111 1111 1111 111|1 1|111 1111 1111 1111
// |0 |0
// 1111 1111 1111 1110 1011 1111 1111 1111
// 0xFFFEBFFF
// iowrite32((ioread32(GPIO_DATA_1) & 0xFFFEBFFF), GPIO_DATA_1);
iowrite32((ioread32(GPIO_DATA_1) & 0xFFFEFFFF), GPIO_DATA_1);
printk("GPIO_DATA_1 = 0x%x\n", *GPIO_DATA_1);
return 0;
}
含义:将已经配置 GPIO 的 MIO40,MIO42 的输出设置为 0,也就是关闭输出。
具体分析:
这是通过控制 DATA 寄存器的方式熄灭 LED,还可以使用 MASK_DATA 的方式熄灭 LED。具体的设置的值需要自己查寄存器手册修改。DATA 寄存器的说明在介绍部分。
9:内存释放
static __init int led_drv_v1_init(void)
{
//9.内存释放
iounmap(gpio_addr);
iounmap(iou_slcr);
return 0;
}
含义:将申请的虚拟内存资源进行释放。
具体分析:
在通过 ioremap 向系统申请了系统资源之后,在驱动退出的时候,需要将资源释放。释放用 iounmap 函数。
1.5交叉编译
- 将led_drv_v1.c和Makefile文件在虚拟机中进行编译。
- 将编译生成的led_drv_v1.ko文件上传到开发板的Linux中。
1.6演示
- 在终端中使用输入“ls”,查看是否存在“led_drv_v1.ko”文件。
- 确认存在之后,进入 root 权限,然后在终端输入“insmod led_drv_v1.ko”。
输入 - “lsmod”查看是否安装成功。
- 在终端输入“rmmod led_drv_v1.ko”,PS的灯会亮起。