SPI 配置寄存器程序

=========================================================================

=========================================================================

/**************************************************
*   
**************************************************/
module  zhm_mspi    #(
parameter   C_SPI_CPHA        = 1 ,// clock phase   ,0,在 SCLK 的第一个跳变沿进行采样;1,在 SCLK 的第二个跳变沿进行采样 
parameter   C_SPI_CPOL        = 1 ,// clock polarity,0,SCLK 在空闲时为低电平;         1,SCLK 在空闲时为高电平          
parameter   C_SPI_ADDR_WIDTH  = 8 ,// 地址位宽,广义上的
parameter   C_SPI_DATA_WIDTH  = 8 ,// 数据位宽
parameter   C_SPI_DIV_VALUE   = 10,// sclk 分频数
//  计数器位宽 
parameter   C_SPI_CMD_CWIDTH  = 4 ,// 地址/数据  -计数器位宽
parameter   C_SPI_DIV_CWIDTH  = 4 ,// sclk 分频数-计数器位宽
//                            
parameter   C_SPI_CMD_WIDTH   = C_SPI_ADDR_WIDTH+C_SPI_DATA_WIDTH
)(
input                                   aclk    ,//
input                                   aresetn ,//
output                                  s_tready,//
input                                   s_tvalid,//
input   [C_SPI_CMD_WIDTH-1:0]           s_tdata ,//
output                                  m_tvalid,//
output  [C_SPI_CMD_WIDTH-1:0]           m_tdata ,//
output                                  spi_cs  ,//
output                                  spi_sclk,//
output                                  spi_mosi,//
input                                   spi_miso,//
output                                  spi_t   ,//
output  [02:00]                         cursta  ,
output  [C_SPI_DIV_CWIDTH-1:0]          curcntD ,
output  [C_SPI_CMD_CWIDTH-1:0]          curcntB
);
localparam  TB_EDGE = C_SPI_DIV_VALUE/2-1;// 
/**************************************************
*   分频计数器 D
**************************************************/
localparam  CntD_End_Value = C_SPI_DIV_VALUE-1;
localparam  CntD_Width     = C_SPI_DIV_CWIDTH ;
reg     [CntD_Width-1:0]    cntD = {CntD_Width{1'b0}};
wire                        cntD_dom;
wire                        cntD_add;
wire                        cntD_end;
/**************************************************
*   位计数器 B
**************************************************/
//localparam  CntB_End_Value = 100-1;
localparam  CntB_Width     = C_SPI_CMD_CWIDTH;
reg     [CntB_Width-1:0]    cntB = {CntB_Width{1'b0}};
wire                        cntB_dom;
wire                        cntB_add;
wire                        cntB_end;
/**************************************************
*   状态机
*   IDLE==>S1==>S2==>S3==>S4==>S5==>S6==>S7==>S1
**************************************************/
localparam          IDLE = 3'd0;
localparam          S1   = 3'd1;
localparam          S2   = 3'd2;
localparam          S3   = 3'd3;
localparam          S4   = 3'd4;
localparam          S5   = 3'd5;
localparam          S6   = 3'd6;
localparam          S7   = 3'd7;
reg     [02:00]     state_c = IDLE;
reg     [02:00]     state_n       ;
wire                idle2s1_start ;
wire                s12s2_start   ;
wire                s22s3_start   ;
wire                s32s4_start   ;
wire                s42s5_start   ;
wire                s52s6_start   ;
wire                s62s7_start   ;
wire                s72s1_start   ;
always  @(posedge aclk)begin
    if(!aresetn)state_c <= IDLE;
    else        state_c <= state_n;
end
always  @(*)begin
    case(state_c)
        IDLE:begin  if(idle2s1_start)   state_n = S1;else   state_n = state_c;end
        S1  :begin  if(s12s2_start)     state_n = S2;else   state_n = state_c;end
        S2  :begin  if(s22s3_start)     state_n = S3;else   state_n = state_c;end
        S3  :begin  if(s32s4_start)     state_n = S4;else   state_n = state_c;end
        S4  :begin  if(s42s5_start)     state_n = S5;else   state_n = state_c;end
        S5  :begin  if(s52s6_start)     state_n = S6;else   state_n = state_c;end
        S6  :begin  if(s62s7_start)     state_n = S7;else   state_n = state_c;end
        S7  :begin  if(s72s1_start)     state_n = S1;else   state_n = state_c;end
        default:;
    endcase
end
assign  idle2s1_start = state_c == IDLE && 1'b1    ;// 复位中
assign  s12s2_start   = state_c == S1   && s_tvalid;// 等待数据中
assign  s22s3_start   = state_c == S2   && cntD_end;// CS 下降沿前等待
assign  s32s4_start   = state_c == S3   && cntD_end;// CS 下降沿后等待
assign  s42s5_start   = state_c == S4   && cntB_end;// 下发地址中
assign  s52s6_start   = state_c == S5   && cntB_end;// 收发数据中
assign  s62s7_start   = state_c == S6   && cntD_end;// CS 上升沿前等待
assign  s72s1_start   = state_c == S7   && cntD_end;// CS 上升沿后等待
assign  cursta        = state_c;
assign  s_tready      = (state_c == S1);
/**************************************************
*   分频计数器 D
**************************************************/
wire                cntD_flag;
assign  cntD_flag = (state_c == S2 ||
                     state_c == S3 ||
                     state_c == S4 ||
                     state_c == S5 ||
                     state_c == S6 ||
                     state_c == S7 );
always  @(posedge aclk)begin
    if(cntD_dom)begin if(cntD_add)begin
            if(cntD_end)cntD <= {CntD_Width{1'b0}};
            else        cntD <= cntD + 1'd1;
        end end else    cntD <= {CntD_Width{1'b0}};
end
assign  cntD_dom = cntD_flag;
assign  cntD_add = cntD_dom && 1'b1;
assign  cntD_end = cntD_add && cntD == CntD_End_Value;
assign  curcntD  = cntD;
/**************************************************
*   位计数器 B
**************************************************/
reg     [CntB_Width-1:0]        cntb_end_value;
always  @(*)begin 
    if(state_c == S4)   cntb_end_value = C_SPI_ADDR_WIDTH - 1'b1;
    else                cntb_end_value = C_SPI_DATA_WIDTH - 1'b1;
end 

always  @(posedge aclk)begin
    if(cntB_dom)begin if(cntB_add)begin
            if(cntB_end)cntB <= {CntB_Width{1'b0}};
            else        cntB <= cntB + 1'd1;
        end end else    cntB <= {CntB_Width{1'b0}};
end
assign  cntB_dom = (state_c == S4 || state_c == S5);
assign  cntB_add = cntB_dom && cntD_end;
assign  cntB_end = cntB_add && cntB == cntb_end_value;
assign  curcntB  = cntB;
/**************************************************
*   cs
**************************************************/
reg                         cs_r = 1'b1;
wire                        cs_w;
assign  cs_w = (state_c == S3 ||
                state_c == S4 ||
                state_c == S5 ||
                state_c == S6 );
always  @(posedge aclk)begin 
    cs_r <= !cs_w;
end 
assign  spi_cs = cs_r;
/**************************************************
*   方向控制 
**************************************************/
wire                            rhwl;
reg     [C_SPI_CMD_WIDTH-1:0]   data_sc;
assign  rhwl = data_sc[C_SPI_CMD_WIDTH-1:0];

wire                            t_flag;
assign  t_flag = (state_c == S4 ||
                  state_c == S5 );

reg                         t_r = 1'b1;
always  @(posedge aclk)begin 
    if(rhwl)    t_r <= (state_c != S4);
    else        t_r <= !t_flag;
end 
assign  spi_t = t_r;
/**************************************************
*   sclk
**************************************************/
reg                         sclk_r;
generate
    if(C_SPI_CPHA==0 && C_SPI_CPOL==0)begin: S00
        always  @(posedge aclk)begin 
            if(state_c == S4 || state_c == S5)begin 
                if(cntD == TB_EDGE) sclk_r <= 1'b1;
                else if(cntD_end)   sclk_r <= 1'b0;
            end 
            else                    sclk_r <= 1'b0;
        end 
    end 
    else if(C_SPI_CPHA==0 && C_SPI_CPOL==1)begin: S01
        always  @(posedge aclk)begin 
            if(state_c == S4 || state_c == S5)begin 
                if(cntD == TB_EDGE) sclk_r <= 1'b0;
                else if(cntD_end)   sclk_r <= 1'b1;
            end 
            else                    sclk_r <= 1'b1;
        end 
    end 
    else if(C_SPI_CPHA==1 && C_SPI_CPOL==0)begin: S10
        always  @(posedge aclk)begin 
            if(s32s4_start)         sclk_r <= 1'b1;
            else if(s52s6_start)    sclk_r <= 1'b0;
            else if(state_c == S4 || state_c == S5)begin 
                if(cntD == TB_EDGE) sclk_r <= 1'b0;
                else if(cntD_end)   sclk_r <= 1'b1;
            end 
            else                    sclk_r <= 1'b0;
        end 
    end
    else if(C_SPI_CPHA==1 && C_SPI_CPOL==1)begin: S11
        always  @(posedge aclk)begin 
            if(s32s4_start)         sclk_r <= 1'b0;
            else if(s52s6_start)    sclk_r <= 1'b1;
            else if(state_c == S4 || state_c == S5)begin 
                if(cntD == TB_EDGE) sclk_r <= 1'b1;
                else if(cntD_end)   sclk_r <= 1'b0;
            end 
            else                    sclk_r <= 1'b1;
        end 
    end
endgenerate

reg                         sclk_rr;
always  @(posedge aclk)begin 
    sclk_rr <= sclk_r;
end 
assign  spi_sclk = sclk_rr;
/**************************************************
*   数据发送 
**************************************************/
always  @(posedge aclk)begin
    if(s12s2_start) data_sc <= s_tdata;
end
reg     [C_SPI_CMD_WIDTH-1:0]   data_yw;
always  @(posedge aclk)begin
    if(cntD_end)begin 
        if(s32s4_start) data_yw <= data_sc;
        else            data_yw <= {data_yw[C_SPI_CMD_WIDTH-2:0],1'b0};
    end 
end
reg                             mosi_r;
always  @(posedge aclk)begin
    mosi_r <=  data_yw[C_SPI_CMD_WIDTH-1];
end
assign  spi_mosi = mosi_r;
/**************************************************
*   数据接收
**************************************************/
reg     [C_SPI_CMD_WIDTH-1:0]   rece_yw;
reg                             resc_sc_edge = 1'b0;
always  @(posedge aclk)begin
    resc_sc_edge <= t_flag && (cntD == TB_EDGE);
end
always  @(posedge aclk)begin
    if(resc_sc_edge)    rece_yw <= {rece_yw[C_SPI_CMD_WIDTH-2:0],spi_miso};
end

reg     [C_SPI_CMD_WIDTH-1:0]   rece_sc;
always  @(posedge aclk)begin
    if(s52s6_start)     rece_sc <= rece_yw;
end

reg                             valid_r = 1'b0;
always  @(posedge aclk)begin
    valid_r <= s52s6_start && rhwl;
end  

assign  m_tvalid = valid_r;
assign  m_tdata  = rece_sc;
endmodule
/**************************************************
zhm_mspi    #(
    .C_SPI_CPHA      (),// clock phase   ,0,在 SCLK 的第一个跳变沿进行采样;1,在 SCLK 的第二个跳变沿进行采样 
    .C_SPI_CPOL      (),// clock polarity,0,SCLK 在空闲时为低电平;         1,SCLK 在空闲时为高电平          
    .C_SPI_ADDR_WIDTH(),// 地址位宽,广义上的
    .C_SPI_DATA_WIDTH(),// 数据位宽
    .C_SPI_DIV_VALUE (),// sclk 分频数
    .C_SPI_CMD_CWIDTH(),// 地址/数据  -计数器位宽
    .C_SPI_DIV_CWIDTH())// sclk 分频数-计数器位宽                       
zhm_mspi_u(
    .aclk    (),//input                                   
    .aresetn (),//input                                   
    .s_tready(),//output                                  
    .s_tvalid(),//input                                   
    .s_tdata (),//input   [C_SPI_CMD_WIDTH-1:0]           
    .m_tvalid(),//output                                  
    .m_tdata (),//output  [C_SPI_CMD_WIDTH-1:0]           
    .spi_cs  (),//output                                  
    .spi_sclk(),//output                                  
    .spi_mosi(),//output                                  
    .spi_miso(),//input                                   
    .spi_t   (),//output                                  
    .cursta  (),//output  [02:00]                         
    .curcntD (),//output  [C_SPI_DIV_CWIDTH-1:0]          
    .curcntB () //output  [C_SPI_CMD_CWIDTH-1:0]          
);
**************************************************/

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Linux和Windows下查看CPU运行频率的方法

文章目录 0.前言1.Linux系统中查看CPU运行频率的方法&#xff08;经测试在UnRaid中适用的&#xff09;1.1.最简单的lscpu命令1.2.查看CPU实时运行频率的watch -n 1 cpufreq-info命令 2.WIndows系统中查看CPU运行频率的方法2.1.系统属性大法2.2.任务管理器大法2.3.CPU-Z等硬件检…

Codeforces Global Round 26 D. “a“ String Problem 【Z函数】

D. “a” String Problem 题意 给定一个字符串 s s s&#xff0c;要求把 s s s 拆分成若干段&#xff0c;满足以下要求&#xff1a; 拆分出来的每一个子段&#xff0c;要么是子串 t t t&#xff0c;要么是字符 a a a子串 t t t 至少出现一次 t ≠ " a " t \ne…

零基础非科班也能掌握的C语言知识22 预处理详解(完结)

预处理详解 1.预处理符号2.#define 定义常量3.#define 定义宏4.带有副作用的宏参数5.宏替换的规则6.宏函数的对比6.1 例子6.1 .16.1.26.1.3 7.命名约定8.undefin9.命令行定义(博主没办法演示)10.条件编译11.头文件的包含11.1本地文件11.2库文件的包含11.3 嵌套文件的包含 12.其…

python实现自动化测试框架如何进行数据参数化?这个包可以了解下

1.数据参数化介绍 只要你是负责编写自动化测试脚本的&#xff0c;数据参数化这个思想你就肯定会用 &#xff0c;数据参数化的工具你肯定的懂一些 &#xff0c;因为它能大大的提高我们自动化脚本编写效率 。 1.1什么是数据参数化 所谓的数据参数化 &#xff0c;是指所执行的测…

Unity 自定义房间布局系统 设计与实现一个灵活的房间放置系统 ——物体占用的区域及放置点自动化

放置物体功能 效果&#xff1a; 功能&#xff1a; 自定义物体占用区域的大小一键调整占用区域调整旋转度数&#xff0c;分四个挡位&#xff1a; NoRotation&#xff1a;该物体不能调整旋转。MaximumAngle&#xff1a;每次转动90。NormalAngle&#xff1a;每次转动45&#xff…

Solr 日志系统7.4.0部署和迁移到本地,Core Admin 添加新的core报错

文章目录 Solr部署Docker部署二进制部署 Tips:Solr设置账号密码方法1&#xff1a;(不使用)方法2&#xff1a; Core Admin 添加新的core报错Solr数据迁移 Solr部署 Docker部署 docker run -d -p 8983:8983 --name solr solr:latest docker run -d -p 8983:8983 -v /opt/solr:/…

操作系统入门系列-MIT6.828(操作系统工程)学习笔记(五)---- 操作系统的组织结构(OS design)

系列文章目录 操作系统入门系列-MIT6.S081&#xff08;操作系统&#xff09;学习笔记&#xff08;一&#xff09;---- 操作系统介绍与接口示例 操作系统入门系列-MIT6.828&#xff08;操作系统工程&#xff09;学习笔记&#xff08;二&#xff09;----课程实验环境搭建&#x…

网工内推 | 深信服、中软国际技术支持工程师,最高13k*13薪

01 深信服 &#x1f537;招聘岗位&#xff1a;远程技术支持工程师 &#x1f537;任职要求&#xff1a; 一、专业能力和行业经验&#xff1a; ①具备友商同岗位工作经验1.5年以上&#xff0c;具备良好的分析和判断能力&#xff0c;有独立问题处理思路&#xff0c;具备常见协…

python中魔术方法__str__与__repr__的区别

在Python中&#xff0c;__str__和__repr__是两个常见的魔法方法&#xff08;也称为双下方法或dunder方法&#xff09;&#xff0c;它们用于定义对象的字符串表示形式。它们的主要区别在于它们的用途和使用场景。 __str__ 用途&#xff1a;__str__方法用于为用户提供一个易读的…

【嵌入式DIY实例】-Nokia 5110显示DHT11/DHT22传感器数据

Nokia 5110显示DHT11/DHT22传感器数据 文章目录 Nokia 5110显示DHT11/DHT22传感器数据1、硬件准备2、代码实现2.1 显示DHT11数据2.2 显示DHT22数据本文介绍如何将 ESP8266 NodeMCU 开发板 (ESP-12E) 与 DHT11 数字湿度和温度传感器以及诺基亚 5110 LCD 连接。 NodeMCU 从 DHT11…

.NET Core 服务注册步骤总结

总结一下 .NET Core 服务注册的步骤&#xff1a; .NET Core Web Api 项目服务注册步骤&#xff1a; 创建一个接口&#xff0c;和实现类 比如&#xff1a;IMyService, CnService 在 Program.cs 的 var app builder.Build(); 语句之前加上&#xff1a; var builder WebApplic…

【面经总结】 Java基础 - 异常

异常 介绍一下 Java 的异常体系 Java 的异常体系是由 Throwable 类及其子类构成的。 Throwable 包含两个子类&#xff1a;Error&#xff08;错误&#xff09;和 Exception&#xff08;异常&#xff09; Error 表示错误&#xff0c;通常不需要程序员处理&#xff0c;如内存溢…

python中的turtle

turtle个别指令 初始箭头默认指向为东&#xff08;右&#xff09; 往前&#xff08;右&#xff09;三个格&#xff1a;turtle.forward(3) 往后&#xff08;左&#xff09;三个格&#xff1a;turtle.backward(3) 往左转90度&#xff1a;turtle.left(90) 往右转90度&#xf…

Attention与轻量级ResNet融合,低资源消耗下实现效率和性能完美平衡

注意力机制通过让模型关注图像关键区域提升了识别精度&#xff0c;而轻量级残差网络通过减少参数和计算量&#xff0c;实现了在低资源消耗下的优秀性能。 结合注意力机制与轻量级残差网络&#xff0c;既能让模型能够更高效地关注输入数据中的关键信息&#xff0c;提升模型处理…

vs调试时无法找到文件-chromium源码编译

一直跟着教程走结果报错了&#xff0c;找了半天的教程无法解决&#xff0c;于是乎只好重来&#xff0c;因为这个是属于项目调试&#xff0c;报错了可以重新编译项目就好。在重新做的过程中发现路径写错了

人工智能的等价形式

经典的人工智能&#xff0c;采用“梯度下降法”&#xff0c;运算量很大&#xff0c;约是esp2。其中e是epoch&#xff0c;训练的周期数&#xff1b;s是sample&#xff0c;训练样本的数量&#xff1b;p是parameter&#xff0c;参数的数量。 人工智能有等价形式&#xff0c;它不需…

DPI简析

DPI简析 一、DPI与PPI二、硬件设备的DPI2.1打印机DPI2.2显示器DPI2.2.1显示器DPI计算2.2.2显示器分辨率与系统分辨率2.2.3常见分辨率 2.3鼠标DPI 三、图片DPI3.1图片DPI与打印尺寸3.1.1图片打印尺寸计算3.1.2常用的照片尺寸及DPI 3.2图片DPI与屏幕显示3.3修改图片DPI 参考文档 …

Windos10上Podman安装运行mysql8

记录以下在windows10系统上Podman v5.1.1安装MySQL8全过程。 目录 一、拉取mysql8镜像二、创建宿主目录三、创建 my.cnf文件四、创建Mysql8容器五、windows上Podman安装运行mysql8失败问题描述 解决办法① 通过PowerShell进入wsl② 修改wsl系统配置③ 重启wsl&#xff0c;Podma…

3个月搞定计算机二级C语言!高效刷题系列进行中

文章目录 前言备考计算机二级C语言为什么考二级C语言&#xff1f;刷题总结后发布系列文章后记免责声明 前言 大家好&#xff0c;我是梁国庆。 计算机二级应该是每一位大学生的必修课&#xff0c;相信很多同学的大学flag中都会有它的身影。 我在大学里也不止一次的想要考计算…

【运维知识大神篇】运维界的超神器Kubernetes教程14(RBAC三种认证方式详解:基于用户+基于用户组+基于服务账号)

本篇文章继续给大家介绍Kubernetes&#xff0c;内容依旧烧脑&#xff0c;不过内容也已经过了一大半了&#xff0c;如果你把我Kubernetes前面的教程都看懂了的话&#xff0c;那么你已经很厉害了&#xff0c;坚持下去&#xff01;本篇文章主要介绍RBAC的三种认证方式&#xff0c;…