当前位置: 首页 > news >正文

Verilog:【8】基于FPGA实现SD NAND FLASH的SPI协议读写

Verilog:【8】基于FPGA实现SD NAND FLASH的SPI协议读写

  在此介绍的是使用FPGA实现SD NAND FLASH的读写操作,以雷龙发展提供的CS创世SD NAND FLASH样品为例,分别讲解电路连接、读写时序与仿真和实验结果。

目录

  1 FLASH背景介绍

  2 样品申请

  3 电路结构与接口协议

  3.1 SD NAND

  3.2 SD NAND测试板

  3.3 FPGA开发板

  4 SD卡协议与时序流程

  4.1 SD卡协议

  4.2 SD卡2.0版本初始化步骤

  4.3 SD卡的读步骤

  4.4 SD卡的写步骤

  5 模块代码

  5.1 sd_card_top

  5.2 sd_card_cmd

  5.3 sd_card_sec_read_write

  5.4 spi_master

  5.5 其余代码

  5.5.1 sd_card_test

  5.5.2 ax_debounce

  5.5.3 seg_decoder

  5.5.4 seg_scan

  6 实验结果

  

  使用FPGA讲解SD NAND FLASH的文章网上也有很多比较详实的内容,本文的部分思路也是参考了其他博主的博客思路。

  1 FLASH背景介绍

简介

Flash是近些年应用最广、速度最快的只读存储器,原理是从 EEPROM 基础上改进发展来的,特点是擦除和编程速度快,因此得名为闪速(或闪烁)存储器,简称闪存。

NOR Flash 和 NAND Flash 是现在市场上两种主要的闪存技术。Intel于1988年首先开发出 NOR Flash 技术,彻底改变了原先由 EPROM 和 EEPROM 一统天下的局面。紧接着,1989年,东芝公司发表了 NAND Flash 结构,后者的单元电路尺寸几乎只是 NOR 器件的一半,可以在给定的芯片尺寸内提供更高的容量,也就相应地降低了价格。

NOR Flash 的特点是以字节为单位随机存取。这样,应用程序可以直接在 Flash中执行,不必再把程序代码预先读到 RAM 中。NOR Flash的接口一般以SPI为主,与通常的扩展存储器一样,可以直接连接到处理器的外围总线上。

NAND Flash应该是目前最热门的存储芯片了。因为我们生活中经常使用的电子产品都会涉及到它。比如你买手机,肯定会考虑64GB,还是256GB?买笔记本是买256GB,还是512GB容量的硬盘呢?(目前电脑大部分采用了基于 NAND Flash 产品的固态硬盘)。

NAND Flash 主要分为SLC,MLC,TLC,3D TLC ,3DQLC等,随时科技的发展和大众的需求,单位面积内的存储容量越来越大。SLC是指单个存储单元中,能容纳1bit 代表2种状态,0或者1.  MLC 则是指单个存储单元中,能容纳2bit,代表4种状态 ,00,01,10,11。 TLC 则是指单个存储单元中,能容纳3bit,代表8种状态,000,001,010,011,100,101,110,111。最开始整个存储单元是2D展开的,也就是平面的,随着需要在单位空间内容纳更多的信息,就开始类似盖楼房一样,在3D也就是立体的方面来发展了。

如果在产品中选择nor flash 还是NAND FLASH,更多的时候是从容量角度来考量,如果存储的内容大于128Mbit 就选择NAND Flash ,小于128Mbit就选择 Nor flash。Nor flash受限于自己的工艺,大于128Mbit的容量,价格就会比128MB SLC NANDFLASH的价格还要贵。

未来发展

当前,NAND flash正在从 2D 发展到 3D。对于 2D NAND flash,如果在同一区域实现更多的单元数量,形成更小的工作区和栅级,便能增大存储容量。直至 2010 年初,2D NAND flash中的扩展一直是这项技术的主要焦点所在;然而,由于内部结构的限制,且储存数据会随时间推移而丢失导致使用寿命缩短,2D的技术已无法再实现扩展。

因此,3D NAND flash逐渐取而代之,成为业界关注焦点,现在所有 NAND 制造商都在开发和制造 3D NAND flash产品。

在 3D NAND flash 的结构中,存储容量会随着三维叠层中堆叠层数的增加而变大,类似盖楼房,一层一层叠加上去。3D NAND flash 使用了堆叠多层氮氧化物的方法,形成一个被称为“塞子”的垂直深孔,在其中形成一个由氧化物-氮化物-氧化物制成的存储设备。通过这种方法,仅需少量工艺即可同时形成大量单元。在 3D NAND flash 中,电流通过位于圆柱单元中心的多晶硅通道,便能根据存储在氮化硅中的电荷类型实现存储编程和擦除信息。虽然2D NAND flash 技术发展的目标是实现形成较小的单元, 3D NAND flash 的核心技术却是实现更多层数的三维堆叠。  

