FPGA SPI 驱动程序

1.引言

此驱动程序已经完成很久了,花了2个星期的时间,主要是提升程序运行的效率。最近整理文件的时候又看到了,记录一下。

2.程序框架分解

module adc7254_Ctrl(input               sys_clk,    //system clkc 50Minput               reset_n,    //reset flaginput               iData_a_in,      //ADC to fpga input               iData_b_in, output               sclk_out,   //to ADC output              cs_out,     //to ADC  output              sdin,    //to ADCoutput 		 [11:0]	oData_a, //get dataoutput		 [11:0]	oData_b	//get data
);
wire clk_200M;                   //PLL驱动
AD_PLL AD_PLL_inst(.inclk0(sys_clk),.c0(Clk_200M),.c1(sclk_out)
);
wire En_conv,En_send;            //定义ADC发送和接收程序的状态切换时间
adc_test adc_test_inst(.iRst_n(reset_n),.iDclk(sclk_out),.iSend_down(Send_down),.oAdc_rst_n(adc_rst_n),.oEn_conv(En_conv),.oEn_send(En_send)
);
wire Send_down;                   //ADC指令发送部分
adc_in_send adc_in_send_inst
(.iClk_200M(Clk_200M),.iRst_n(adc_rst_n),.iDcLK(sclk_out),.iEn_send(En_send),.oSDATA(sdin),.oSend_down(Send_down)
); 
wire Conv_down;
adc_out_conv adc_out_conv_inst				//ADC数据采样部分
(.iClk_200M		( Clk_200M  	    ),.iRst_n		   ( adc_rst_n 	 ),.iEn_conv		( En_conv 		),.iDcLK			( sclk_out 		),.iData_a_in		( iData_a_in 	),.iData_b_in		( iData_b_in 	),.oData_a		( oData_a	 	),	.oData_b		( oData_b 		),.oConv_down     ( Conv_down     )
);
assign cs_out = Conv_down & Send_down;       //状态完成
endmodule

3.子任务分解

(1)状态控制程序

其主要是以空状态,写状态,读状态三个状态顺序执行的。

module adc_test(input iRst_n,input iDclk,input iSend_down,output oAdc_rst_n,output reg oEn_send,output reg oEn_conv
);reg [1:0] state;
reg [5:0] sclk_cnt;
localparam state_IDLE  = 2'd0;
localparam state_Write = 2'd1;
localparam state_Read  = 2'd3;assign oAdc_rst_n = (iRst_n & state);
always @(posedge iDclk or negedge iRst_n) beginif(!iRst_n)beginsclk_cnt <= 6'd0;state <= state_IDLE;oEn_conv = 1'd0;oEn_send = 1'd0;  end else begin   case(state)state_IDLE:beginif (sclk_cnt > 6'd30 ) beginsclk_cnt <= 6'd0;state <= state_Write;oEn_conv = 1'd0;oEn_send = 1'd1;end else beginsclk_cnt <= sclk_cnt + 1'd1;state <= state_IDLE; oEn_conv = 1'd0;oEn_send = 1'd0;   end    endstate_Write:beginif (iSend_down == 1'd1 && sclk_cnt > 6'd30 ) beginsclk_cnt <= 0;state <= state_Read;oEn_conv = 1'd1;oEn_send = 1'd0;end else beginsclk_cnt <= sclk_cnt + 1'd1;state <= state_Write;oEn_conv = 1'd0;oEn_send = 1'd1;endendstate_Read:beginstate <= state_Read;sclk_cnt <= 1'd0;oEn_conv = 1'd1;oEn_send = 1'd0;endendcaseend
endendmodule

(2)写命令程序部分

