适用于板卡型号:
AXU2CGA/AXU2CGB/AXU3EG/AXU4EV-E/AXU4EV-P/AXU5EV-E/AXU5EV-P /AXU9EG/AXU15EG
实验Vivado工程目录为“custom_pwm_ip /vivado”。
实验vitis工程目录为“custom_pwm_ip /vitis”。
Xilinx官方为大家提供了很多IP核,在Vivado的IP Catalog中可以查看这些IP核,用户在构建自己的系统中,不可能只使用Xilinx官方的免费IP核,很多时候需要创建属于自己的用户IP核,创建自己的IP核有很多好处,例如系统设计定制化;设计复用,可以在在IP核中加入license, 有偿提供给别人使用;简化系统设计和缩短设计时间。用ZYNQ系统设计IP核,最常用的就是使用AXI总线将PS同PL部分的IP核连接起来。本实验将为大家介绍如何在Vivado中构建AXI总线类型的IP核,此IP核用来产生一个PWM,用这个控制开发板上的LED,做一个呼吸灯的效果。
FPGA工程师工作内容
以下为FPGA工程师负责内容。
1. PWM介绍
我们经常使用PWM来控制LED,蜂鸣器等,通过调节脉冲的占空比来调节LED的亮度。
在其他开发板中我们使用过的一个pwm模块如下:
//////////////////////////////////////////////////////////////////////////////////
// //
//////////////////////////////////////////////////////////////////////////////////
// //
// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd //
// All rights reserved //
// //
// This source file may be used and distributed without restriction provided //
// that this copyright statement is not removed from the file and that any //
// derivative work contains the original copyright notice and the associated //
// disclaimer. //
// //
//////////////////////////////////////////////////////////////////////////////////
//================================================================================
// Description: pwm model
// pwm out period = frequency(pwm_out) * (2 ** N) / frequency(clk);
//
//================================================================================
// Revision History:
// Date By Revision Change Description
//--------------------------------------------------------------------------------
// 2017/5/3 meisq 1.0 Original
//********************************************************************************/
`timescale1ns/1ps
module ax_pwm
#(
parameter N =32//pwm bit width
)
(
input clk,
input rst,
input[N -1:0]period,
input[N -1:0]duty,
output pwm_out
);
reg[N -1:0] period_r;
reg[N -1:0] duty_r;
reg[N -1:0] period_cnt;
reg pwm_r;
assign pwm_out = pwm_r;
always@(posedge clk orposedge rst)
begin
if(rst==1)
begin
period_r <={ N {1'b0}};
duty_r <={ N {1'b0}};
end
else
begin
period_r <= period;
duty_r <= duty;
end
end
always@(posedge clk orposedge rst)
begin
if(rst==1)
period_cnt <={ N {1'b0}};
else
period_cnt <= period_cnt + period_r;
end
always@(posedge clk orposedge rst)
begin
if(rst==1)
begin
pwm_r <=1'b0;
end
else
begin
if(period_cnt >= duty_r)
pwm_r <=1'b1;
else
pwm_r <=1'b0;
end
end
endmodule
可以看到这个PWM模块需要2个参数“period”、“duty”来控制频率和占空比,”period”为步进值,也就是计数器每个周期要加的值。Duty为占空比的值。我们需要设计一些寄存器来控制这些参数,这里需要使用AXI总线,PS通过AXI总线来读写寄存器。
data:image/s3,"s3://crabby-images/e2aa5/e2aa536f8a7ca2fc39b08c019b82deff6acbfd8e" alt=""
2. Vivado工程建立
用”ps_hello”工程另存为一个名为“custom_pwm_ip”工程
2.1 创建自定义IP
1)点击菜单“Tools->Create and Package IP...”
data:image/s3,"s3://crabby-images/202e5/202e58cfea30c0e1d4cc38ec74a774d4c1a64a5d" alt=""
2)选择“Next”
data:image/s3,"s3://crabby-images/a2ee8/a2ee8e1316301875f582ab3407ce02cf3c4b23d1" alt=""
3)选择创建一个新的AXI4设备
data:image/s3,"s3://crabby-images/f0618/f0618916b5eb7f90a244c45b395d4f8db74f9722" alt=""
4)名称填写“ax_pwm”,描述填写“alinx pwm”,然后选择一个合适的位置用来放IP
data:image/s3,"s3://crabby-images/125cf/125cfa3ab78b3f489a5df13bcbbc288ff70687be" alt=""
5)下面参数可以指定接口类型、寄存器数量等,这里不需要修改,使用AXI Lite Slave接口,4个寄存器。
data:image/s3,"s3://crabby-images/27a02/27a022d6594e76df2c2c5b78a7e472a87210a82c" alt=""
6)点击“Finish”完成IP的创建
data:image/s3,"s3://crabby-images/b0010/b001076c06da0f3950e93187afa6cdc267ed5245" alt=""
7)在“IP Catalog”中可以看到刚才创建的IP
data:image/s3,"s3://crabby-images/cf6c3/cf6c32867c94fd2e2b9e99a8eb0fb25ec7ad6c24" alt=""
8)这个时候的IP只有简单的寄存器读写功能,我们需要修改IP,选择IP,右键“Edit in IP Packager”
data:image/s3,"s3://crabby-images/18598/18598f622f408e03644ea780da91e97c2bb924b6" alt=""
9)这是弹出一个对话框,可以填写工程名称和路径,这里默认,点击“OK”
data:image/s3,"s3://crabby-images/237a4/237a4de2dadbffe8a78dce69e3dabbb944d305a9" alt=""
10)Vivado打开了一个新的工程
data:image/s3,"s3://crabby-images/3c43e/3c43e43831500a00fa2ba7fda9f57a9284e38ecb" alt=""
11)添加PWM功能的核心代码
data:image/s3,"s3://crabby-images/0d644/0d64414d3723f4da9dd6736eedc5a8b6cb5ca1ac" alt=""
12)添加代码时选择复制代码到IP目录
data:image/s3,"s3://crabby-images/854ba/854ba706f3d4ce3028180267e1df42154f0ef99d" alt=""
13)修改“ax_pwm_v1_0.v”,添加一个pwm输出端口
data:image/s3,"s3://crabby-images/ee727/ee72748a194c7018968810b02fb0156ca65a6054" alt=""
14)修改“ax_pwm_v1_0.v”,在例化“ax_pwm_V1_0_S00_AXI”,中添加pwm端口的例化
data:image/s3,"s3://crabby-images/fb1fe/fb1fea3495f06517c6b529f2ded93aab55a432f7" alt=""
15)修改“ax_pwm_v1_0_s00_AXI.v”文件,添加pwm端口,这个文件是实现AXI4 Lite Slave的核心代码
data:image/s3,"s3://crabby-images/63907/639079f9025d7f447d823512ac99e1c32e768a6e" alt=""
16)修改“ax_pwm_v1_0_s00_AXI.v”文件,例化pwm核心功能代码,将寄存器slv_reg0和slv_reg1用于pwm模块的参数控制。
data:image/s3,"s3://crabby-images/ffd36/ffd3677bb97dbf8abfdb14ddb3588fe31166d3b0" alt=""
17)双击“component.xml”文件
data:image/s3,"s3://crabby-images/f48bd/f48bd1c7949553d5e322bd26288834bbaa3c1b6b" alt=""
18)在“File Groups”选项中点击“Merge changers from File Groups Wizard”
data:image/s3,"s3://crabby-images/a153c/a153c01d4edd49d5e3abae23a3ead337b9650808" alt=""
19)在“Customization Parameters”选项中点击“Merge changes form Customization Parameters Wizard”
data:image/s3,"s3://crabby-images/83a74/83a7474f061300d63d4da91281356b719707866f" alt=""
20)点击“Re-Package IP”完成IP的修改
data:image/s3,"s3://crabby-images/1e67b/1e67bdd1d41863a994adad9d0684bd5153cf5fac" alt=""
2.2 添加自定义IP到工程
1)搜索“pwm”,添加“ax_pwm_v1.0”
data:image/s3,"s3://crabby-images/d0808/d0808c97d6c33b8ae94bb63ef62ce174b219d304" alt=""
2)点击“Run Connection Automation”
data:image/s3,"s3://crabby-images/df975/df975d556c62dbdf1c697f56b8d4594c0184c54a" alt=""
3)导出pwm端口
data:image/s3,"s3://crabby-images/a1d3c/a1d3c00b1e5a31e89df5327d2c45a7ec23026859" alt=""
data:image/s3,"s3://crabby-images/c8543/c854300f3cfba0739e6a185ff311f7573aa193ed" alt=""
4)保存设计,并Generate Output Products
data:image/s3,"s3://crabby-images/08c82/08c82bbba32f2daeaa311f2bd1bc54bf99ae03e9" alt=""
5)添加xdc文件分配管脚,把pwm_0输出端口分配给LED1,做一个呼吸灯,编译生成bit文件,导出硬件
软件工程师工作内容
以下为软件工程师负责内容。
3. Vitis软件编写调试
1)启动Vitis,新建APP,模板选择“Hello World”
data:image/s3,"s3://crabby-images/ce1da/ce1dafcfda296d8d5c69c029120abf890e60b33b" alt=""
2)在bsp里找到“xparameters.h”文件,这个非常重要的文件,里面找到了自定IP的寄存器基地址,可以找到自定义IP的基地址。
data:image/s3,"s3://crabby-images/66c3c/66c3c1c6c5f54d3796e6934e617b8444041ac4dd" alt=""
3)有个寄存器读写宏和自定义IP的基地址,我们开始编写代码,测试自定义IP,我们先通过写寄存器AX_PWM_S00_AXI_SLV_REG0_OFFSET,控制PWM输出频率,然后通过写寄存器AX_PWM_S00_AXI_SLV_REG1_OFFSET控制PWM输出的占空比。
#include
#include "platform.h"
#include "xil_printf.h"
#include "ax_pwm.h"
#include "xil_io.h"
#include "xparameters.h"
#include "sleep.h"
unsignedint duty;
int main()
{
init_platform();
print("Hello World\n\r");
//pwm out period = frequency(pwm_out) * (2^N) / frequency(clk);
AX_PWM_mWriteReg(XPAR_AX_PWM_0_S00_AXI_BASEADDR, AX_PWM_S00_AXI_SLV_REG0_OFFSET,17179);//200hz
//duty = (2^N) * (1 - (duty cycle)) - 1
while(1){
for(duty =0x8fffffff; duty <0xffffffff; duty = duty +100000){
AX_PWM_mWriteReg(XPAR_AX_PWM_0_S00_AXI_BASEADDR, AX_PWM_S00_AXI_SLV_REG1_OFFSET, duty);
usleep(100);
}
}
cleanup_platform();
return0;
}
4)通过运行代码,我们可以看到PLLED1呈现出一个呼吸灯的效果。
5)通过debug,我们来查看一下寄存器
data:image/s3,"s3://crabby-images/3d170/3d17088976515c949dc42da3adbabd1b9559dfc6" alt=""
6)进入debug状态,按“F6”可以单步运行。
data:image/s3,"s3://crabby-images/02ad5/02ad54ec4faafcab690705d8d4afa66f08aeef55" alt=""
7)通过菜单可以查看“Memory”窗口
data:image/s3,"s3://crabby-images/34d2e/34d2e4af4a42c051aa8f0d3d390f91c7fc89914b" alt=""
8)添加一个监视地址“0x80000000”
data:image/s3,"s3://crabby-images/e44d8/e44d81e143fcf71855d47e7e8e62e0087cc908d2" alt=""
data:image/s3,"s3://crabby-images/e37c9/e37c99a39e1708a8216ee8c932ddb3c2a83c83ce" alt=""
9)单步运行,观察变化
data:image/s3,"s3://crabby-images/ac306/ac306fad9d4dc21631fdc1678d1800362242f739" alt=""
4. 实验总结
通过本实验我们掌握了更多的Vitis调试技巧,掌握了ARM + FPGA开发的核心内容,就是ARM和FPGA数据交互。