STM32与FPGA进行SPI通信

Posted cnlntr

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了STM32与FPGA进行SPI通信相关的知识,希望对你有一定的参考价值。

一、器件

32单片机:STM32F407ZG

FPGA     :EP4CE6E22C8N

二、通信方式

STM32作为主机(软件);

FPGA作为从机;

SPI通信方式为0;

三、STM32源代码

技术图片
 1 #include "delay.h" 
 2 #include "stm32f4xx.h"
 3 
 4 #ifndef __SPI_H
 5 #define __SPI_H        
 6 
 7 #define    SPI1_SCK        PBout(2)
 8 #define    SPI1_MOSI        PBout(3)
 9 #define SPI1_MISO   PBin(4)
10 #define    CS                    PBout(5)
11 
12 //CPOL=0,CPHA=0
13 u8 SOFT_SPI_RW(u8 byte);
14 //SPI初始化
15 void SPIInit(void);
16 
17 #endif
spi.h
技术图片
 1 #include "spi.h"
 2 #include "delay.h" 
 3 #include "stm32f4xx.h"
 4 
 5 //CPOL=0,CPHA=0
 6 u8 SOFT_SPI_RW(u8 byte) 
 7 { 
 8     u8 i;
 9     u8 Temp=0;                                                //接收数据存储
10     SPI1_SCK = 0;
11     delay_init(168);                                    //初始化延时函数
12     for(i=0;i<8;i++)                                // 循环8次 
13     {     
14         if(byte&0x80) SPI1_MOSI = 1;      //若字节最高位为1,则输出高
15         else SPI1_MOSI = 0;                //若字节最高位为0,则输出低 
16         byte <<= 1;                                        //低一位移位到最高位 
17         delay_us(1);                                        //延时1us
18         SPI1_SCK = 1;                                    //拉低时钟 
19         Temp <<= 1;                                        //数据左移 
20         if(SPI1_MISO) Temp++;                        //若从从机接收到高电平,数据自加一 
21         delay_us(1);                                        //延时1us
22         SPI1_SCK = 0;                                        //拉低时钟    
23     } 
24     return (Temp);                                        //返回数据 
25 }
26 
27 void SPIInit(void)
28 {         
29     GPIO_InitTypeDef GPIO_InitStructure;
30     RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB,ENABLE);//使能PORTB时钟 
31     
32     //PB2,PB3,PB5设置
33   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3 | GPIO_Pin_5;
34   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
35   GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
36   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
37   GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
38     GPIO_Init(GPIOB,&GPIO_InitStructure);
39     //PB4设置
40     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
41   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;
42   GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
43   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
44   GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
45     GPIO_Init(GPIOB,&GPIO_InitStructure);
46 }
spi.c

四、FPGA源代码