由于NAND FLASH在大容量应用中的便利性,因此作为今天介绍的主角~

  什么是SD NAND呢(以下省略FLASH)?下面的内容是从雷龙发展官网的介绍中得到:

  SD NAND俗称贴片式TF卡,尽管与TF卡名称类似,但是有较大的区别:

  相比常见的TF卡,SD NAND是专门为内置存储进行设计,焊接在PCB板上以供工业级产品的应用。因此对品质稳定性、一致性、以及尺寸都有较高的要求。

  

  2 样品申请

  本文所使用的CS创世SD NAND是从深圳雷龙发展申请获得,可以在官网中最上面找到申请样品的入口:

  雷龙发展有限公司创立于2008年,专注NAND Flash领域13年。创始人均为步步高/华为技术背景出身。如果有技术问题也可以和其公司人员进行沟通,相关的工作人员非常专业和热心。

  

  3 电路结构与接口协议

  3.1 SD NAND

  本文所使用的产品是CSNP4GCR01-AMW,是雷龙的第二代产品,产品如下图所示:

  数据手册可以在立创商城进行下载,其封装与连接的电路原理参考图如下图所示:

  芯片共包含8个引脚,包括4根数据线(6、7、1、2);2根电源线(4、8);1根时钟线(3);1根命令控制线(5)

  手册中提供了SD NAND的两种使用模式,分别为SD MODE 以及 SPI MODE。他们所对应的引脚定义,如下图所示:

  对于两种模式的切换,官方给出了初始化的方式。下文在代码的时序部分也会涉及到相关内容。

  在对SD卡数据读写速度要求不高的情况下,选用SPI通信模式可以说是一种最佳方案。因为在该模式下,同只需要通过四根线就是可以完成所有的数据交换,可以为我们节省出宝贵的FPGA I/O资源。下图给出了SPI一对一通信时,主设备与从设备之间的连接关系。

  因此本文主要介绍SPI MODE下各个引脚的功能:

 

  确定了通讯模式后,也就便于我们后文中,利用这种通讯模式按照SD卡的读写时序进行读写操作。

  3.2 SD NAND测试板

  单独的SD NAND不便于我们使用FPGA进行读写测试,好在官方提供了测试板,如下图所示:

  有了它就可以轻松实现SD NAND与我们常见的FPGA开发板上的Micro SD插槽进行连接与测试了。

  适用产品:LGA8,6x8mm 封装的SD NAND产品。

  测试板尺寸:长度6.22厘米,宽度2.49厘米,接口长度2.53厘米。

  使用方法:将芯片焊接至测试板上,可在原有的Micro SD卡座上直接调试和测试。

  准备工具:热风枪,烙铁,锡膏,镊子。

  焊接方式: 先用烙铁将芯片的8个PIN脚上锡,中间的一个PIN脚不需要上锡保持NC即可。再将接板板上,对应芯片的8个PIN上锡。

  最后用镊子将芯片放到PCB上,热风枪温度调至350℃ 在芯片表面均匀加热即可焊接。

  

       其它事项:测试板上其它元器件无需理会,直接将芯片焊接在测试板上即可当SD卡一样调试。

  焊接好后,可以将转接板插入到读卡器,再将读卡器连接到电脑上看看是否能正确识别到容量,通过这个方式来判断芯片是否已经焊接正常。

  3.3 FPGA开发板

  本文所使用的是黑金的AX301开发板,上面装有一个 Micro SD 卡座, FPGA 通过 SPI 数据总线访问 Micro SD 卡,SD 卡座和 FPGA 的硬件电路连接如下:

  借由硬件电路的连接,FPGA可以直接与我们的SD NAND进行通信了。

  至此,我们已经实现了SD NANDSPI通信方式方案的确定以及基于此的硬件电路连接,下一步就是根据SD卡的读写时序讲通信方式初始化为SPI模式,并按照SD卡协议进行读写操作。

  4 SD卡协议与时序流程

  4.1 SD卡协议

  以下内容来自黑金的实验手册:

  SD 卡的协议是一种简单的命令/响应的协议。全部命令由主机发起, SD 卡接收到命令后并返

  回响应数据。根据命令的不同,返回的数据内容和长度也不同。 SD 卡命令是一个 6 字节组成的命

  令包,其中第一个字节为命令号, 命令号高位 bit7 和 bit6 为固定的“01“,其它 6 个 bit 为具体

  的命令号。第 2 个字节到第 5 个字节为命令参数。第 6 个字节为 7 个 bit 的 CRC 校验加 1 个 bit 的结束位。 如果在 SPI 模式的时候, CRC 校验位为可选。 如下图所示, Command 表示命令,通常使用十进制表示名称,例如 CMD17,这个时候 Command 就是十进制的 17。

  对于详细的SD卡协议内容,可以参考传送门中的相关内容,给出了比较具体的解释。

  SD 卡对每个命令会返回一个响应,每个命令有一定的响应格式。响应的格式跟给它的命令号

  有关。在 SPI 模式中,有三种响应格式: R1, R2, R3。

  在进行SD NAND的SPI模式读写操作时,主要使用到了以下几种SD卡命令,下面的表格进行简单介绍,这里可以找到完整版:

  4.2 SD卡2.0版本初始化步骤

  上电后延时至少 74clock,等待 SD 卡内部操作完成

  片选 CS 低电平选中 SD 卡

  发送 CMD0,需要返回 0x01,进入 Idle 状态

  为了区别 SD 卡是 2.0 还是 1.0,或是 MMC 卡,这里根据协议向上兼容的,首先发送只有SD2.0 才有的命令 CMD8,如果 CMD8 返回无错误,则初步判断为 2.0 卡,进一步循环发送命令 CMD55+ACMD41,直到返回 0x00,确定 SD2.0 卡

  如果 CMD8 返回错误则判断为 1.0 卡还是 MMC 卡,循环发送 CMD55+ACMD41,返回无错误,则为 SD1.0 卡,到此 SD1.0 卡初始成功,如果在一定的循环次数下,返回为错误,则进一步发送 CMD1 进行初始化,如果返回无错误,则确定为 MMC 卡,如果在一定的次数下,返回为错误,则不能识别该卡,初始化结束。 (通过 CMD16 可以改变 SD 卡一次性读写的长度)

  CS 拉高

  4.3 SD卡的读步骤

  发送 CMD17(单块)或 CMD18(多块)读命令,返回 0X00

  接收数据开始令牌 fe(或 fc) +正式数据 512Bytes + CRC 校验 2Bytes(默认正式传输的数据长度是 512Bytes)

  4.4 SD卡的写步骤

  发送 CMD24(单块)或 CMD25(多块)写命令,返回 0X00

  发送数据开始令牌 fe(或 fc) +正式数据 512Bytes + CRC 校验 2Bytes

  5 模块代码

  本代码所实现的功能,是基于黑金AX301B,实现对SD NAND FLASH的数据写入与读取,并显示在开发板的数码管上。当按下开发板上的按键时,会自动将数据加一操作,并进行同步显示。

  前文介绍的是SD NAND的协议以及初始化、读写操作的流程,下面介绍代码的组成部分,整个工程主要由以下部分模块构成:

  sd_card_test(top模块)

  ax_debounce:ax_debounce_m0(按键消抖模块)

  sd_card_top:sd_card_top_m0(SD卡top模块)

  sd_card_cmd:sd_card_cmd_m0(SD卡指令)

  sd_card_sec_read_write:sd_card_sec_read_write_m0(SD卡读写)

  spi_master:spi_master_m0(SPI一个字节读写)

  seg_decoder:seg_decoder_m0(数码管控制)

  seg_decoder:seg_decoder_m1(数码管控制)

  seg_scan:seg_scan_m0(数码管控制)

  下面主要介绍上述四个加粗的模块以及其功能

  5.1 sd_card_top

  本模块是SD card的top模块,用来实现不同子模块之间的连接。

//

//                                                                              //

//                                                                              //

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//==========================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//--------------------------------------------------------------------------

//  2017/6/21    meisq         1.0         Original

//*************************************************************************/

module sd_card_top

#(

parameter  SPI_LOW_SPEED_DIV = 248,         // SD card low speed mode frequency division parameter,spi clk speed = clk speed /((SPI_LOW_SPEED_DIV + 2) * 2 )

parameter  SPI_HIGH_SPEED_DIV = 0           // SD card high speed mode frequency division parameter,spi clk speed = clk speed /((SPI_HIGH_SPEED_DIV + 2) * 2 )

)

(

input            clk,

input            rst,

output           SD_nCS,                    //SD card chip select (SPI mode)

output           SD_DCLK,                   //SD card clock

output           SD_MOSI,                   //SD card controller data output

input            SD_MISO,                   //SD card controller data input

output           sd_init_done,              //SD card initialization is complete

input            sd_sec_read,               //SD card sector read

input[31:0]      sd_sec_read_addr,          //SD card sector read address

output[7:0]      sd_sec_read_data,          //SD card sector read data

output           sd_sec_read_data_valid,    //SD card sector read data valid

output           sd_sec_read_end,           //SD card sector read end

input            sd_sec_write,              //SD card sector write

input[31:0]      sd_sec_write_addr,         //SD card sector write address

input[7:0]       sd_sec_write_data,         //SD card sector write data

output           sd_sec_write_data_req,     //SD card sector write data next clock is valid

output           sd_sec_write_end           //SD card sector write end

);

wire[15:0]           spi_clk_div;               //SPI module clock division parameter

wire                 cmd_req;                   //SD card command request

wire                 cmd_req_ack;               //SD card command request response

wire                 cmd_req_error;             //SD card command request error

wire[47:0]           cmd;                       //SD card command

wire[7:0]            cmd_r1;                    //SD card expect response

wire[15:0]           cmd_data_len;              //SD card command read data length

wire                 block_read_req;            //SD card sector data read request

wire                 block_read_valid;          //SD card sector data read data valid

wire[7:0]            block_read_data;           //SD card sector data read data

wire                 block_read_req_ack;        //SD card sector data read response

wire                 block_write_req;           //SD card sector data write request

wire[7:0]            block_write_data;          //SD card sector data write data next clock is valid

wire                 block_write_data_rd;       //SD card sector data write data

wire                 block_write_req_ack;       //SD card sector data write response

wire                 nCS_ctrl;                  //SPI module chip select control

wire                 spi_wr_req;                //SPI module data sending request

wire                 spi_wr_ack;                //SPI module data request response

wire[7:0]            spi_data_in;               //SPI module send data

wire[7:0]            spi_data_out;              //SPI module data returned

wire[15:0]           clk_div;

sd_card_sec_read_write

#(

.SPI_LOW_SPEED_DIV(SPI_LOW_SPEED_DIV),

.SPI_HIGH_SPEED_DIV(SPI_HIGH_SPEED_DIV)

)