module adc_in_send(input iClk_200M,     //200Minput iRst_n,input iDcLK,    //最小T>60ns input iEn_send,	 output  oSDATA,output oSend_down     );//==================使能接收标志位en==================////一旦启动不会突然停止除非复位信号到来reg en;reg [5:0] sclk_cnt;always @(posedge iDcLK or negedge iRst_n ) beginif (!iRst_n) beginen <= 1'd0;sclk_cnt <= 6'd0;end else if ( iEn_send == 1'd1 && sclk_cnt == 6'd0 ) beginen <= 1'd1;sclk_cnt <= 6'd32;end else if ( sclk_cnt > 6'd1 ) beginen <= en;sclk_cnt <= sclk_cnt - 1'd1;end else if (oSend_down == 1'd1 && sclk_cnt == 6'd1 ) beginen <= 1'd0;sclk_cnt <= sclk_cnt - 1'd1;end else beginen <= en;sclk_cnt <= sclk_cnt;endend
//==================使能接收标志位en==================//
//==================SDATA输出操作=========================//
reg [15:0]CFR_16bit_data = 16'h8840;	//需要写入寄存器中的数据
assign oSDATA = (en > 1'd0) ? ((sclk_cnt > 6'd17) ? CFR_16bit_data[sclk_cnt-6'd17] : 0 ): 0;
//==================SDATA操作=========================//
//==================oSend_down操作======================//
assign oSend_down = (sclk_cnt > 6'd1) ? 0 : 1;
//==================oSend_down操作======================//
endmodule

(3)读数据程序部分

module adc_out_conv(input iClk_200M,     //200Minput iRst_n,input iData_a_in,input iData_b_in,  input iDcLK,    //最小T=60ns input iEn_conv,output reg [11:0] oData_a,output reg [11:0] oData_b,output  oConv_down     //T>70ns);
//下降沿接收
//==================使能接收标志位en==================//
//一旦启动不会突然停止除非复位信号到来
reg en;//接收使能标志位
reg [5:0] sclk_cnt;
always @(posedge iDcLK or negedge iRst_n ) beginif (!iRst_n) beginen <= 1'd0;sclk_cnt <= 5'd0;end else if (iEn_conv == 1'd1   &&  sclk_cnt == 4'd0) beginen <= 1'd1;sclk_cnt <= 6'd17;end else if (sclk_cnt > 4'd1) beginen <= en;sclk_cnt  <= sclk_cnt - 1'd1;end else if (oConv_down == 1'd1 &&  sclk_cnt == 4'd1)beginen <= 1'd0;sclk_cnt <= sclk_cnt - 1'd1;end else beginen <= en;sclk_cnt <= sclk_cnt;end
end
//==================使能接收标志位en==================//
//==================dclk时钟采样==================//
reg	[6:0] dclk;
always@(posedge iClk_200M or negedge iRst_n) beginif(!iRst_n) begindclk <= 7'd0;end	else if(!en) begindclk <= 7'd0;end  else begindclk <= {dclk[5:0],iDcLK};endend
//==================dclk时钟采样==================//
//==================状态切换==================//reg [1:0] state;parameter state_IDLE = 2'd0;parameter state_Read = 2'd1;parameter state_Write = 2'd2;always@(posedge iClk_200M or negedge iRst_n)  beginif(!iRst_n ) beginstate <= state_IDLE;end else if(!en) beginstate <= state_IDLE;end else if(dclk[1] == 1 & dclk[2] == 0) beginstate <= state_Read;end else if (dclk[1] == 0 & dclk[2] == 1) beginstate <= state_Write;end else beginstate <=state;end            end
//==================状态切换==================//
//==================data串行转并行==================//
reg [2:0] Data_a_in_temp,Data_b_in_temp;//保证7次采样有4次为1
reg [11:0]  Data_a_temp,Data_b_temp;
always@(posedge iClk_200M or negedge iRst_n)
beginif(!iRst_n ) beginData_a_temp    <= 12'd0;Data_b_temp    <= 12'd0;Data_a_in_temp <= 3'd0;Data_b_in_temp <= 3'd0;end else if(sclk_cnt > 6'd16)beginData_a_in_temp <= 3'd0;Data_b_in_temp <= 3'd0;Data_a_temp    <= 12'd0;Data_b_temp    <= 12'd0;end else if(sclk_cnt > 6'd3 ) beginif(state == state_Read && dclk[6] == 0 ) beginData_a_in_temp <= Data_a_in_temp + iData_a_in;Data_b_in_temp <= Data_b_in_temp + iData_b_in;Data_a_temp    <= Data_a_temp;Data_b_temp    <= Data_b_temp;end else if(state == state_Write && dclk[0]!=dclk[1])beginData_a_in_temp <= 3'd0;Data_b_in_temp <= 3'd0;Data_a_temp    <= {Data_a_temp[10:0],Data_a_in_temp[2]};Data_b_temp    <= {Data_b_temp[10:0],Data_b_in_temp[2]};end else beginData_a_in_temp <= Data_a_in_temp;Data_b_in_temp <= Data_b_in_temp;Data_a_temp    <= Data_a_temp;Data_b_temp    <= Data_b_temp;endend else beginData_a_in_temp <= Data_a_in_temp;Data_b_in_temp <= Data_b_in_temp;Data_a_temp    <= Data_a_temp;Data_b_temp    <= Data_b_temp;end
end
//==================data串行转并行==================//
//==================oConv_down操作======================//
assign oConv_down = (sclk_cnt > 6'd1) ? 0 : 1;
//==================oConv_down操作======================//
//==================数据按帧输出==================//
always@(posedge iClk_200M or negedge iRst_n)
beginif(!iRst_n )beginoData_a <= 12'd0;oData_b <= 12'd0;endelse if( oConv_down == 1'd1)beginoData_a <= Data_a_temp;oData_b <= Data_b_temp;endelsebeginoData_a <= oData_a;oData_b <= oData_b;end
end
//==================数据按帧输出==================//
endmodule

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/65329.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

react跳转页面redux数据被清除

关键代码如下&#xff0c;页面中有根据redux中state展示的数据&#xff0c;然后在组件卸载的时候会清空redux中存的数据&#xff0c;点击a标签可以打开新的标签页&#xff0c;如下代码会在打开新的标签页&#xff0c;组件卸载&#xff0c;清空redux数据&#xff0c;页面展示的也…

MySQL中的索引事务(1)索引----》数据库运行的原理知识+面试题~

本篇文章主要讲述MySQL索引事务 所谓的索引index就是指&#xff1a;目录~~ 索引存在的意义&#xff1a;加快查找的速度&#xff08;省略了遍历的过程&#xff09;&#xff0c;但付出了一定的代价~ 付出的代价如下&#xff1a; 1.需要付出额外的空间代价来保存索引数据2.索引…

Nginx安装及使用

这里写目录标题 Nginx一.IO模型1.I/O模型相关概念2.网络I/O模型阻塞型I/O模型非阻塞型I/O模型多路复用 I/O 型信号驱动式 I/O模型异步I/O模型总结 二.Nginx概述三.Nginx模块四.安装及使用Nginx1.编译安装Nginx2.yum安装3.命令及信号使用4.USR1分割日志5.升级Nginx 五.配置详细解…

c#object类中方法的使用

C#中的Object类是所有类的基类&#xff0c;它定义了一些通用的方法和属性&#xff0c;可以在任何对象上使用。以下是Object类中常用的方法和属性的使用&#xff1a; 1.ToString()&#xff1a;将对象转换为字符串表示形式。 string str obj.ToString();2.Equals()&#xff1a;…

Windows下Redis的安装和配置

文章目录 一,Redis介绍二,Redis下载三,Redis安装-解压四,Redis配置五,Redis启动和关闭(通过terminal操作)六,Redis连接七,Redis使用 一,Redis介绍 远程字典服务,一个开源的,键值对形式的在线服务框架,值支持多数据结构,本文介绍windows下Redis的安装,配置相关,官网默认下载的是…

【力扣】两数相除(c/c++)

目录 题目 注意&#xff1a; 示例 1: 示例 2: 提示&#xff1a; 题目解析 题目思路 代码思路 数据处理 注意 减法函数 第一次使用的函数 问题 第二次改良后的代码 处理i的值并且返回 总代码 力扣的代码 注意 题目 给你两个整数&#xff0c;被除数 dividend 和…

关于Comparable、Comparator接口返回值决定顺序的问题

Comparable和Comparator接口都是实现集合中元素的比较、排序的&#xff0c;下面先简单介绍下他们的用法。 1. 使用示例 public class Person {private String name;private Integer age;public Person() {}public Person(String name, Integer age) {this.name name;this.ag…

【UE】Texture Coordinate 材质节点

目录 一、简介 二、属性介绍 &#xff08;1&#xff09;参数&#xff1a;U平铺 &#xff08;2&#xff09;参数&#xff1a;V平铺 &#xff08;3&#xff09;参数&#xff1a;解除镜像U &#xff08;4&#xff09;参数&#xff1a;解除镜像V 三、 节点构成原理 四、初级…

基于YOLOV8模型的农作机器和行人目标检测系统(PyTorch+Pyside6+YOLOv8模型)

摘要&#xff1a;基于YOLOV8模型的农作机器和行人目标检测系统可用于日常生活中检测与定位农作机和行人目标&#xff0c;利用深度学习算法可实现图片、视频、摄像头等方式的目标检测&#xff0c;另外本系统还支持图片、视频等格式的结果可视化与结果导出。本系统采用YOLOv8目标…

统一网关Gateway

文章目录 概览网关的作用搭建网关断言工厂路由过滤器全局过滤器案例 过滤器执行顺序跨域问题 概览 网关的作用 搭建网关 断言工厂 路由过滤器 全局过滤器 案例 过滤器执行顺序 跨域问题

DB2 HADR+TSA运维,TSA添加资源组的命令

Tivoli System Automation&#xff08;TSA&#xff09;是一个高可用性集群管理软件,DB2 TSAHADR高可用方案可以实现DB2 hadr主备的自动检测切换。本文详细介绍了TSA的常用命令&#xff0c;如何把CDC或者DSG添加到TSA集群中&#xff0c;以及TSA的错误分析方法 常用命令&#xf…

论文阅读:Distortion-Free Wide-Angle Portraits on Camera Phones

论文阅读&#xff1a;Distortion-Free Wide-Angle Portraits on Camera Phones 今天介绍一篇谷歌 2019 年的论文&#xff0c;是关于广角畸变校正的。 Abstract 广角摄影&#xff0c;可以带来不一样的摄影体验&#xff0c;因为广角的 FOV 更大&#xff0c;所以能将更多的内容…

2006 - MySQL server has gone away

解决方法&#xff1a; 1、修改my.cnf配置文件 vim my.cnf 添加如下配置 wait_timeout288000 interactive_timeout 288000 max_allowed_packet100M 解释&#xff1a; wait_timeout&#xff1a;服务器关闭交互式连接前等待活动的秒数 interactive_timeout&#xff1a;服务器关闭…

单片机的ADC

如何理解ADC。ADC就是将模拟量转换成数字量的过程&#xff0c;就是转换为计算机所能存储的0和1序列&#xff0c;比如将模拟量转换为一个字节&#xff0c;所以这个字节的大小要能反应模拟量的大小&#xff0c;比如一个0-5V的电压测量量&#xff08;外部输入电压最小0V,最大为5V&…

websocket基础

下面就以代码来进行说明 1&#xff0c;先导入websocket依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId></dependency> 2.编写websocket相关bean管理配置 Config…

【ES6】迭代器Iterator

JavaScript中的Iterator是一种特殊对象&#xff0c;它允许我们访问并操作对象的每一个元素。Iterator对象由具有next方法的对象创建&#xff0c;next方法返回一个包含两个属性的对象&#xff1a;value和done。value属性是当前元素的值&#xff0c;done属性是一个布尔值&#xf…

Unity生命周期函数

1、Awake 当对象&#xff08;自己这个类对象&#xff0c;就是这个脚本&#xff09;被创建时 才会调用该生命周期函数 类似构造函数的存在 我们可以在一个类对象创建时进行一些初始化操作 2、OnEnable 失活激活&#xff08;这个勾&#xff09; 想要当一个对象&#xff08;游戏…

【杂言】写在研究生开学季

这两天搬进了深研院的宿舍&#xff0c;比中南的本科宿舍好很多&#xff0c;所以个人还算满意。受台风 “苏拉” 的影响&#xff0c;原本的迎新计划全部打乱&#xff0c;导致我现在都还没报道。刚开学的半个月将被各类讲座、体检以及入学教育等活动占满&#xff0c;之后又是比较…

ZDH-权限模块

本次介绍基于ZDH v5.1.2版本 目录 项目源码 预览地址 安装包下载地址 ZDH权限模块 ZDH权限模块-重要名词划分 ZDH权限模块-菜单管理 ZDH权限模块-角色管理 ZDH权限模块-用户配置 ZDH权限模块-权限申请 项目源码 zdh_web: GitHub - zhaoyachao/zdh_web: 大数据采集,抽…

ROS 2官方文档(基于humble版本)学习笔记(一)

ROS 2官方文档&#xff08;基于humble版本&#xff09;学习笔记&#xff08;一&#xff09; 一、安装ROS 2二、按教程学习1.CLI 工具配置环境使用turtlesim&#xff0c;ros2和rqt安装 turtlesim启动 turtlesim使用 turtlesim安装 rqt使用 rqt重映射关闭turtlesim 由于市面上专门…