技术图片
  1 //FPGA作为从设备
  2 module spi_s(I_clk,I_rst_n,I_data_in,O_data_out,O_tx_done,O_rx_done,I_spi_miso,I_spi_sck,I_spi_cs,O_spi_mosi,sck_posedge,sck_negedge,rxd_flag,txd_flag);
  3 input I_clk;                //全局时钟50MHz
  4 input I_rst_n;                //复位信号,低电平有效
  5 //input I_rx_en;                //读(接收)使能信号
  6 //input I_tx_en;                //写(发送)使能信号
  7 input [7:0]I_data_in;    //要发送的数据
  8 output[7:0]O_data_out;    //接收到的数据
  9 output O_tx_done;            //发送一个数据完毕标志位
 10 output O_rx_done;            //接收一个字节完毕标志位
 11 
 12 //四线标准SPI信号定义
 13 input I_spi_miso;            //SPI串行输入,用来接收从机的数据
 14 input I_spi_sck;            //SPI时钟
 15 input I_spi_cs;            //SPI片选信号
 16 output O_spi_mosi;        //SPI输出,用来给从机发送数据
 17 
 18 output sck_posedge,sck_negedge;
 19 output rxd_flag;//接收标志位
 20 output txd_flag;//发送标志位
 21 
 22 reg [7:0] O_data_out;
 23 reg O_tx_done;
 24 reg O_rx_done;
 25 reg O_spi_mosi;
 26 
 27 reg [2:0] R_tx_state;
 28 reg [2:0] R_rx_state;
 29 
 30 reg sck_r0,sck_r1;//当前SCK态,之前SCK态
 31 wire sck_posedge;    //SCK上升沿
 32 wire sck_negedge;    //SCK下降沿
 33 
 34 //获取SCK时钟当前态以及之前态
 35 always @(posedge I_clk,negedge I_rst_n)
 36 begin
 37     if(!I_rst_n)
 38     begin
 39         sck_r0 <= 1b0;
 40         sck_r1 <= 1b0;//工作方式0,空闲时为低电平
 41     end
 42     else
 43     begin
 44         sck_r0 <= I_spi_sck;    //当前SCK态
 45         sck_r1 <= sck_r0;        //之前SCK态
 46     end
 47 end
 48 
 49 //捕获SCK时钟上升沿以及下降沿
 50 assign sck_posedge = (~sck_r0 & sck_r1)? 1b1:1b0;//上升沿
 51 assign sck_negedge = (~sck_r1 & sck_r0)? 1b1:1b0;//下降沿
 52 
 53 //发送数据触发
 54 always @(posedge I_clk,negedge I_rst_n)
 55 begin
 56     //置位
 57     if(!I_rst_n)
 58     begin
 59             R_tx_state <= 3b0;
 60             O_spi_mosi <= 1b0;
 61             O_tx_done <= 0;
 62     end
 63     //SCK上跳沿时发送数据(方式0)
 64     else if(sck_posedge && !I_spi_cs)
 65     begin
 66         case(R_tx_state)
 67             3d0://发送第七位
 68             begin
 69                 O_spi_mosi <= I_data_in[7];
 70                 R_tx_state <= R_tx_state + 1b1;
 71                 O_tx_done <= 1b0;
 72             end
 73             3d1://发送第六位
 74             begin
 75                 O_spi_mosi <= I_data_in[6];
 76                 R_tx_state <= R_tx_state + 1b1;
 77                 O_tx_done <= 1b0;
 78             end
 79             3d2://发送第五位
 80             begin
 81                 O_spi_mosi <= I_data_in[5];
 82                 R_tx_state <= R_tx_state + 1b1;
 83                 O_tx_done <= 1b0;
 84             end
 85             3d3://发送第四位
 86             begin
 87                 O_spi_mosi <= I_data_in[4];
 88                 R_tx_state <= R_tx_state + 1b1;
 89                 O_tx_done <= 1b0;
 90             end
 91             3d4://发送第三位
 92             begin
 93                 O_spi_mosi <= I_data_in[3];
 94                 R_tx_state <= R_tx_state + 1b1;
 95                 O_tx_done <= 1b0;
 96             end
 97             3d5://发送第二位
 98             begin
 99                 O_spi_mosi <= I_data_in[2];