sd_card_sec_read_write_m0(

.clk                            (clk                    ),

.rst                            (rst                    ),

.sd_init_done                   (sd_init_done           ),

.sd_sec_read                    (sd_sec_read            ),

.sd_sec_read_addr               (sd_sec_read_addr       ),

.sd_sec_read_data               (sd_sec_read_data       ),

.sd_sec_read_data_valid         (sd_sec_read_data_valid ),

.sd_sec_read_end                (sd_sec_read_end        ),

.sd_sec_write                   (sd_sec_write           ),

.sd_sec_write_addr              (sd_sec_write_addr      ),

.sd_sec_write_data              (sd_sec_write_data      ),

.sd_sec_write_data_req          (sd_sec_write_data_req  ),

.sd_sec_write_end               (sd_sec_write_end       ),

.spi_clk_div                    (spi_clk_div            ),

.cmd_req                        (cmd_req                ),

.cmd_req_ack                    (cmd_req_ack            ),

.cmd_req_error                  (cmd_req_error          ),

.cmd                            (cmd                    ),

.cmd_r1                         (cmd_r1                 ),

.cmd_data_len                   (cmd_data_len           ),

.block_read_req                 (block_read_req         ),

.block_read_valid               (block_read_valid       ),

.block_read_data                (block_read_data        ),

.block_read_req_ack             (block_read_req_ack     ),

.block_write_req                (block_write_req        ),

.block_write_data               (block_write_data       ),

.block_write_data_rd            (block_write_data_rd    ),

.block_write_req_ack            (block_write_req_ack    )

);

sd_card_cmd sd_card_cmd_m0(

.sys_clk                        (clk                    ),

.rst                            (rst                    ),

.spi_clk_div                    (spi_clk_div            ),

.cmd_req                        (cmd_req                ),

.cmd_req_ack                    (cmd_req_ack            ),

.cmd_req_error                  (cmd_req_error          ),

.cmd                            (cmd                    ),

.cmd_r1                         (cmd_r1                 ),

.cmd_data_len                   (cmd_data_len           ),

.block_read_req                 (block_read_req         ),

.block_read_req_ack             (block_read_req_ack     ),

.block_read_data                (block_read_data        ),

.block_read_valid               (block_read_valid       ),

.block_write_req                (block_write_req        ),

.block_write_data               (block_write_data       ),

.block_write_data_rd            (block_write_data_rd    ),

.block_write_req_ack            (block_write_req_ack    ),

.nCS_ctrl                       (nCS_ctrl               ),

.clk_div                        (clk_div                ),

.spi_wr_req                     (spi_wr_req             ),

.spi_wr_ack                     (spi_wr_ack             ),

.spi_data_in                    (spi_data_in            ),

.spi_data_out                   (spi_data_out           )

);

