IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> FPGA(主)与STM32(从)SPI通信 -> 正文阅读

[嵌入式]FPGA(主)与STM32(从)SPI通信

一、概述

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

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

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

? ? ? ? 下面贴出FPGA和STM32的代码


二、FPGA端代码

spi主模式

module spi_master(
	input clk_1m,					//时钟信号 1m
	input rst_n,					//按键复位信号
	input ena_mo,					//模块使能
	input [7:0]spi_tdata,		//spi要发送的数据
	input spi_miso,				//spi 主进从出
	output reg spi_mosi,			//spi 从进主出
	output reg spi_sck,			//spi 时钟信号
	output reg spi_nss,			//spi 片选信号
	output reg [7:0]spi_rdata,	//spi接收到的数据
	output tr_done					//spi 完成一个字节发送
);

localparam waiten=0;				//等待模块使能
localparam tx_revl=1;			//sck低电平
localparam tx_revh=2;			//sck高电平
localparam done=3;				//完成一次发送
localparam DELAY = 5;			//完成时等待5个时钟开始下一次发送

//reg define			
reg [3:0] us_cnt;					//us计数器
reg us_cnt_clr;					//计数器清零信号
reg[1:0] state,next_state;		//当前状态和下一个状态
reg[3:0] w_cnt;					//位计数器

//1微秒计数器
always @ (posedge clk_1m,negedge rst_n) begin
    if (!rst_n)
        us_cnt <= 21'd0;
    else if (us_cnt_clr)
        us_cnt <= 21'd0;
    else 
        us_cnt <= us_cnt + 1'b1;
end 

