• 【Xilinx AX7103 MicroBalze学习笔记6】MicroBlaze 自定义 IP 核封装实验


    目录

    实验任务

    实验框图

    创建自定义 IP

    封装 IP

    IP 封装界面配置

    硬件设计(Vivado部分)

    Block Design搭建

    添加 IP 库

    约束文件

    软件设计(SDK部分)


    实验任务

    本节介绍基于 MicroBlaze 的自定义 IP 核封装实验,实验任务是通过自定义一个呼吸灯 IP 核,来控制 LED 呈现呼吸灯的效果,并且可以通过 AXI 接口来控制呼吸灯的开关和呼吸的频率。

    实验框图

    实验框图比较简单,框图中的 UART 用于打印信息, Breath LED IP 核为自定义的 IP 核, McroBlaze 处理器通过 AXI 接口为LED IP 模块发送配置数据,从而来控制 LED 灯

    创建自定义 IP

    创建一个新工程,添加呼吸灯代码文件

    代码如下:

    1. `timescale 1ns / 1ps
    2. //
    3. // Company:
    4. // Engineer:
    5. // Create Date: 2022/06/10 19:50:37
    6. // Design Name:
    7. // Module Name: breath_led_ip
    8. // Project Name:
    9. //
    10. module breath_led_ip(
    11. input sys_clk ,
    12. input sys_rst_n ,
    13. input sw_ctrl ,
    14. input set_en ,
    15. input [9:0] set_freq_step ,
    16. output led
    17. );
    18. //*****************************************************
    19. //** main code
    20. //*****************************************************
    21. //parameter define
    22. parameter START_FREQ_STEP = 10'd100;
    23. //reg define
    24. reg [15:0] period_cnt ;
    25. reg [9:0] freq_step ;
    26. reg [15:0] duty_cycle ;
    27. reg inc_dec_flag;
    28. wire led_t ;
    29. assign led_t = ( period_cnt <= duty_cycle ) ? 1'b1 : 1'b0 ;
    30. assign led = led_t & sw_ctrl;
    31. always @ (posedge sys_clk) begin
    32. if (!sys_rst_n)
    33. period_cnt <= 16'd0;
    34. else if(!sw_ctrl)
    35. period_cnt <= 16'd0;
    36. else if( period_cnt == 16'd50_000 )
    37. period_cnt <= 16'd0;
    38. else
    39. period_cnt <= period_cnt + 16'd1;
    40. end
    41. always @(posedge sys_clk) begin
    42. if(!sys_rst_n)
    43. freq_step <= START_FREQ_STEP;
    44. else if(set_en) begin
    45. if(set_freq_step == 0)
    46. freq_step <= 10'd1;
    47. else if(set_freq_step >= 10'd1_000)
    48. freq_step <= 10'd1_000;
    49. else
    50. freq_step <= set_freq_step;
    51. end
    52. end
    53. always @(posedge sys_clk) begin
    54. if (sys_rst_n == 1'b0) begin
    55. duty_cycle <= 16'd0;
    56. inc_dec_flag <= 1'b0;
    57. end
    58. else if(!sw_ctrl) begin
    59. duty_cycle <= 16'd0;
    60. inc_dec_flag <= 1'b0;
    61. end
    62. else if( period_cnt == 16'd50_000 ) begin
    63. if( inc_dec_flag ) begin
    64. if( duty_cycle == 16'd0 )
    65. inc_dec_flag <= 1'b0;
    66. else if(duty_cycle < freq_step)
    67. duty_cycle <= 16'd0;
    68. else
    69. duty_cycle <= duty_cycle - freq_step;
    70. end
    71. else begin
    72. if( duty_cycle >= 16'd50_000 )
    73. inc_dec_flag <= 1'b1;
    74. else
    75. duty_cycle <= duty_cycle + freq_step;
    76. end
    77. end
    78. else
    79. duty_cycle <= duty_cycle ;
    80. end
    81. endmodule

    封装 IP

    点击菜单栏中的 Tools 的创建和封装新 IP,

    选择创建带有 AXI4 接口的

    下一个配置页中的一些命名以及文件存储地址可以自定义。

    下一个的界面内容保持默认即可,可以根据自己的需要更改,这里保持默认,因为 MB 通过AXI总线配置此模块,因此作为从机slave。

    最后界面中的 Next Steps 中的选项可以根据需要选择,这里直接保持默认,最后点击finish即可。

    接下来开始设置 IP 封装,将界面切换至 Package IP,如果不小心关闭的话,可以通过 IP-XACT 界面下的 component.xml 重新打开

    IP 封装界面配置

    Identification 这一栏的选项直接保持默认,需要注意的是,我们可以点击图 Categories 选项下的“+”按钮来修改 IP 的分类,这里不做修改。

    点击“Compatibility” ,修改该 IP 核支持的器件。点击“Family” 一栏下的“+”图标,选择“Add FamilyExplicitly…”,可以选择你所需要适配的器件型号,尽可能的选多一些的器件。

    这里勾选“artix7(artix-7)”,表示该 IP 核支持 artix7 器件。而 Life-cycle 表明该 IP 核当前的产品生命周期,这里选择“Pre-Production”。

    点击“File Groups” ,然后点击界面上的“Merge Changes from Gile Groups Wizard”,如图所示:

    此时可以在 Verilog Synthesis 一栏中查看工程中的两个模块。

    点击“Customization Parameters”,点击界面上的“Merge Changes from Customization Parameters Wizard”,如图所示:

    此时多了 Hidden Parameters 一栏,展开这个界面,可以看到程序中自定义的参数 START_FREQ_STEP,右击这个参数,选择“Edit Parameter…”,弹出编辑参数的界面,如图所示:

    在弹出的页面中勾选“Visible in Customization GUI”, 将此参数显示在 GUI 参数界面中;Format 格式改为“long”;勾选“Specify Range”来设定此参数的范围。将 Type 改为“Range of integers”, Minimum 的值改为“1”,Maximum 的值改为“1000” ,将 Default Value 的值改为“100” ,点击“OK”按钮,如图所示:

    点击“Customization GUI”,可以在“Layout”界面拖动 Page 0 下的参数来调整参数在 GUI 显示的位置,如图所示:

    最后点击re-package IP即可。

    硬件设计(Vivado部分)

    Block Design搭建

    打开第一个 hello world 实验工程,将其另存为新的工程,命名为 breath_led 然后开始 BD 的搭建。

    添加 IP 库

    点击左侧菜单栏的 setting

    点击 IP 中的 repository 的加号提娜佳刚刚创建的 IP 至此 IP 库,点击 ok 即可。

    这是点击左侧菜单栏的 IP catalog ,在 User repository 中就可以看到刚刚添加用户自定义的 IP

    打开 block design 添加搜索 IP 并添加,这时 IP 就成功显示在 BD 上了。

    点开此 IP 就可以看到自定的参数是可以在配置界面中随意配置的。

    最后点击自动连线并将输出 led 的管脚引出,效果图如下:

    约束文件

    添加约束文件最后直接生成比特流即可。

    软件设计(SDK部分)

    进入到 SDK 的开发界面中,新建一个工程文件,命名为 breath_led,并在工程中新建一个设计文件命名为main.c。

    1. #include "stdio.h"
    2. #include "xparameters.h"
    3. #include "xil_printf.h"
    4. #include "breath_led_ip.h"
    5. #include "xil_io.h"
    6. #include "sleep.h"
    7. #define LED_IP_BASEADDR XPAR_BREATH_LED_IP_0_S0_AXI_BASEADDR //LED IP 基地址
    8. #define LED_IP_REG0 BREATH_LED_IP_S0_AXI_SLV_REG0_OFFSET //LED IP 寄存器地址 0
    9. #define LED_IP_REG1 BREATH_LED_IP_S0_AXI_SLV_REG1_OFFSET //LED IP 寄存器地址 1
    10. //main 函数
    11. int main()
    12. {
    13. int freq_flag; //定义频率状态,用于循环改变呼吸灯的呼吸频率
    14. int led_state; //定义 LED 灯的状态
    15. xil_printf("LED User IP Test!\n");
    16. while(1){
    17. //根据 freq_flag 的标志位,切换呼吸灯的频率
    18. if(freq_flag == 0){
    19. BREATH_LED_IP_mWriteReg(LED_IP_BASEADDR,LED_IP_REG1,0x800000ef);
    20. freq_flag = 1;
    21. }
    22. else{
    23. BREATH_LED_IP_mWriteReg(LED_IP_BASEADDR,LED_IP_REG1,0x8000002f);
    24. freq_flag = 0;
    25. }
    26. //获取 LED 当前开关状态 1:打开 0:关闭
    27. led_state = BREATH_LED_IP_mReadReg(LED_IP_BASEADDR,LED_IP_REG0);
    28. //如果开关关闭,打开呼吸灯
    29. if(led_state == 0){
    30. BREATH_LED_IP_mWriteReg (LED_IP_BASEADDR, LED_IP_REG0, 1);
    31. xil_printf("Breath LED ON\n");
    32. }
    33. sleep(5);
    34. //获取 LED 当前开关状态 1:打开 0:关闭
    35. led_state = BREATH_LED_IP_mReadReg(LED_IP_BASEADDR,LED_IP_REG0);
    36. //如果开关打开,关闭呼吸灯
    37. if(led_state == 1){
    38. BREATH_LED_IP_mWriteReg (LED_IP_BASEADDR, LED_IP_REG0, 0);
    39. xil_printf("Breath LED OFF\n");
    40. }
    41. sleep(1);
    42. }
    43. }
    44. //代码来自正点原子

    最后点击 run,就可以看到板子的 led 等呈现呼吸灯的效果

    同样的在串口助手上也可以成功看到 led 灯的实时状态

  • 相关阅读:
    支付宝支付前端如何显示
    Sora 的工作原理(及其意义) [译]
    TCP/IP协议群
    c++中的模板(8) -- 模板和友元函数
    android java读写yaml文件
    音视频项目—基于FFmpeg和SDL的音视频播放器解析(十三)
    Tensorflow2.0:CNN、ResNet实现MNIST分类识别
    想要精通算法和SQL的成长之路 - 分割数组的最大值
    美团加大了对AI大模型领域的投入!
    pdm使用经验
  • 原文地址:https://blog.csdn.net/m0_61298445/article/details/125352404