当前位置:   article > 正文

FPGA(主)与STM32(从)SPI通信

FPGA(主)与STM32(从)SPI通信

一、概述

        主要实现了FPGA向STM32快速发送数据(比较稳定),至于STM32发送过来的数据,大概率还是处于丢位状态。但因为我主要是要完成一个FPGA这边持续向STM32的发送,所以我也没仔细去研究如何实现双方通信更稳定(同时本人能力也有限)。

        至于SPI的通信时序,有四种模式,我选择的是空闲状态为高电平,采集在上升沿的一种模式。具体的时序可以参考其他博客,我自己的理解(以我选择的这种模式)就是在sck为低电平的时候寄存数据,然后一直保持数据到下一次sck为低电平。获取数据的话在sck拉高的瞬间,也就是sck的上升沿获取数据。

        STM32端我使用的是SPI5加DMA的方式来进行数据的收发。

        下面贴出FPGA和STM32的代码


二、FPGA端代码

spi主模式

  1. module spi_master(
  2. input clk_1m, //时钟信号 1m
  3. input rst_n, //按键复位信号
  4. input ena_mo, //模块使能
  5. input [7:0]spi_tdata, //spi要发送的数据
  6. input spi_miso, //spi 主进从出
  7. output reg spi_mosi, //spi 从进主出
  8. output reg spi_sck, //spi 时钟信号
  9. output reg spi_nss, //spi 片选信号
  10. output reg [7:0]spi_rdata, //spi接收到的数据
  11. output tr_done //spi 完成一个字节发送
  12. );
  13. localparam waiten=0; //等待模块使能
  14. localparam tx_revl=1; //sck低电平
  15. localparam tx_revh=2; //sck高电平
  16. localparam done=3; //完成一次发送
  17. localparam DELAY = 5; //完成时等待5个时钟开始下一次发送
  18. //reg define
  19. reg [3:0] us_cnt; //us计数器
  20. reg us_cnt_clr; //计数器清零信号
  21. reg[1:0] state,next_state; //当前状态和下一个状态
  22. reg[3:0] w_cnt; //位计数器
  23. //1微秒计数器
  24. always @ (posedge clk_1m,negedge rst_n) begin
  25. if (!rst_n)
  26. us_cnt <= 21'd0;
  27. else if (us_cnt_clr)
  28. us_cnt <= 21'd0;
  29. else
  30. us_cnt <= us_cnt + 1'b1;
  31. end
  32. //下一个状态确认
  33. always @(*) begin
  34. if(!rst_n)
  35. next_state = waiten;
  36. case(state)
  37. //模块使能且计数达到延时则进入下一个状态
  38. waiten: next_state = (ena_mo&&us_cnt==DELAY) ? tx_revl : waiten;
  39. tx_revl: next_state = tx_revh;
  40. tx_revh: next_state = (w_cnt == 4'd8) ? done : tx_revl;
  41. done: next_state = (ena_mo&&us_cnt==DELAY) ? tx_revl : waiten;
  42. endcase
  43. end
  44. //状态的各变量赋值
  45. always @(posedge clk_1m,negedge rst_n)begin
  46. if(!rst_n)begin
  47. spi_sck <= 1'b1; //sck空闲状态为高电平
  48. spi_mosi <= 1'b0;
  49. spi_rdata <= 8'd0;
  50. us_cnt_clr <= 1'b1;
  51. spi_nss <= 1'b1;
  52. end
  53. else begin
  54. case(state)
  55. waiten:begin
  56. spi_sck <= 1'b1;
  57. spi_mosi <= 1'b0;
  58. us_cnt_clr <= 1'b0;
  59. spi_nss <= us_cnt > 3'd3 ? 1'b0 : 1'b1; //nss信号在等待时提前拉低 选中 做好发送准备
  60. end
  61. tx_revl:begin
  62. us_cnt_clr <= 1'b1;
  63. spi_mosi <= spi_tdata[3'd7-w_cnt]; //sck低电平时寄存数据
  64. spi_sck <= 1'b0; //sck拉低
  65. end
  66. tx_revh:begin
  67. spi_sck <= 1'b1; //sck拉高 数据值保持不变
  68. spi_rdata[3'd7-(w_cnt-1'b1)] <= spi_miso; //在上升沿捕获数据值
  69. end
  70. done:begin
  71. us_cnt_clr <= 1'b0; //发送完成 时钟计数器关闭复位,延时等待一下
  72. spi_sck <= 1'b1;
  73. spi_mosi <= 1'b0;
  74. spi_nss <= us_cnt > 3'd3 ? 1'b0 : 1'b1; //nss信号在等待时提前拉低 选中 做好发送准备
  75. end
  76. endcase
  77. end
  78. end
  79. //状态转换
  80. always @(posedge clk_1m,negedge rst_n)begin
  81. if(!rst_n)
  82. state <= waiten;
  83. else
  84. state <= next_state;
  85. end
  86. //位计数器赋值
  87. always @(posedge clk_1m,negedge rst_n)begin
  88. if(!rst_n)begin
  89. w_cnt <= 4'd0;
  90. end
  91. else begin
  92. case(state)
  93. tx_revl:w_cnt <= w_cnt + 1'b1;//sck低电平时位计数器加1,所以高电平时就会多一个值,需要减1
  94. waiten: w_cnt <= 4'd0;
  95. done: w_cnt <= 4'd0;
  96. default: w_cnt <= w_cnt;
  97. endcase
  98. end
  99. end
  100. //完成信号输出赋值
  101. assign tr_done = (state == done);
  102. endmodule

testbench仿真代码

  1. `timescale 1ns/1ns //仿真单位为1ns,精度为1ns
  2. module spi_master_tb();
  3. reg clk_1m;
  4. reg rst_n;
  5. reg ena_mo;
  6. reg [7:0] spi_tdata;
  7. reg spi_miso;
  8. wire spi_mosi;
  9. wire spi_sck;
  10. wire spi_nss;
  11. wire [7:0] spi_rdata;
  12. spi_master u_spi_master(
  13. .clk_1m(clk_1m),
  14. .rst_n(rst_n),
  15. .ena_mo(ena_mo),
  16. .spi_tdata(spi_tdata),
  17. .spi_miso(spi_miso),
  18. .spi_mosi(spi_mosi),
  19. .spi_sck(spi_sck),
  20. .spi_nss(spi_nss),
  21. .spi_rdata(spi_rdata)
  22. );
  23. initial begin
  24. #0 clk_1m = 0;
  25. rst_n = 0;
  26. ena_mo = 1;
  27. spi_tdata = 8'b11001010;
  28. spi_miso = 0;
  29. #20 rst_n = 1;
  30. end
  31. always #5 clk_1m = ~clk_1m;
  32. always #32 spi_miso = ~spi_miso;
  33. endmodule

仿真结果


 顶层模块

  1. module oscilloscope(
  2. input sys_clk, //系统时钟 50m
  3. input rst_n, //按键复位
  4. input spi_miso, //spi 主进从出信号
  5. output spi_mosi, //spi 主出从进信号
  6. output spi_sck, //spi 时钟信号
  7. output spi_nss //spi 片选信号(使用这个信号可以提高传输的稳定性)
  8. );
  9. wire [7:0] spi_rdata; //spi读取到的数据
  10. wire [7:0] spi_tdata; //spi要发送的数据
  11. wire tr_done; //一个字节发送完成信号
  12. wire clk_1m; //1M 时钟信号
  13. reg [7:0] tx_data; //给要发送的数据赋值
  14. assign spi_tdata = tx_data;//要发送的数据赋值
  15. always @(posedge clk_1m,negedge rst_n)begin
  16. if(!rst_n)
  17. tx_data <= 8'd0;
  18. else begin
  19. tx_data <= tr_done ? tx_data + 1'b1 : tx_data;//每发送完成一个字节 数据的值增加1
  20. end
  21. end
  22. //时钟分频模块 产生1MHz时钟
  23. clk_fenpin u_clk_fenpin(
  24. .sys_clk(sys_clk),
  25. .rst_n(rst_n),
  26. .clk_1m(clk_1m)
  27. );
  28. //spi 主模式模块
  29. spi_master u_spi_master(
  30. .clk_1m(clk_1m),
  31. .rst_n(rst_n),
  32. .ena_mo(1'b1),
  33. .spi_tdata(tx_data),
  34. .spi_miso(spi_miso),
  35. .spi_mosi(spi_mosi),
  36. .spi_sck(spi_sck),
  37. .spi_nss(spi_nss),
  38. .spi_rdata(spi_rdata),
  39. .tr_done(tr_done)
  40. );
  41. endmodule

时钟分频模块

  1. module clk_fenpin(
  2. input sys_clk,
  3. input rst_n,
  4. output reg clk_1m
  5. );
  6. reg [25:0] clk_cnt ; //分频计数器
  7. //得到1Mhz分频时钟
  8. always @ (posedge sys_clk or negedge rst_n) begin
  9. if (!rst_n) begin
  10. clk_cnt <= 5'd0;
  11. clk_1m <= 1'b0;
  12. end
  13. else if (clk_cnt < 26'd24)
  14. clk_cnt <= clk_cnt + 1'b1;
  15. else begin
  16. clk_cnt <= 5'd0;
  17. clk_1m <= ~ clk_1m;
  18. end
  19. end
  20. endmodule

三、STM32断配置

        32的配置主要是NSS信号不能再由软件来控制,不然接收的数据很不稳定,每一次复位都会看到不一样的值,可能是因为32端对起始信号的判断不是很对,所以导致总是,丢位。虽然也有概率正确接收,但是极其不稳定,所以将NSS信号交由硬件控制。FPGA作为主机,在每次将要发送前,先将NSS信号拉低,选中32,然后发送完成一个字节后又将NSS信号拉高,这样传输的数据会稳定很多。

spi配置

  1. u8 spi_revdata[spirv_max_len]; //接收数据数组
  2. SPI_HandleTypeDef SPI5_Handler; //SPI句柄
  3. u8 tx = 15; //发送的数据
  4. //以下是SPI模块的初始化代码,配置成主机模式
  5. //SPI口初始化
  6. //这里针是对SPI5的初始化
  7. void SPI5_Init(void)
  8. {
  9. SPI5_Handler.Instance=SPI5; //SP5
  10. SPI5_Handler.Init.Mode=SPI_MODE_SLAVE; //设置SPI工作模式,设置为从模式
  11. SPI5_Handler.Init.Direction=SPI_DIRECTION_2LINES; //设置SPI单向或者双向的数据模式:SPI设置为双线模式
  12. SPI5_Handler.Init.DataSize=SPI_DATASIZE_8BIT; //设置SPI的数据大小:SPI发送接收8位帧结构
  13. SPI5_Handler.Init.CLKPolarity=SPI_POLARITY_HIGH; //串行同步时钟的空闲状态为高电平
  14. SPI5_Handler.Init.CLKPhase=SPI_PHASE_2EDGE; //串行同步时钟的第二个跳变沿(上升或下降)数据被采样
  15. SPI5_Handler.Init.NSS=SPI_NSS_HARD_INPUT; //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理 选择硬件管理
  16. SPI5_Handler.Init.BaudRatePrescaler=SPI_BAUDRATEPRESCALER_4;//定义波特率预分频的值:波特率预分频值为4
  17. SPI5_Handler.Init.FirstBit=SPI_FIRSTBIT_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
  18. SPI5_Handler.Init.TIMode=SPI_TIMODE_DISABLE; //关闭TI模式
  19. SPI5_Handler.Init.CRCCalculation=SPI_CRCCALCULATION_DISABLE;//关闭硬件CRC校验
  20. SPI5_Handler.Init.CRCPolynomial=7; //CRC值计算的多项式
  21. HAL_SPI_Init(&SPI5_Handler);//初始化
  22. __HAL_SPI_ENABLE(&SPI5_Handler); //使能SPI5
  23. DMA2_Init(); //DMA初始化
  24. //开启SPI的DMA传输 1024个字节
  25. HAL_SPI_TransmitReceive_DMA(&SPI5_Handler,&tx,spi_revdata,spirv_max_len);
  26. //发送和接收如果同时使用的话最好使用上面那个函数,不然开启发送,再开接收可能导致没有开启成功
  27. // HAL_SPI_Transmit_DMA(&SPI5_Handler,&tx,1024);
  28. // HAL_SPI_Receive_DMA(&SPI5_Handler,spi_revdata,spirv_max_len);
  29. }
  30. //SPI5底层驱动,时钟使能,引脚配置
  31. //此函数会被HAL_SPI_Init()调用
  32. //hspi:SPI句柄
  33. void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi)
  34. {
  35. GPIO_InitTypeDef GPIO_Initure;
  36. __HAL_RCC_GPIOF_CLK_ENABLE(); //使能GPIOF时钟
  37. __HAL_RCC_SPI5_CLK_ENABLE(); //使能SPI5时钟
  38. //PF6,7,8,9
  39. GPIO_Initure.Pin=GPIO_PIN_6|GPIO_PIN_7|GPIO_PIN_8|GPIO_PIN_9;
  40. GPIO_Initure.Mode=GPIO_MODE_AF_PP; //复用推挽输出
  41. GPIO_Initure.Pull=GPIO_NOPULL; //上拉
  42. GPIO_Initure.Speed=GPIO_SPEED_FAST; //快速
  43. GPIO_Initure.Alternate=GPIO_AF5_SPI5; //复用为SPI5
  44. HAL_GPIO_Init(GPIOF,&GPIO_Initure);
  45. }

dma配置

  1. DMA_HandleTypeDef DMA_SPI5TX_Handler; //SPI5 发送DMA句柄
  2. DMA_HandleTypeDef DMA_SPI5RX_Handler; //SPI5 接收DMA句柄
  3. void DMA2_Init(void)
  4. {
  5. __HAL_RCC_DMA2_CLK_ENABLE(); //使能DMA2的时钟
  6. DMA_SPI5RX_Handler.Instance = DMA2_Stream3; //数据流选择
  7. DMA_SPI5RX_Handler.Init.Channel = DMA_CHANNEL_2; //通道选择
  8. DMA_SPI5RX_Handler.Init.Direction = DMA_PERIPH_TO_MEMORY; //外设到存储器
  9. DMA_SPI5RX_Handler.Init.PeriphInc = DMA_PINC_DISABLE; //外设地址非增量模式
  10. DMA_SPI5RX_Handler.Init.MemInc = DMA_MINC_ENABLE; //存储器地址增量模式
  11. DMA_SPI5RX_Handler.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE; //外设数据长度:8
  12. DMA_SPI5RX_Handler.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE; //存储器数据长度:8
  13. DMA_SPI5RX_Handler.Init.Mode = DMA_CIRCULAR; //DMA循环模式 一次开启 一直搬运
  14. DMA_SPI5RX_Handler.Init.Priority = DMA_PRIORITY_HIGH; //高优先级
  15. DMA_SPI5RX_Handler.Init.FIFOMode = DMA_FIFOMODE_DISABLE; //不使用FIFO模式
  16. HAL_DMA_DeInit(&DMA_SPI5RX_Handler); //清除配置
  17. HAL_DMA_Init(&DMA_SPI5RX_Handler); //初始化配置
  18. __HAL_LINKDMA(&SPI5_Handler,hdmarx,DMA_SPI5RX_Handler); //将SPI的发送和DMA联系起来
  19. DMA_SPI5TX_Handler.Instance = DMA2_Stream4; //数据流选择
  20. DMA_SPI5TX_Handler.Init.Channel = DMA_CHANNEL_2; //通道选择
  21. DMA_SPI5TX_Handler.Init.Direction = DMA_MEMORY_TO_PERIPH; //存储器到外设
  22. DMA_SPI5TX_Handler.Init.PeriphInc = DMA_PINC_DISABLE; //外设地址非增量模式
  23. DMA_SPI5TX_Handler.Init.MemInc = DMA_MINC_DISABLE; //存储器地址非增量模式
  24. DMA_SPI5TX_Handler.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE; //外设数据长度:8
  25. DMA_SPI5TX_Handler.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE; //存储器数据长度:8
  26. DMA_SPI5TX_Handler.Init.Mode = DMA_CIRCULAR; //DMA循环模式 一次开启 一直搬运
  27. DMA_SPI5TX_Handler.Init.Priority = DMA_PRIORITY_MEDIUM; //中等优先级
  28. DMA_SPI5TX_Handler.Init.FIFOMode = DMA_FIFOMODE_DISABLE; //不使用FIFO模式
  29. HAL_DMA_DeInit(&DMA_SPI5TX_Handler); //清除配置
  30. HAL_DMA_Init(&DMA_SPI5TX_Handler); //初始化配置
  31. __HAL_LINKDMA(&SPI5_Handler,hdmatx,DMA_SPI5TX_Handler); //将SPI的发送和DMA联系起来
  32. }

四、实现效果

        stm32接收到的数据,打印的是接收数组中的值(此时的发送频率应该在几十KHz左右,更高的没有测试过,但是改改时钟分频看看就好)

        FPGA收到的数据非常不稳定,很小概率会出现正确的值,就不贴图了

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/空白诗007/article/detail/972281
推荐阅读
相关标签
  

闽ICP备14008679号