//下一个状态确认
always @(*) begin
	if(!rst_n)
		next_state = waiten;
	case(state)
		//模块使能且计数达到延时则进入下一个状态
		waiten: next_state = (ena_mo&&us_cnt==DELAY) ? tx_revl : waiten;
		tx_revl: next_state = tx_revh;
		tx_revh: next_state = (w_cnt == 4'd8) ? done : tx_revl;
		done: next_state = (ena_mo&&us_cnt==DELAY) ? tx_revl : waiten;
	endcase
end

//状态的各变量赋值
always @(posedge clk_1m,negedge rst_n)begin
	if(!rst_n)begin
		spi_sck <= 1'b1;												//sck空闲状态为高电平
		spi_mosi <= 1'b0;
		spi_rdata <= 8'd0;
		us_cnt_clr <= 1'b1;
		spi_nss <= 1'b1;
	end
	else begin
		case(state)
			waiten:begin
				spi_sck <= 1'b1;
				spi_mosi <= 1'b0;
				us_cnt_clr <= 1'b0;
				spi_nss <= us_cnt > 3'd3 ? 1'b0 : 1'b1;		//nss信号在等待时提前拉低 选中 做好发送准备
			end
			
			tx_revl:begin
				us_cnt_clr <= 1'b1;		
				spi_mosi <= spi_tdata[3'd7-w_cnt];				//sck低电平时寄存数据
				spi_sck <= 1'b0;										//sck拉低
			end
			
			tx_revh:begin
				spi_sck <= 1'b1;										//sck拉高 数据值保持不变
				spi_rdata[3'd7-(w_cnt-1'b1)] <= spi_miso;		//在上升沿捕获数据值
			end
			
			done:begin
				us_cnt_clr <= 1'b0;									//发送完成 时钟计数器关闭复位,延时等待一下
				spi_sck <= 1'b1;
				spi_mosi <= 1'b0;
				spi_nss <= us_cnt > 3'd3 ? 1'b0 : 1'b1;		//nss信号在等待时提前拉低 选中 做好发送准备
			end
			
		endcase
	end
end

//状态转换
always @(posedge clk_1m,negedge rst_n)begin
	if(!rst_n)
		state <= waiten;
	else
		state <= next_state;
end

//位计数器赋值
always @(posedge clk_1m,negedge rst_n)begin
	if(!rst_n)begin
		w_cnt <= 4'd0;
	end
	else begin
		case(state)
			tx_revl:w_cnt <= w_cnt + 1'b1;//sck低电平时位计数器加1,所以高电平时就会多一个值,需要减1
			waiten: w_cnt <= 4'd0;
			done: w_cnt <= 4'd0;
			default: w_cnt <= w_cnt;
		endcase
	end
end

//完成信号输出赋值
assign tr_done = (state == done);

endmodule

testbench仿真代码


`timescale 1ns/1ns //仿真单位为1ns,精度为1ns

module spi_master_tb();
	reg clk_1m;
	reg rst_n;
	reg ena_mo;
	reg [7:0] spi_tdata;
	reg spi_miso;
	wire spi_mosi;
	wire spi_sck;
	wire spi_nss;
	wire [7:0] spi_rdata;
	
	spi_master u_spi_master(
		.clk_1m(clk_1m),
		.rst_n(rst_n),
		.ena_mo(ena_mo),
		.spi_tdata(spi_tdata),
		.spi_miso(spi_miso),
		.spi_mosi(spi_mosi),
		.spi_sck(spi_sck),
		.spi_nss(spi_nss),
		.spi_rdata(spi_rdata)
	);
	
	initial begin
		#0	clk_1m = 0;
			rst_n = 0;
			ena_mo = 1;
			spi_tdata = 8'b11001010;
			spi_miso = 0;
			
		#20 rst_n = 1;
		
	end
	
	always #5 clk_1m = ~clk_1m;

	always #32 spi_miso = ~spi_miso;
	
endmodule

仿真结果


?顶层模块

module oscilloscope(
	input sys_clk,				//系统时钟 50m
	input rst_n,				//按键复位
	input spi_miso,			//spi 主进从出信号
	output spi_mosi,			//spi 主出从进信号
	output spi_sck,			//spi 时钟信号
	output spi_nss				//spi 片选信号(使用这个信号可以提高传输的稳定性)
);

wire [7:0] spi_rdata;		//spi读取到的数据
wire [7:0] spi_tdata;		//spi要发送的数据
wire tr_done;					//一个字节发送完成信号
wire clk_1m;					//1M 时钟信号

reg [7:0] tx_data;			//给要发送的数据赋值

assign spi_tdata = tx_data;//要发送的数据赋值

always @(posedge clk_1m,negedge rst_n)begin
	if(!rst_n)
		tx_data <= 8'd0;
	else begin
		tx_data <= tr_done ? tx_data + 1'b1 : tx_data;//每发送完成一个字节 数据的值增加1
	end
end

//时钟分频模块 产生1MHz时钟
clk_fenpin u_clk_fenpin(
	.sys_clk(sys_clk),
	.rst_n(rst_n),
	.clk_1m(clk_1m)
);

//spi 主模式模块
spi_master u_spi_master(
	.clk_1m(clk_1m),
	.rst_n(rst_n),
	.ena_mo(1'b1),
	.spi_tdata(tx_data),
	.spi_miso(spi_miso),
	.spi_mosi(spi_mosi),
	.spi_sck(spi_sck),
	.spi_nss(spi_nss),
	.spi_rdata(spi_rdata),
	.tr_done(tr_done)
);

endmodule

时钟分频模块

module clk_fenpin(
	input sys_clk,
	input rst_n,
	output reg clk_1m
);

	reg    [25:0]   clk_cnt     ;        //分频计数器
	//得到1Mhz分频时钟
	always @ (posedge sys_clk or negedge rst_n) begin
		 if (!rst_n) begin
			  clk_cnt <= 5'd0;
			  clk_1m  <= 1'b0;
		 end 
		 else if (clk_cnt < 26'd24) 
			  clk_cnt <= clk_cnt + 1'b1;       
		 else begin
			  clk_cnt <= 5'd0;
			  clk_1m  <= ~ clk_1m;
		 end 
	end

endmodule

三、STM32断配置

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

spi配置

u8 spi_revdata[spirv_max_len];    //接收数据数组

SPI_HandleTypeDef SPI5_Handler;   //SPI句柄

u8 tx = 15;                        //发送的数据

//以下是SPI模块的初始化代码,配置成主机模式 						  
//SPI口初始化
//这里针是对SPI5的初始化
void SPI5_Init(void)
{
    SPI5_Handler.Instance=SPI5;                         //SP5
    SPI5_Handler.Init.Mode=SPI_MODE_SLAVE;             	//设置SPI工作模式,设置为从模式
    SPI5_Handler.Init.Direction=SPI_DIRECTION_2LINES;   //设置SPI单向或者双向的数据模式:SPI设置为双线模式
    SPI5_Handler.Init.DataSize=SPI_DATASIZE_8BIT;       //设置SPI的数据大小:SPI发送接收8位帧结构
    SPI5_Handler.Init.CLKPolarity=SPI_POLARITY_HIGH;    //串行同步时钟的空闲状态为高电平
    SPI5_Handler.Init.CLKPhase=SPI_PHASE_2EDGE;         //串行同步时钟的第二个跳变沿(上升或下降)数据被采样
    SPI5_Handler.Init.NSS=SPI_NSS_HARD_INPUT;                 //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理 选择硬件管理
    SPI5_Handler.Init.BaudRatePrescaler=SPI_BAUDRATEPRESCALER_4;//定义波特率预分频的值:波特率预分频值为4
    SPI5_Handler.Init.FirstBit=SPI_FIRSTBIT_MSB;        //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
    SPI5_Handler.Init.TIMode=SPI_TIMODE_DISABLE;        //关闭TI模式
    SPI5_Handler.Init.CRCCalculation=SPI_CRCCALCULATION_DISABLE;//关闭硬件CRC校验
    SPI5_Handler.Init.CRCPolynomial=7;                  //CRC值计算的多项式
    HAL_SPI_Init(&SPI5_Handler);//初始化
    
    __HAL_SPI_ENABLE(&SPI5_Handler);                    //使能SPI5
	
	DMA2_Init();                                        //DMA初始化
	
	
	                                                    //开启SPI的DMA传输 1024个字节
	HAL_SPI_TransmitReceive_DMA(&SPI5_Handler,&tx,spi_revdata,spirv_max_len);
	
    //发送和接收如果同时使用的话最好使用上面那个函数,不然开启发送,再开接收可能导致没有开启成功
//	HAL_SPI_Transmit_DMA(&SPI5_Handler,&tx,1024);
//	HAL_SPI_Receive_DMA(&SPI5_Handler,spi_revdata,spirv_max_len);
}

//SPI5底层驱动,时钟使能,引脚配置
//此函数会被HAL_SPI_Init()调用
//hspi:SPI句柄
void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi)
{
    GPIO_InitTypeDef GPIO_Initure;
    
    __HAL_RCC_GPIOF_CLK_ENABLE();                   //使能GPIOF时钟
    __HAL_RCC_SPI5_CLK_ENABLE();                    //使能SPI5时钟
    
    //PF6,7,8,9
    GPIO_Initure.Pin=GPIO_PIN_6|GPIO_PIN_7|GPIO_PIN_8|GPIO_PIN_9;
    GPIO_Initure.Mode=GPIO_MODE_AF_PP;              //复用推挽输出
    GPIO_Initure.Pull=GPIO_NOPULL;                  //上拉
    GPIO_Initure.Speed=GPIO_SPEED_FAST;             //快速            
    GPIO_Initure.Alternate=GPIO_AF5_SPI5;           //复用为SPI5
    HAL_GPIO_Init(GPIOF,&GPIO_Initure);
}

dma配置


DMA_HandleTypeDef DMA_SPI5TX_Handler;									//SPI5 发送DMA句柄
DMA_HandleTypeDef DMA_SPI5RX_Handler;									//SPI5 接收DMA句柄

void DMA2_Init(void)
{
	__HAL_RCC_DMA2_CLK_ENABLE();                                        //使能DMA2的时钟
	
	DMA_SPI5RX_Handler.Instance = DMA2_Stream3;							//数据流选择
    DMA_SPI5RX_Handler.Init.Channel = DMA_CHANNEL_2;					//通道选择
    DMA_SPI5RX_Handler.Init.Direction = DMA_PERIPH_TO_MEMORY;			//外设到存储器
    DMA_SPI5RX_Handler.Init.PeriphInc = DMA_PINC_DISABLE;				//外设地址非增量模式
    DMA_SPI5RX_Handler.Init.MemInc = DMA_MINC_ENABLE;					//存储器地址增量模式
    DMA_SPI5RX_Handler.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;	//外设数据长度:8位
    DMA_SPI5RX_Handler.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;		//存储器数据长度:8位
    DMA_SPI5RX_Handler.Init.Mode = DMA_CIRCULAR;						//DMA循环模式 一次开启 一直搬运
    DMA_SPI5RX_Handler.Init.Priority = DMA_PRIORITY_HIGH;				//高优先级
    DMA_SPI5RX_Handler.Init.FIFOMode = DMA_FIFOMODE_DISABLE;			//不使用FIFO模式
	
	HAL_DMA_DeInit(&DMA_SPI5RX_Handler);								//清除配置
	HAL_DMA_Init(&DMA_SPI5RX_Handler);									//初始化配置
	
	__HAL_LINKDMA(&SPI5_Handler,hdmarx,DMA_SPI5RX_Handler);				//将SPI的发送和DMA联系起来
	
	
	DMA_SPI5TX_Handler.Instance = DMA2_Stream4;							//数据流选择
    DMA_SPI5TX_Handler.Init.Channel = DMA_CHANNEL_2;                    //通道选择
    DMA_SPI5TX_Handler.Init.Direction = DMA_MEMORY_TO_PERIPH;           //存储器到外设
    DMA_SPI5TX_Handler.Init.PeriphInc = DMA_PINC_DISABLE;               //外设地址非增量模式
    DMA_SPI5TX_Handler.Init.MemInc = DMA_MINC_DISABLE;                  //存储器地址非增量模式
    DMA_SPI5TX_Handler.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;  //外设数据长度:8位
    DMA_SPI5TX_Handler.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;     //存储器数据长度:8位
    DMA_SPI5TX_Handler.Init.Mode = DMA_CIRCULAR;                        //DMA循环模式 一次开启 一直搬运
    DMA_SPI5TX_Handler.Init.Priority = DMA_PRIORITY_MEDIUM;             //中等优先级
    DMA_SPI5TX_Handler.Init.FIFOMode = DMA_FIFOMODE_DISABLE;            //不使用FIFO模式
	                                                                    
	HAL_DMA_DeInit(&DMA_SPI5TX_Handler);                                //清除配置
	HAL_DMA_Init(&DMA_SPI5TX_Handler);                                  //初始化配置
	                                                                    
	__HAL_LINKDMA(&SPI5_Handler,hdmatx,DMA_SPI5TX_Handler);             //将SPI的发送和DMA联系起来
}

四、实现效果

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

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

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-12-07 12:12:19  更:2021-12-07 12:12:53 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/9 2:06:23-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码