100                 R_tx_state <= R_tx_state + 1b1;
101                 O_tx_done <= 1b0;
102             end
103             3d6://发送第一位
104             begin
105                 O_spi_mosi <= I_data_in[1];
106                 R_tx_state <= R_tx_state + 1b1;
107                 O_tx_done <= 1b0;
108             end
109             3d7://发送第零位
110             begin
111                 O_spi_mosi <= I_data_in[0];
112                 R_tx_state <= R_tx_state + 1b1;
113                 O_tx_done <= 1b1;//发送完毕
114             end
115             default:R_tx_state <= 3d0;
116         endcase
117     end
118 end
119 
120 
121 
122 //接收数据触发
123 always @(posedge I_clk,negedge I_rst_n)
124 begin
125     if(!I_rst_n)
126     begin
127         O_data_out <= 8b0;
128         R_rx_state <= 3b0;
129         O_rx_done <= 1b0;
130     end
131     else if(sck_negedge && !I_spi_cs)
132         case(R_rx_state)
133             3d0://接收第七位
134             begin
135                 R_rx_state <= R_rx_state + 1b1;
136                 O_rx_done <= 1b0;
137                 O_data_out[7] <= I_spi_miso;
138             end
139             3d1://接收第六位
140             begin
141                 R_rx_state <= R_rx_state + 1b1;
142                 O_rx_done <= 1b0;
143                 O_data_out[6] <= I_spi_miso;
144             end
145             3d2://接收第五位
146             begin
147                 R_rx_state <= R_rx_state + 1b1;
148                 O_rx_done <= 1b0;
149                 O_data_out[5] <= I_spi_miso;
150             end
151             3d3://接收第四位
152             begin
153                 R_rx_state <= R_rx_state + 1b1;
154                 O_rx_done <= 1b0;
155                 O_data_out[4] <= I_spi_miso;
156             end
157             3d4://接收第三位
158             begin
159                 R_rx_state <= R_rx_state + 1b1;
160                 O_rx_done <= 1b0;
161                 O_data_out[3] <= I_spi_miso;
162             end
163             3d5://接收第二位
164             begin
165                 R_rx_state <= R_rx_state + 1b1;
166                 O_rx_done <= 1b0;
167                 O_data_out[2] <= I_spi_miso;
168             end
169             3d6://接收第一位
170             begin
171                 R_rx_state <= R_rx_state + 1b1;
172                 O_rx_done <= 1b0;
173                 O_data_out[1] <= I_spi_miso;
174             end
175             3d7://接收第零位
176             begin
177                 R_rx_state <= R_rx_state + 1b1;
178                 O_rx_done <= 1b1;//接收完毕
179                 O_data_out[0] <= I_spi_miso;
180             end
181             default:R_rx_state <= 3d0;
182         endcase
183 end
184 
185 reg rxd_flag_r0,rxd_flag_r1;//接收标志位(当前态),接收标志位(之前态)
186 always@(posedge I_clk or negedge I_rst_n)
187 begin
188     if(!I_rst_n)
189         begin
190             rxd_flag_r0 <= 1b0;
191             rxd_flag_r1 <= 1b0;
192         end
193     else
194         begin
195             rxd_flag_r0 <= O_rx_done;
196             rxd_flag_r1 <= rxd_flag_r0;
197         end
198 end
199 //接收标志位
200 assign rxd_flag = (~rxd_flag_r1 & rxd_flag_r0)? 1b1:1b0;
201 
202 reg txd_flag_r0,txd_flag_r1;//发送标志位(当前态),发送标志位(之前态)
203 always@(posedge I_clk or negedge I_rst_n)
204 begin
205     if(!I_rst_n)
206         begin
207             txd_flag_r0 <= 1b0;
208             txd_flag_r1 <= 1b0;
209         end
210     else
211         begin
212             txd_flag_r0 <= O_tx_done;
213             txd_flag_r1 <= txd_flag_r0;
214         end
215 end
216 //发送标志位
217 assign txd_flag = (~txd_flag_r1 & txd_flag_r0)? 1b1:1b0;
218 
219 
220 endmodule
spi_s.v
技术图片
  1 `timescale 1 ns/ 1 ps
  2 module spi_s_vlg_tst();
  3 // constants                                           
  4 // general purpose registers
  5 reg eachvec;
  6 // test vector input registers
  7 reg I_clk;
  8 reg [7:0] I_data_in;
  9 reg I_rst_n;
 10 reg I_spi_cs;
 11 reg I_spi_miso;
 12 reg I_spi_sck;
 13 // wires                                               
 14 wire [7:0]  O_data_out;
 15 wire O_rx_done;
 16 wire O_spi_mosi;
 17 wire O_tx_done;
 18 wire rxd_flag;
 19 wire sck_negedge;
 20 wire sck_posedge;
 21 wire txd_flag;
 22 
 23 // assign statements (if any)                          
 24 spi_s i1 (
 25 // port map - connection between master ports and signals/registers   
 26     .I_clk(I_clk),
 27     .I_data_in(I_data_in),
 28     .I_rst_n(I_rst_n),
 29     .I_spi_cs(I_spi_cs),
 30     .I_spi_miso(I_spi_miso),
 31     .I_spi_sck(I_spi_sck),
 32     .O_data_out(O_data_out),
 33     .O_rx_done(O_rx_done),
 34     .O_spi_mosi(O_spi_mosi),
 35     .O_tx_done(O_tx_done),
 36     .rxd_flag(rxd_flag),
 37     .sck_negedge(sck_negedge),
 38     .sck_posedge(sck_posedge),
 39     .txd_flag(txd_flag)
 40 );
 41 reg [2:0]state;
 42 initial                                                
 43 begin                                                  
 44     I_clk = 0;
 45     I_rst_n = 0;
 46     I_data_in = 8h00;
 47     I_spi_miso = 0;
 48     I_spi_cs = 1;
 49     I_spi_sck = 0;
 50     #100
 51     I_rst_n = 1;
 52     I_spi_cs = 0;
 53 end                                                    
 54 always                                                                  
 55 begin                                                  
 56     #10 I_clk = ~I_clk;  
 57 end
 58     
 59 always                                                                  
 60 begin                                                  
 61     #20 I_spi_sck = ~I_spi_sck; 
 62 end
 63 
 64 always @(posedge I_spi_sck,negedge I_rst_n)               
 65 begin                                                  
 66     if(!I_rst_n)
 67         I_data_in <= 8h00;
 68     else if(I_data_in == 8hff)
 69     begin
 70         I_data_in <= 8h00;
 71     end
 72     else if(txd_flag)
 73         I_data_in <= I_data_in + 1b1;
 74 end                
 75 //11101010
 76 always @(negedge I_spi_sck,negedge I_rst_n)
 77 begin 
 78     if(!I_rst_n)
 79         state <= 3b000;
 80     else
 81     case(state)
 82         3d0:
 83         begin
 84             state <= state + 1;
 85             I_spi_miso <= 1b1;
 86         end
 87         3d1:
 88         begin
 89             state <= state + 1;
 90             I_spi_miso <= 1b1;
 91         end
 92         3d2:
 93         begin
 94             state <= state + 1;
 95             I_spi_miso <= 1b1;
 96         end
 97         3d3:
 98         begin
 99             state <= state + 1;
100             I_spi_miso <= 1b0;
101         end
102         3d4:
103         begin
104             state <= state + 1;
105             I_spi_miso <= 1b1;
106         end
107         3d5:
108         begin
109             state <= state + 1;
110             I_spi_miso <= 1b0;
111         end
112         3d6:
113         begin
114             state <= state + 1;
115             I_spi_miso <= 1b1;
116         end
117         3d7:
118         begin
119             state <= state + 1;
120             I_spi_miso <= 1b0;
121         end
122         default:state <= 3b000;
123     endcase
124 end
125 endmodule
spi_s.vt(测试代码)

五、仿真波形图

技术图片

 

 六、参考资料

https://www.cnblogs.com/wanghuaijun/p/7627065.html

https://blog.csdn.net/weixin_42509369/article/details/83096349

https://blog.csdn.net/qq_22168673/article/details/101546898

https://www.cnblogs.com/liujinggang/p/9609739.html

以上是关于STM32与FPGA进行SPI通信的主要内容,如果未能解决你的问题,请参考以下文章

FPGA作为从机与STM32进行SPI协议通信---Verilog实现

STM32如何通过串口与FPGA之间进行通信?.及要注意的问题。刚上手希望越详细越好,谢谢。

STM32 SPI 与 HAL 通信

我现在需要用STM32控制FPGA并且和FPGA进行通讯,请问STM32和FPGA的接口怎样才能实现呢?请高手们帮忙解答

STM32------- SPI通信

STM32与FPGA用FMC进行通讯