spi_master spi_master_m0(

.sys_clk                        (clk                    ),

.rst                            (rst                    ),

.nCS                            (SD_nCS                 ),

.DCLK                           (SD_DCLK                ),

.MOSI                           (SD_MOSI                ),

.MISO                           (SD_MISO                ),

.clk_div                        (clk_div                ),

.CPOL                           (1'b1                   ),

.CPHA                           (1'b1                   ),

.nCS_ctrl                       (nCS_ctrl               ),

.wr_req                         (spi_wr_req             ),

.wr_ack                         (spi_wr_ack             ),

.data_in                        (spi_data_in            ),

.data_out                       (spi_data_out           )

);

endmodule

  5.2 sd_card_cmd

  sd_card_cmd 模块主要实验 sd 卡基本命令操作,还有上电初始化的 88 个周期的时钟,数据

  块的读写,状态机如下:

  代码如下:

//

//                                                                              //

//                                                                              //

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//==========================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//--------------------------------------------------------------------------

//  2017/6/21     meisq         1.0         Original

//*************************************************************************/

module sd_card_cmd(

input                       sys_clk,

input                       rst,

input[15:0]                 spi_clk_div,                  //SPI module clock division parameter

input                       cmd_req,                      //SD card command request

output                      cmd_req_ack,                  //SD card command request response

output reg                  cmd_req_error,                //SD card command request error

input[47:0]                 cmd,                          //SD card command

input[7:0]                  cmd_r1,                       //SD card expect response

input[15:0]                 cmd_data_len,                 //SD card command read data length

input                       block_read_req,               //SD card sector data read request

output reg                  block_read_valid,             //SD card sector data read data valid

output reg[7:0]             block_read_data,              //SD card sector data read data

output                      block_read_req_ack,           //SD card sector data read response

input                       block_write_req,              //SD card sector data write request

input[7:0]                  block_write_data,             //SD card sector data write data next clock is valid

output                      block_write_data_rd,          //SD card sector data write data

output                      block_write_req_ack,          //SD card sector data write response

output                      nCS_ctrl,                     //SPI module chip select control

output reg[15:0]            clk_div,

output reg                  spi_wr_req,                   //SPI module data sending request

input                       spi_wr_ack,                   //SPI module data request response

output[7:0]                 spi_data_in,                  //SPI module send data

input[7:0]                  spi_data_out                  //SPI module data returned

);

parameter S_IDLE         = 0;

parameter S_WAIT         = 1;

parameter S_INIT         = 2;

parameter S_CMD_PRE      = 3;

parameter S_CMD          = 4;

parameter S_CMD_DATA     = 5;

parameter S_READ_WAIT    = 6;

parameter S_READ         = 7;

parameter S_READ_ACK     = 8;

parameter S_WRITE_TOKEN  = 9;

parameter S_WRITE_DATA_0 = 10;

parameter S_WRITE_DATA_1 = 11;

parameter S_WRITE_CRC    = 12;

parameter S_WRITE_ACK    = 13;

parameter S_ERR          = 14;

parameter S_END          = 15;

reg[3:0]                      state;

reg                           CS_reg;

reg[15:0]                     byte_cnt;

reg[7:0]                      send_data;

wire[7:0]                     data_recv;

reg[9:0]                      wr_data_cnt;

assign cmd_req_ack = (state == S_END);

assign block_read_req_ack = (state == S_READ_ACK);

assign block_write_req_ack= (state == S_WRITE_ACK);

assign block_write_data_rd = (state == S_WRITE_DATA_0);

assign spi_data_in = send_data;

assign data_recv = spi_data_out;

assign nCS_ctrl = CS_reg;

always@(posedge sys_clk or posedge rst)

begin

if(rst == 1'b1)

begin

CS_reg <= 1'b1;

spi_wr_req <= 1'b0;

byte_cnt <= 16'd0;

clk_div <= 16'd0;

send_data <= 8'hff;

state <= S_IDLE;

cmd_req_error <= 1'b0;

wr_data_cnt <= 10'd0;

end

else

case(state)

S_IDLE:

begin

state <= S_INIT;

clk_div <= spi_clk_div;

CS_reg <= 1'b1;

end

S_INIT:

begin

//send 11 bytes on power(at least 74 SPI clocks)

if(spi_wr_ack == 1'b1)

begin

if(byte_cnt >= 16'd10)

begin

byte_cnt <= 16'd0;

spi_wr_req <= 1'b0;

state <= S_WAIT;

end

begin

byte_cnt <= byte_cnt + 16'd1;

end

end

else

begin

spi_wr_req <= 1'b1;

send_data <= 8'hff;

end

end

S_WAIT:

begin

cmd_req_error <= 1'b0;

wr_data_cnt <= 10'd0;

//wait for  instruction

if(cmd_req == 1'b1)

state <= S_CMD_PRE;

else if(block_read_req == 1'b1)

state <= S_READ_WAIT;

else if(block_write_req == 1'b1)

state <= S_WRITE_TOKEN;

clk_div <= spi_clk_div;

end

S_CMD_PRE:

begin

//before sending a command, send an byte 'ff',provide some clocks

if(spi_wr_ack == 1'b1)

begin

state <= S_CMD;

spi_wr_req <= 1'b0;

byte_cnt <= 16'd0;

end

else

begin

spi_wr_req <= 1'b1;

CS_reg <= 1'b1;

send_data <= 8'hff;

end

end

S_CMD:

begin

if(spi_wr_ack == 1'b1)

begin

if((byte_cnt == 16'hffff) || (data_recv != cmd_r1 && data_recv[7] == 1'b0))

begin

state <= S_ERR;

spi_wr_req <= 1'b0;

end

else if(data_recv == cmd_r1)

begin

spi_wr_req <= 1'b0;

if(cmd_data_len != 16'd0)

begin

state <= S_CMD_DATA;

byte_cnt <= 16'd0;

end

else

state <= S_END;

end

else

byte_cnt <=  byte_cnt + 16'd1;

end

else

begin

spi_wr_req <= 1'b1;

CS_reg <= 1'b0;

if(byte_cnt == 16'd0)

send_data <= (cmd[47:40] | 8'h40);

else if(byte_cnt == 16'd1)

send_data <= cmd[39:32];

else if(byte_cnt == 16'd2)

send_data <= cmd[31:24];

else if(byte_cnt == 16'd3)

send_data <= cmd[23:16];

else if(byte_cnt == 16'd4)

send_data <= cmd[15:8];

else if(byte_cnt == 16'd5)

send_data <= cmd[7:0];

else

send_data <= 8'hff;

end

end

S_CMD_DATA:

begin

if(spi_wr_ack == 1'b1)

begin

if(byte_cnt == cmd_data_len - 16'd1)

begin

state <= S_END;

spi_wr_req <= 1'b0;

byte_cnt <= 16'd0;

end

else

begin

byte_cnt <= byte_cnt + 16'd1;

end

end

else

begin

spi_wr_req <= 1'b1;

send_data <= 8'hff;

end

end

S_READ_WAIT:

begin

if(spi_wr_ack == 1'b1 && data_recv == 8'hfe)

begin

spi_wr_req <= 1'b0;

state <= S_READ;

byte_cnt <= 16'd0;

end

else

begin

spi_wr_req <= 1'b1;

send_data <= 8'hff;

end

end

S_READ:

begin

if(spi_wr_ack == 1'b1)

begin

if(byte_cnt == 16'd513)

begin

state <= S_READ_ACK;

spi_wr_req <= 1'b0;

byte_cnt <= 16'd0;

end

else

begin

byte_cnt <= byte_cnt + 16'd1;

end

end

else

begin

spi_wr_req <= 1'b1;

send_data <= 8'hff;

end

end

S_WRITE_TOKEN:

if(spi_wr_ack == 1'b1)

begin

state <= S_WRITE_DATA_0;

spi_wr_req <= 1'b0;

end

else

begin

spi_wr_req <= 1'b1;

send_data <= 8'hfe;

end

S_WRITE_DATA_0:

begin

state <= S_WRITE_DATA_1;

wr_data_cnt <= wr_data_cnt + 10'd1;

end

S_WRITE_DATA_1:

begin

if(spi_wr_ack == 1'b1 && wr_data_cnt == 10'd512)

begin

state <= S_WRITE_CRC;

spi_wr_req <= 1'b0;

end

else if(spi_wr_ack == 1'b1)

begin

state <= S_WRITE_DATA_0;

spi_wr_req <= 1'b0;

end

else

begin

spi_wr_req <= 1'b1;

send_data <= block_write_data;

end

end

S_WRITE_CRC:

begin

if(spi_wr_ack == 1'b1)

begin

if(byte_cnt == 16'd2)

begin

state <= S_WRITE_ACK;

spi_wr_req <= 1'b0;

byte_cnt <= 16'd0;

end

else

begin

byte_cnt <= byte_cnt + 16'd1;

end

end

else

begin

spi_wr_req <= 1'b1;

send_data <= 8'hff;

end

end

S_ERR:

begin

state <= S_END;

cmd_req_error <= 1'b1;

end

S_READ_ACK,S_WRITE_ACK,S_END:

begin

state <= S_WAIT;

end

default:

state <= S_IDLE;

endcase

end

always@(posedge sys_clk or posedge rst)

begin

if(rst == 1'b1)

block_read_valid <= 1'b0;

else if(state == S_READ && byte_cnt < 16'd512)

block_read_valid <= spi_wr_ack;

else

block_read_valid <= 1'b0;

end

always@(posedge sys_clk or posedge rst)

begin

if(rst == 1'b1)

block_read_data <= 8'd0;

else if(state == S_READ && spi_wr_ack == 1'b1)

block_read_data <= data_recv;

end

endmodule

  5.3 sd_card_sec_read_write

  sd_card_sec_read_write 模块继续完成 SD 卡初始化,然后等待扇区读写指令,并完成扇区的

  读写操作。 下图为模块的状态机转换图,首先发送 CMD0 命令,然后发送 CMD8 命令,再发送

  CMD55,接着发送 ACMD41,如果应答正常, sd 卡初始化完成,等待扇区的读写。

  代码如下:

//

//                                                                              //

//                                                                              //

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//===============================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//-------------------------------------------------------------------------------

//  2017/6/21     meisq         1.0         Original

//*******************************************************************************/

module sd_card_sec_read_write

#(

parameter  SPI_LOW_SPEED_DIV = 248,         // spi clk speed = clk speed /((SPI_LOW_SPEED_DIV + 2) * 2 )

parameter  SPI_HIGH_SPEED_DIV = 0           // spi clk speed = clk speed /((SPI_HIGH_SPEED_DIV + 2) * 2 )

)

(

input            clk,

input            rst,

output reg       sd_init_done,

input            sd_sec_read,

input[31:0]      sd_sec_read_addr,

output[7:0]      sd_sec_read_data,

output           sd_sec_read_data_valid,

output           sd_sec_read_end,

input            sd_sec_write,

input[31:0]      sd_sec_write_addr,

input[7:0]       sd_sec_write_data,

output           sd_sec_write_data_req,

output           sd_sec_write_end,

output reg[15:0] spi_clk_div,

output reg       cmd_req,

input            cmd_req_ack,

input            cmd_req_error,

output reg[47:0] cmd,

output reg[7:0]  cmd_r1,

output reg[15:0] cmd_data_len,

output reg       block_read_req,

input            block_read_valid,

input[7:0]       block_read_data,

input            block_read_req_ack,

output reg       block_write_req,

output[7:0]      block_write_data,

input            block_write_data_rd,

input            block_write_req_ack

);

reg[7:0] read_data;

reg[31:0] timer;

localparam S_IDLE               = 0;

localparam S_CMD0               = 1;

localparam S_CMD8               = 2;

localparam S_CMD55              = 3;

localparam S_CMD41              = 4;

localparam S_CMD17              = 5;

localparam S_READ               = 6;

localparam S_CMD24              = 7;

localparam S_WRITE              = 8;

localparam S_ERR                = 14;

localparam S_WRITE_END          = 15;

localparam S_READ_END           = 16;

localparam S_WAIT_READ_WRITE    = 17;

localparam S_CMD16              = 18;

reg[4:0]                       state;

reg[31:0]                      sec_addr;

assign sd_sec_read_data_valid = (state == S_READ) && block_read_valid;

assign sd_sec_read_data = block_read_data;

assign sd_sec_read_end = (state == S_READ_END);

assign sd_sec_write_data_req = (state == S_WRITE) && block_write_data_rd;

assign block_write_data = sd_sec_write_data;

assign sd_sec_write_end = (state == S_WRITE_END);

always@(posedge clk or posedge rst)

begin

if(rst == 1'b1)

begin

state <= S_IDLE;

cmd_req <= 1'b0;

cmd_data_len <= 16'd0;

cmd_r1 <= 8'd0;

cmd <= 48'd0;

spi_clk_div <= SPI_LOW_SPEED_DIV[15:0];

block_write_req <= 1'b0;

block_read_req <= 1'b0;

sec_addr <= 32'd0;

sd_init_done <= 1'b0;

end

else

case(state)

S_IDLE:

begin

state <= S_CMD0;

sd_init_done <= 1'b0;

spi_clk_div <= SPI_LOW_SPEED_DIV[15:0];

end

S_CMD0:

begin

if(cmd_req_ack & ~cmd_req_error)

begin

state <= S_CMD8;

cmd_req <= 1'b0;

end

else

begin

cmd_req <= 1'b1;

cmd_data_len <= 16'd0;

cmd_r1 <= 8'h01;

cmd <= {8'd0,8'h00,8'h00,8'h00,8'h00,8'h95};

end

end

S_CMD8:

begin

if(cmd_req_ack & ~cmd_req_error)

begin

state <= S_CMD55;

cmd_req <= 1'b0;

end

else

begin

cmd_req <= 1'b1;

cmd_data_len <= 16'd4;

cmd_r1 <= 8'h01;

cmd <= {8'd8,8'h00,8'h00,8'h01,8'haa,8'h87};

end

end

S_CMD55:

begin

if(cmd_req_ack & ~cmd_req_error)

begin

state <= S_CMD41;

cmd_req <= 1'b0;

end

else

begin

cmd_req <= 1'b1;

cmd_data_len <= 16'd0;

cmd_r1 <= 8'h01;

cmd <= {8'd55,8'h00,8'h00,8'h00,8'h00,8'hff};

end

end

S_CMD41:

begin

if(cmd_req_ack & ~cmd_req_error)

begin

state <= S_CMD16;

cmd_req <= 1'b0;

sd_init_done <= 1'b1;

spi_clk_div <= SPI_HIGH_SPEED_DIV[15:0];

end

else if(cmd_req_ack)

begin

state <= S_CMD55;

end

else

begin

cmd_req <= 1'b1;

cmd_data_len <= 16'd0;

cmd_r1 <= 8'h00;

cmd <= {8'd41,8'h40,8'h00,8'h00,8'h00,8'hff};

end

end

S_CMD16:

begin

if(cmd_req_ack & ~cmd_req_error)

begin

state <= S_WAIT_READ_WRITE;

cmd_req <= 1'b0;

sd_init_done <= 1'b1;

spi_clk_div <= SPI_HIGH_SPEED_DIV[15:0];

end

else if(cmd_req_ack)

begin

state <= S_CMD55;

end

else

begin

cmd_req <= 1'b1;

cmd_data_len <= 16'd0;

cmd_r1 <= 8'h00;

cmd <= {8'd16,32'd512,8'hff};

end

end

S_WAIT_READ_WRITE:

begin

if(sd_sec_write ==  1'b1)

begin

state <= S_CMD24;

sec_addr <= sd_sec_write_addr;

end

else if(sd_sec_read == 1'b1)

begin

state <= S_CMD17;

sec_addr <= sd_sec_read_addr;

end

spi_clk_div <= 16'd0;

end

S_CMD24:

begin

if(cmd_req_ack & ~cmd_req_error)

begin

state <= S_WRITE;

cmd_req <= 1'b0;

end

else

begin

cmd_req <= 1'b1;

cmd_data_len <= 16'd0;

cmd_r1 <= 8'h00;

cmd <= {8'd24,sec_addr,8'hff};

end

end

S_WRITE:

begin

if(block_write_req_ack == 1'b1)

begin

block_write_req <= 1'b0;

state <= S_WRITE_END;

end

else

block_write_req <= 1'b1;

end

S_CMD17:

begin

if(cmd_req_ack & ~cmd_req_error)

begin

state <= S_READ;

cmd_req <= 1'b0;

end

else

begin

cmd_req <= 1'b1;

cmd_data_len <= 16'd0;

cmd_r1 <= 8'h00;

cmd <= {8'd17,sec_addr,8'hff};

end

end

S_READ:

begin

if(block_read_req_ack)

begin

state <= S_READ_END;

block_read_req <= 1'b0;

end

else

begin

block_read_req <= 1'b1;

end

end

S_WRITE_END:

begin

state <= S_WAIT_READ_WRITE;

end

S_READ_END:

begin

state <= S_WAIT_READ_WRITE;

end

default:

state <= S_IDLE;

endcase

end

endmodule

  5.4 spi_master

  这一模块用来完成SPI一个字节的读写。

  spi master 状态机设计, 主要完成一个字节 spi 数据的读写,由于是全双工的,写一个字节的

  同时也读一个字节。 首先空闲状态“IDLE”接收到写请求后进入“DCLK_IDLE”状态,这个状态为

  spi 时钟沿变化保持一定的时间,用来控制 spi 时钟的周期,然后进入 spi 时钟沿的变化状态,一

  个字节上升沿和下降沿一共 16 个数据沿。 在最后一个数据沿进入“LAST_HALF_CYCLE”状态,为

  让最后一个沿也保持一定的时间,再进入应答状态,完成一次写请求。spi_master 模块中模拟了一个 spi 时钟,在状态机进入到‘DCLK_EDGE’时进行翻转。状态机图示如下:

  代码如下:

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//==========================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//--------------------------------------------------------------------------

//  2017/6/19     meisq         1.0         Original

//*************************************************************************/

module spi_master

(

input                       sys_clk,

input                       rst,

output                      nCS,       //chip select (SPI mode)

output                      DCLK,      //spi clock

output                      MOSI,      //spi data output

input                       MISO,      //spi input

input                       CPOL,

input                       CPHA,

input                       nCS_ctrl,

input[15:0]                 clk_div,

input                       wr_req,

output                      wr_ack,

input[7:0]                  data_in,

output[7:0]                 data_out

);

localparam                   IDLE            = 0;

localparam                   DCLK_EDGE       = 1;

localparam                   DCLK_IDLE       = 2;

localparam                   ACK             = 3;

localparam                   LAST_HALF_CYCLE = 4;

localparam                   ACK_WAIT        = 5;

reg                          DCLK_reg;

reg[7:0]                     MOSI_shift;

reg[7:0]                     MISO_shift;

reg[2:0]                     state;

reg[2:0]                     next_state;

reg [15:0]                   clk_cnt;

reg[4:0]                     clk_edge_cnt;

assign MOSI = MOSI_shift[7];

assign DCLK = DCLK_reg;

assign data_out = MISO_shift;

assign wr_ack = (state == ACK);

assign nCS = nCS_ctrl;

always@(posedge sys_clk or posedge rst)

begin

if(rst)

state <= IDLE;

else

state <= next_state;

end

always@(*)

begin

case(state)

IDLE:

if(wr_req == 1'b1)

next_state <= DCLK_IDLE;

else

next_state <= IDLE;

DCLK_IDLE:

//half a SPI clock cycle produces a clock edge

if(clk_cnt == clk_div)

next_state <= DCLK_EDGE;

else

next_state <= DCLK_IDLE;

DCLK_EDGE:

//a SPI byte with a total of 16 clock edges

if(clk_edge_cnt == 5'd15)

next_state <= LAST_HALF_CYCLE;

else

next_state <= DCLK_IDLE;

//this is the last data edge

LAST_HALF_CYCLE:

if(clk_cnt == clk_div)

next_state <= ACK;

else

next_state <= LAST_HALF_CYCLE;

//send one byte complete

ACK:

next_state <= ACK_WAIT;

//wait for one clock cycle, to ensure that the cancel request signal

ACK_WAIT:

next_state <= IDLE;

default:

next_state <= IDLE;

endcase

end

always@(posedge sys_clk or posedge rst)

begin

if(rst)

DCLK_reg <= 1'b0;

else if(state == IDLE)

DCLK_reg <= CPOL;

else if(state == DCLK_EDGE)

DCLK_reg <= ~DCLK_reg;//SPI clock edge

end

//SPI clock wait counter

always@(posedge sys_clk or posedge rst)

begin

if(rst)

clk_cnt <= 16'd0;

else if(state == DCLK_IDLE || state == LAST_HALF_CYCLE)

clk_cnt <= clk_cnt + 16'd1;

else

clk_cnt <= 16'd0;

end

//SPI clock edge counter

always@(posedge sys_clk or posedge rst)

begin

if(rst)

clk_edge_cnt <= 5'd0;

else if(state == DCLK_EDGE)

clk_edge_cnt <= clk_edge_cnt + 5'd1;

else if(state == IDLE)

clk_edge_cnt <= 5'd0;

end

//SPI data output

always@(posedge sys_clk or posedge rst)

begin

if(rst)

MOSI_shift <= 8'd0;

else if(state == IDLE && wr_req)

MOSI_shift <= data_in;

else if(state == DCLK_EDGE)

if(CPHA == 1'b0 && clk_edge_cnt[0] == 1'b1)

MOSI_shift <= {MOSI_shift[6:0],MOSI_shift[7]};

else if(CPHA == 1'b1 && (clk_edge_cnt != 5'd0 && clk_edge_cnt[0] == 1'b0))

MOSI_shift <= {MOSI_shift[6:0],MOSI_shift[7]};

end

//SPI data input

always@(posedge sys_clk or posedge rst)

begin

if(rst)

MISO_shift <= 8'd0;

else if(state == IDLE && wr_req)

MISO_shift <= 8'h00;

else if(state == DCLK_EDGE)

if(CPHA == 1'b0 && clk_edge_cnt[0] == 1'b0)

MISO_shift <= {MISO_shift[6:0],MISO};

else if(CPHA == 1'b1 && (clk_edge_cnt[0] == 1'b1))

MISO_shift <= {MISO_shift[6:0],MISO};

end

endmodule

  5.5 其余代码

  5.5.1 sd_card_test

//

//                                                                              //

//                                                                              //

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//================================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//--------------------------------------------------------------------------------

//  2017/6/19     meisq         1.0         Original

//*******************************************************************************/

module sd_card_test(

input            clk,

input            rst_n,

input            key1,

output           SD_nCS,

output           SD_DCLK,

output           SD_MOSI,

input            SD_MISO,

output [5:0]     seg_sel,

output [7:0]     seg_data

);

parameter S_IDLE         = 0;

parameter S_READ         = 1;

parameter S_WRITE        = 2;

parameter S_END          = 3;

reg[3:0] state;

wire             sd_init_done;

reg              sd_sec_read;

wire[31:0]       sd_sec_read_addr;

wire[7:0]        sd_sec_read_data;

wire             sd_sec_read_data_valid;

wire             sd_sec_read_end;

reg              sd_sec_write;

wire[31:0]       sd_sec_write_addr;

reg [7:0]        sd_sec_write_data;

wire             sd_sec_write_data_req;

wire             sd_sec_write_end;

reg[9:0]         wr_cnt;

reg[9:0]         rd_cnt;

wire             button_negedge;

reg[7:0]         read_data;

ax_debounce ax_debounce_m0

(

.clk             (clk),

.rst             (~rst_n),

.button_in       (key1),

.button_posedge  (),

.button_negedge  (button_negedge),

.button_out      ()

);

wire[6:0] seg_data_0;

seg_decoder seg_decoder_m0(

    .bin_data  (read_data[3:0]),

    .seg_data  (seg_data_0)

);

wire[6:0] seg_data_1;

seg_decoder seg_decoder_m1(

    .bin_data  (read_data[7:4]),

    .seg_data  (seg_data_1)

);

seg_scan seg_scan_m0(

    .clk        (clk),

    .rst_n      (rst_n),

    .seg_sel    (seg_sel),

    .seg_data   (seg_data),

    .seg_data_0 ({1'b1,7'b1111_111}),

    .seg_data_1 ({1'b1,7'b1111_111}),

    .seg_data_2 ({1'b1,7'b1111_111}),

    .seg_data_3 ({1'b1,7'b1111_111}),

    .seg_data_4 ({1'b1,seg_data_1}),

    .seg_data_5 ({sd_init_done,seg_data_0})

);

always@(posedge clk or negedge rst_n)

begin

if(rst_n == 1'b0)

wr_cnt <= 10'd0;

else if(state == S_WRITE)

begin

if(sd_sec_write_data_req == 1'b1)

wr_cnt <= wr_cnt + 10'd1;

end

else

wr_cnt <= 10'd0;

end

always@(posedge clk or negedge rst_n)

begin

if(rst_n == 1'b0)

rd_cnt <= 10'd0;

else if(state == S_READ)

begin

if(sd_sec_read_data_valid == 1'b1)

rd_cnt <= rd_cnt + 10'd1;

end

else

rd_cnt <= 10'd0;

end

always@(posedge clk or negedge rst_n)

begin

if(rst_n == 1'b0)

read_data <= 8'd0;

else if(state == S_READ)

begin

if(sd_sec_read_data_valid == 1'b1 && rd_cnt == 10'd0)

read_data <= sd_sec_read_data;

end

else if(state == S_END && button_negedge == 1'b1)

read_data <= read_data + 8'd1;

end

always@(posedge clk or negedge rst_n)

begin

if(rst_n == 1'b0)

sd_sec_write_data <= 8'd0;

else if(sd_sec_write_data_req)

sd_sec_write_data <= read_data + wr_cnt[7:0];

end

always@(posedge clk or negedge rst_n)

begin

if(rst_n == 1'b0)

begin

state <= S_IDLE;

sd_sec_read <= 1'b0;

sd_sec_write <= 1'b0;

end

else if(sd_init_done == 1'b0)

begin

state <= S_IDLE;

end

else

case(state)

S_IDLE:

begin

state <= S_READ;

end

S_WRITE:

begin

if(sd_sec_write_end == 1'b1)

begin

sd_sec_write <= 1'b0;

state <= S_READ;

end

else

sd_sec_write <= 1'b1;

end

S_READ:

begin

if(sd_sec_read_end == 1'b1)

begin

state <= S_END;

sd_sec_read <= 1'b0;

end

else

begin

sd_sec_read <= 1'b1;

end                 

end         

S_END:

begin

if(button_negedge == 1'b1)

state <= S_WRITE;

end

default:

state <= S_IDLE;

endcase

end

sd_card_top  sd_card_top_m0(

.clk                       (clk                    ),

.rst                       (~rst_n                 ),

.SD_nCS                    (SD_nCS                 ),

.SD_DCLK                   (SD_DCLK                ),

.SD_MOSI                   (SD_MOSI                ),

.SD_MISO                   (SD_MISO                ),

.sd_init_done              (sd_init_done           ),

.sd_sec_read               (sd_sec_read            ),

.sd_sec_read_addr          (sd_sec_read_addr       ),

.sd_sec_read_data          (sd_sec_read_data       ),

.sd_sec_read_data_valid    (sd_sec_read_data_valid ),

.sd_sec_read_end           (sd_sec_read_end        ),

.sd_sec_write              (sd_sec_write           ),

.sd_sec_write_addr         (sd_sec_write_addr      ),

.sd_sec_write_data         (sd_sec_write_data      ),

.sd_sec_write_data_req     (sd_sec_write_data_req  ),

.sd_sec_write_end          (sd_sec_write_end       )

);

endmodule 

  5.5.2 ax_debounce

//

//                                                                              //

//                                                                              //

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//================================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//--------------------------------------------------------------------------------

//  2017/5/3     meisq          1.0         Original

//*******************************************************************************/

`timescale 1 ns / 100 ps

module  ax_debounce 

(

    input       clk, 

    input       rst, 

    input       button_in,

    output reg  button_posedge,

    output reg  button_negedge,

    output reg  button_out

);

 ---------------- internal constants --------------

parameter N = 32 ;           // debounce timer bitwidth

parameter FREQ = 50;         //model clock :Mhz

parameter MAX_TIME = 20;     //ms

localparam TIMER_MAX_VAL =   MAX_TIME * 1000 * FREQ;

---------------- internal variables ---------------

reg  [N-1 : 0]  q_reg;      // timing regs

reg  [N-1 : 0]  q_next;

reg DFF1, DFF2;             // input flip-flops

wire q_add;                 // control flags

wire q_reset;

reg button_out_d0;

 ------------------------------------------------------

contenious assignment for counter control

assign q_reset = (DFF1  ^ DFF2);          // xor input flip flops to look for level chage to reset counter

assign q_add = ~(q_reg == TIMER_MAX_VAL); // add to counter when q_reg msb is equal to 0

    

 combo counter to manage q_next 

always @ ( q_reset, q_add, q_reg)

begin

    case( {q_reset , q_add})

        2'b00 :

                q_next <= q_reg;

        2'b01 :

                q_next <= q_reg + 1;

        default :

                q_next <= { N {1'b0} };

    endcase     

end

 Flip flop inputs and q_reg update

always @ ( posedge clk or posedge rst)

begin

    if(rst == 1'b1)

    begin

        DFF1 <= 1'b0;

        DFF2 <= 1'b0;

        q_reg <= { N {1'b0} };

    end

    else

    begin

        DFF1 <= button_in;

        DFF2 <= DFF1;

        q_reg <= q_next;

    end

end

 counter control

always @ ( posedge clk or posedge rst)

begin

if(rst == 1'b1)

button_out <= 1'b1;

    else if(q_reg == TIMER_MAX_VAL)

        button_out <= DFF2;

    else

        button_out <= button_out;

end

always @ ( posedge clk or posedge rst)

begin

if(rst == 1'b1)

begin

button_out_d0 <= 1'b1;

button_posedge <= 1'b0;

button_negedge <= 1'b0;

end

else

begin

button_out_d0 <= button_out;

button_posedge <= ~button_out_d0 & button_out;

button_negedge <= button_out_d0 & ~button_out;

end

end

endmodule

  5.5.3 seg_decoder

//

//                                                                              //

//                                                                              //

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//==========================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//--------------------------------------------------------------------------

//  2017/6/19     meisq         1.0         Original

//*************************************************************************/

module seg_decoder

(

input[3:0]      bin_data,     // bin data input

output reg[6:0] seg_data      // seven segments LED output

);

always@(*)

begin

case(bin_data)

4'd0:seg_data <= 7'b100_0000;

4'd1:seg_data <= 7'b111_1001;

4'd2:seg_data <= 7'b010_0100;

4'd3:seg_data <= 7'b011_0000;

4'd4:seg_data <= 7'b001_1001;

4'd5:seg_data <= 7'b001_0010;

4'd6:seg_data <= 7'b000_0010;

4'd7:seg_data <= 7'b111_1000;

4'd8:seg_data <= 7'b000_0000;

4'd9:seg_data <= 7'b001_0000;

4'ha:seg_data <= 7'b000_1000;

4'hb:seg_data <= 7'b000_0011;

4'hc:seg_data <= 7'b100_0110;

4'hd:seg_data <= 7'b010_0001;

4'he:seg_data <= 7'b000_0110;

4'hf:seg_data <= 7'b000_1110;

default:seg_data <= 7'b111_1111;

endcase

end

endmodule

  5.5.4 seg_scan

//  Author: meisq                                                               //

//          msq@qq.com                                                          //

//          ALINX(shanghai) Technology Co.,Ltd                                  //

//          heijin                                                              //

//     WEB: http://www.alinx.cn/                                                //

//     BBS: http://www.heijin.org/                                              //

//                                                                              //

//

//                                                                              //

// Copyright (c) 2017,ALINX(shanghai) Technology Co.,Ltd                        //

//                    All rights reserved                                       //

//                                                                              //

// This source file may be used and distributed without restriction provided    //

// that this copyright statement is not removed from the file and that any      //

// derivative work contains the original copyright notice and the associated    //

// disclaimer.                                                                  //

//                                                                              //

//

//==========================================================================

//  Revision History:

//  Date          By            Revision    Change Description

//--------------------------------------------------------------------------

//  2017/6/19     meisq         1.0         Original

//*************************************************************************/

module seg_scan(

input           clk,

input           rst_n,

output reg[5:0] seg_sel,      //digital led chip select

output reg[7:0] seg_data,     //eight segment digital tube output,MSB is the decimal point

input[7:0]      seg_data_0,

input[7:0]      seg_data_1,

input[7:0]      seg_data_2,

input[7:0]      seg_data_3,

input[7:0]      seg_data_4,

input[7:0]      seg_data_5

);

parameter SCAN_FREQ = 200;     //scan frequency

parameter CLK_FREQ = 50000000; //clock frequency

parameter SCAN_COUNT = CLK_FREQ /(SCAN_FREQ * 6) - 1;

reg[31:0] scan_timer;  //scan time counter

reg[3:0] scan_sel;     //Scan select counter

always@(posedge clk or negedge rst_n)

begin

if(rst_n == 1'b0)

begin

scan_timer <= 32'd0;

scan_sel <= 4'd0;

end

else if(scan_timer >= SCAN_COUNT)

begin

scan_timer <= 32'd0;

if(scan_sel == 4'd5)

scan_sel <= 4'd0;

else

scan_sel <= scan_sel + 4'd1;

end

else

begin

scan_timer <= scan_timer + 32'd1;

end

end

always@(posedge clk or negedge rst_n)

begin

if(rst_n == 1'b0)

begin

seg_sel <= 6'b111111;

seg_data <= 8'hff;

end

else

begin

case(scan_sel)

//first digital led

4'd0:

begin

seg_sel <= 6'b11_1110;

seg_data <= seg_data_0;

end

//second digital led

4'd1:

begin

seg_sel <= 6'b11_1101;

seg_data <= seg_data_1;

end

//...

4'd2:

begin

seg_sel <= 6'b11_1011;

seg_data <= seg_data_2;

end

4'd3:

begin

seg_sel <= 6'b11_0111;

seg_data <= seg_data_3;

end

4'd4:

begin

seg_sel <= 6'b10_1111;

seg_data <= seg_data_4;

end

4'd5:

begin

seg_sel <= 6'b01_1111;

seg_data <= seg_data_5;

end

default:

begin

seg_sel <= 6'b11_1111;

seg_data <= 8'hff;

end

endcase

end

end

endmodule

  6 实验结果

  下载实验程序后,可以看到数码管显示一个数字,这个数字是存储在 sd 卡中第一扇区的第一个数据,数据是随机的,这个时候按键 KEY1 按下,数字加一,并写入了 sd 卡,再次下载程序,可以看到直接显示更新后的数据。

相关文章:

Verilog:【8】基于FPGA实现SD NAND FLASH的SPI协议读写

Verilog&#xff1a;【8】基于FPGA实现SD NAND FLASH的SPI协议读写 在此介绍的是使用FPGA实现SD NAND FLASH的读写操作&#xff0c;以雷龙发展提供的CS创世SD NAND FLASH样品为例&#xff0c;分别讲解电路连接、读写时序与仿真和实验结果。 目录 1 FLASH背景介绍 2 样品申请 3…...

目标检测-RT-DETR

RT-DETR (Real-Time Detection Transformer) 是一种结合了 Transformer 和实时目标检测的创新模型架构。它旨在解决现有目标检测模型在速度和精度之间的权衡问题&#xff0c;通过引入高效的 Transformer 模块和优化的检测头&#xff0c;提升了模型的实时性和准确性。RT-DETR 可…...

业务资源管理模式语言09

示例&#xff1a; 图13 表示了QuoteTheMaintenance 模式的一个实例&#xff0c;在汽车修理店系统中&#xff0c;其中“Vehicle”扮演“Resource”&#xff0c;“Repair Quotation”扮演“Maintenance Quotation”&#xff0c;“Repair shop branch”扮演“Source-party”&…...

Spring Boot + Vue 多级目录的构建详解

1. 背景介绍 1.1 为何选择 Spring Boot Vue&#xff1f; 在现代 Web 开发中&#xff0c;前后端分离已成为一种标准实践。Spring Boot 提供了强大的后端开发能力&#xff0c;尤其在构建企业级应用时&#xff0c;其轻量级、高效性和丰富的生态系统让开发者如虎添翼。而 Vue.js…...

Android的Launch

看了一下资料&#xff0c;其实差别并不像一般的bootloader之类那么大。基本上还是和普通的APK程序差不多&#xff0c;基本上是AMS启动的第一个带界面的程序&#xff0c;这个界面也是常规的开发模式。可以设置各种view&#xff0c;可以设置背景。 然后在这个程序中&#xff0c;…...

Deep Ocr

1.圈出内容,文本那里要有内容.然后你保存,并导出数据集. 2.找出deep_ocr_recognition_training_workflow.hdev 文件.修改“DatasetFilename : Test.hdict” 310行 write_deep_ocr (DeepOcrHandle, BestModelDeepOCRFilename) 3.推理test.hdev 但发现很慢&#xff0c;没有mlp…...

图片验证码

导入依赖 <dependencies><dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.8.25</version></dependency> </dependencies> 代码 Service public class ValidateCodeServi…...

004: VTK读入数据---vtkImageData详细说明

VTK医学图像处理---vtkImageData类 目录 VTK医学图像处理---vtkImageData类 简介&#xff1a; 1 Mricro软件的安装和使用 (1) Mricro安装 (2) Mricro转换DICOM为裸数据 2 从硬盘读取数据到vtkImageData 3 vtkImageData转RGB或RGBA格式 4 练习 总结 简介&#xff1a;…...

分割千万级,将大文件分割为小件 csv

依赖 <dependency><groupId>commons-io</groupId><artifactId>commons-io</artifactId><version>2.9.0</version></dependency> package com.topnet.controller;import com.topnet.utils.R; import lombok.extern.slf4j.Slf4…...

SQL COUNT() 函数深入解析

SQL COUNT() 函数深入解析 SQL&#xff08;Structured Query Language&#xff09;是一种用于管理关系数据库管理系统&#xff08;RDBMS&#xff09;的标准编程语言。在SQL中&#xff0c;COUNT() 函数是一个常用的聚合函数&#xff0c;用于计算数据表中的行数或特定列的值数量…...

vue3和vue2的双向绑定原理

Vue 的双向绑定是其核心特性之一&#xff0c;允许数据和视图之间保持同步。在 Vue 2 和 Vue 3 中&#xff0c;双向绑定的实现原理有所不同。以下是两者的原理对比&#xff1a; Vue 2 的双向绑定原理 在 Vue 2 中&#xff0c;双向绑定是通过以下机制实现的&#xff1a; 响应式…...

[C++]刷题

作者主页&#xff1a; 作者主页 本篇博客专栏&#xff1a;C 创作时间 &#xff1a;2024年6月20日 最后&#xff1a; 十分感谢你可以耐着性子把它读完和我可以坚持写到这里&#xff0c;送几句话&#xff0c;对你&#xff0c;也对我&#xff1a; 1.一个冷知识&#xff1a; …...

职称评审中,论文发表要求?

无论是医生、教师或其他等职业&#xff0c;职称评审无疑是一个非常重要的环节。而职称评审中的论文发表则是评定我们专业能力的重要一环&#xff0c;可如何才能让自己辛苦撰写的的论文被发表&#xff0c;达到论文发表都有哪些要求呢&#xff1f; 一、选题要新颖 编辑和审稿人…...

连续信号的matlab表示

复习信号与系统以及matlab 在matlab中连续信号使用较小的采样间隔来表四 1.单位阶跃信号 阶跃信号:一个理想的单位阶跃信号在时间 t 0 之前值为0&#xff0c;在 t 0 及之后值突然变为常数 A&#xff08;通常取 A 1&#xff09; %matlab表示连续信号,是让信号的采样间隔很小…...

centos7.9搭建mysql5.6主从

mysql5.6 搭建数据库配置主从 搭建数据库 官网下载软件包后上传 基于centos7.9搭建mysql5.6.42 [rootmysql02 ~]# ls anaconda-ks.cfg init.sh MySQL-5.6.42-1.el7.x86_64.rpm-bundle.tar解压 tar -xf MySQL-5.6.42-1.el7.x86_64.rpm-bundle.tar -C /opt/[rootmysql02 ~]…...

C#通过ACE OLEDB驱动程序访问 Access和 Excel

ACE 代表 Access Connectivity Engine。它是 Microsoft 提供的一组组件&#xff0c;用于访问和操作 Microsoft Access 数据库以及其他类似的文件格式&#xff0c;如 Excel 工作簿。ACE 主要包括以下几部分&#xff1a; ACE OLEDB 驱动程序&#xff1a;用于通过 OLE DB 提供程序…...

智能新纪元:GPT-Next引领的AI革命及其跨领域应用

GPT-Next&#xff1a;性能的百倍提升 在当今这个科技日新月异的时代&#xff0c;人工智能&#xff08;AI&#xff09;无疑是最具活力和变革性的领域之一。最近&#xff0c;OpenAI在KDDI峰会上宣布了一项激动人心的消息&#xff1a;他们即将推出名为“GPT-Next”的新一代语言模…...

Nexus配置npm私服

1&#xff0c;配置npm-hub 2&#xff0c;配置proxy-npm 3&#xff0c;配置group-npm 4&#xff0c;配置local-npm 5&#xff0c;配置淘宝...

《OpenCV计算机视觉》—— 图像轮廓检测与绘制

文章目录 一、轮廓的检测二、轮廓的绘制图像轮廓检测与绘制的代码实现 三、轮廓的近似 一、轮廓的检测 轮廓检测是指在包含目标和背景的数字图像中&#xff0c;忽略背景和目标内部的纹理以及噪声干扰的影响&#xff0c;采用一定的技术和方法来实现目标轮廓提取的过程注意:做轮…...

Spark-Yarn模式如何配置历史服务器

在Spark程序结束之后我们也想看到运行过程怎么办&#xff1f; Yarn模式下&#xff0c;通过以下步骤配置历史服务器即可: mv spark-defaults.conf.template spark-defaults.conf修改spark-default.conf 文件&#xff0c;配置日志存储路径 spark.eventLog.enabled true spark.…...

Maven的安装

一、安装 压缩包解压完的目录如下所示&#xff08;此处为绿色免安装版&#xff09;&#xff1a; &#xff08;其余三个文件是针对Maven版本&#xff0c;第三方软件等简要介绍&#xff09; 二、环境变量 前提&#xff1a; jdk最低版本为JAVA7&#xff08;即jdk17&#xff09…...

iOS——APP启动流程

APP启动 APP启动主要分为两个阶段&#xff1a;pre-main和main之后&#xff0c;而APP的启动优化也主要是在这两个阶段进行的。 main之后的优化&#xff1a;1. 减少不必要的任务&#xff0c;2.必要的任务延迟执行&#xff0c;例如放在控制器界面等等。 APP启动的大致过程&#…...

LLM模型:代码讲解Transformer运行原理

视频讲解、获取源码&#xff1a;LLM模型&#xff1a;代码讲解Transformer运行原理(1)_哔哩哔哩_bilibili 1 训练保存模型文件 2 模型推理 3 推理代码 import torch import tiktoken from wutenglan_model import WutenglanModelimport pyttsx3# 设置设备为CUDA&#xff08;如果…...

虚幻引擎VR游戏开发02 | 性能优化设置

常识&#xff1a;VR需要保持至少90 FPS的刷新率&#xff0c;以避免用户体验到延迟或晕眩感。以下是优化性能的一系列设置&#xff08;make sure the frame rate does not drop below a certain threshold&#xff09; In project setting-> &#xff08;以下十个设置都在pr…...

Web应用监控:URL事务监测指标解读

监控易是一款功能强大的IT监控软件&#xff0c;它能够实时监控各种IT资源和应用的运行状态&#xff0c;确保业务的连续性和稳定性。在Web应用监控方面&#xff0c;监控易提供了URL事务监测功能&#xff0c;通过模拟用户访问流程&#xff0c;监测整个事务的执行过程和性能表现。…...

redis之缓存淘汰策略

1.查看redis的最大占用内存 使用redis-cli命令连接redis服务端&#xff0c;输入命令&#xff1a;config get maxmemory 输出的值为0&#xff0c;0代表redis的最大占用内存等同于服务器的最大内存。 2.设置redis的最大占用内存 编辑redis的配置文件&#xff0c;并重启redis服务…...

CMake/C++:一个日志库spdlog

项目仓库 GitHub - gabime/spdlog: Fast C logging library.Fast C logging library. Contribute to gabime/spdlog development by creating an account on GitHub.https://github.com/gabime/spdlog 知乎参考贴 https://zhuanlan.zhihu.com/p/674073158 先将仓库clone一下 然…...

rig——管理不同R语言版本的工具

在Python中&#xff0c;我可以用Conda去管理多个版本的Python&#xff0c;包括一些Python模块&#xff0c;因此想在R语言中也找一个类似的工具。 之前在Mac上&#xff0c;有一个名为 Rswitch 的R语言版本管理工具&#xff0c;可以管理不同版本的R以及相应的R包。 现在想在Win…...

Java内存模型详解

1. 引言 在Java中&#xff0c;内存模型是非常重要的概念&#xff0c;它涉及到线程之间如何共享数据以及保证数据的一致性。了解Java内存模型对于开发高质量的多线程程序是至关重要的。 本篇博客将详细介绍Java内存模型的概念、原则、规则以及相关的概念和术语。同时&#xff…...

空气能热泵热水器

空气能热泵热水器压缩机把低温低压气态冷媒转换成高压高温气态&#xff0c;压缩机压缩功能转化的热量为q1&#xff0c;高温高压的气态冷媒与水进行热交换&#xff0c;高压的冷媒在常温下被冷却、冷凝为液态。这过程中&#xff0c;冷媒放出热量用来加热水&#xff0c;使水升温变…...