当前位置:   article > 正文

fpga时序逻辑(三段式状态机模板、rom实现、边沿检测)

状态机模板

目录

VL21 根据状态转移表实现时序电路

VL22 根据状态转移图实现时序电路

VL23 ROM的简单实现

VL24 边沿检测


VL21 根据状态转移表实现时序电路

题目分析:

1、使用三段式状态机,实现更为方便和简洁。

2、三段式和(一段式、二段式)对比:

        优点:
                (1)时序逻辑和组合逻辑分开,便于分析。
                (2)利于综合软件的分析和优化。
                (3)代码简介明了,便于维护。

        缺点:
                (1)代码结构相较两段式复杂。
                (2)采用时序逻辑输出避免了亚稳态,但是增加了触发器的使用。

数字IC笔面基础,师傅领进门,修行靠个人——人人心中都有一个状态机(状态机简介及Verilog模板)_HFUT90S的博客-CSDN博客

三段式状态机模板:

第1段:描述状态转移(时序逻辑)

第2段:描述状态转移的条件和规律(组合逻辑)

第3段:描述状态输出(组合/时序逻辑)

 代码实现:

  1. module seq_circuit(
  2. input A ,
  3. input clk ,
  4. input rst_n,
  5. output wire Y
  6. );
  7. //三段式状态机
  8. //状态定义
  9. localparam IDLE = 2'b0; //初态
  10. localparam s0 = 2'b01;
  11. localparam s1 = 2'b10;
  12. localparam s2 = 2'b11;
  13. reg [1:0] CS; //现态
  14. reg [1:0] NS; //次态
  15. //描述状态转移(时序逻辑)
  16. always @(posedge clk or negedge rst_n)begin
  17. if(!rst_n)
  18. CS <= IDLE;
  19. else
  20. CS <= NS;
  21. end
  22. //描述状态转移的条件和规律(组合逻辑)
  23. always @(*)begin
  24. case(CS)
  25. IDLE: if(!A)
  26. NS = s0;
  27. else
  28. NS = s2;
  29. s0 : if(!A)
  30. NS = s1;
  31. else
  32. NS = IDLE;
  33. s1 : if(!A)
  34. NS = s2;
  35. else
  36. NS = s0;
  37. s2 : if(!A)
  38. NS = IDLE;
  39. else
  40. NS = s1;
  41. default: NS = IDLE;
  42. endcase
  43. end
  44. //描述状态输出(组合/时序逻辑)
  45. assign Y = ((CS == IDLE)|(CS == s0)|(CS == s1))?0:1;
  46. endmodule

VL22 根据状态转移图实现时序电路

题目分析:

VL22和VL21同理,只不过从状态转移表换成了状态转移图,细心一点就没问题,用的还是三段式状态机。

说明:→上表示“C/Y”,圆圈内为现态,→指向次态。

代码实现:

  1. `timescale 1ns/1ns
  2. module seq_circuit(
  3. input C ,
  4. input clk ,
  5. input rst_n,
  6. output wire Y
  7. );
  8. //三段式状态机
  9. //状态定义
  10. localparam IDLE = 2'b0;
  11. localparam s0 = 2'b01;
  12. localparam s1 = 2'b10;
  13. localparam s2 = 2'b11;
  14. reg [1:0] CS; //现态
  15. reg [1:0] NS; //次态
  16. //描述状态转移(时序逻辑)
  17. always @(posedge clk or negedge rst_n)begin
  18. if(!rst_n)
  19. CS <= IDLE;
  20. else
  21. CS <= NS;
  22. end
  23. //描述状态转移的条件和规律(组合逻辑)
  24. always @(*)begin
  25. case(CS)
  26. IDLE: if(!C)
  27. NS = IDLE; //0
  28. else
  29. NS = s0; //0
  30. s0 : if(!C)
  31. NS = s2; //0
  32. else
  33. NS = s0; //0
  34. s1 : if(!C)
  35. NS = IDLE; //0
  36. else
  37. NS = s1; //1
  38. s2 : if(!C)
  39. NS = s2; //1
  40. else
  41. NS = s1; //1
  42. default: NS = IDLE;
  43. endcase
  44. end
  45. //描述状态输出(组合/时序逻辑)
  46. assign Y = ((CS == s2) | ((CS == s1)& (C == 1)))?1:0;
  47. endmodule

VL23 ROM的简单实现

题目描述:

        ①实现一个深度为8,位宽为4bit的ROM,数据初始化为0,2,4,6,8,10,12,14。可以通过输入地址addr,输出相应的数据data。

        ②使用Verilog HDL实现以上功能并编写testbench验证。

代码实现:

  1. module VL23_rom(
  2. input clk,
  3. input rst_n,
  4. input [7:0]addr,
  5. output [3:0]data
  6. );
  7. //定义一个深度为8 宽度为4的数组
  8. reg [3:0] memory [7:0];
  9. always @(posedge clk or negedge rst_n)begin
  10. if(!rst_n)begin
  11. memory[0] <= 4'b0;
  12. memory[1] <= 4'b0;
  13. memory[2] <= 4'b0;
  14. memory[3] <= 4'b0;
  15. memory[4] <= 4'b0;
  16. memory[5] <= 4'b0;
  17. memory[6] <= 4'b0;
  18. memory[7] <= 4'b0;
  19. end
  20. else begin
  21. memory[0] <= 4'd0;
  22. memory[1] <= 4'd2;
  23. memory[2] <= 4'd4;
  24. memory[3] <= 4'd6;
  25. memory[4] <= 4'd8;
  26. memory[5] <= 4'd10;
  27. memory[6] <= 4'd12;
  28. memory[7] <= 4'd14;
  29. end
  30. end
  31. assign data = (addr == 8'd0)?memory[0]:
  32. (addr == 8'd1)?memory[1]:
  33. (addr == 8'd2)?memory[2]:
  34. (addr == 8'd3)?memory[3]:
  35. (addr == 8'd4)?memory[4]:
  36. (addr == 8'd5)?memory[5]:
  37. (addr == 8'd6)?memory[6]:
  38. (addr == 8'd7)?memory[7]:0;
  39. endmodule

仿真文件:

  1. module tb_VL23_rom;
  2. // Inputs
  3. reg clk;
  4. reg rst_n;
  5. reg [7:0] addr ;
  6. // Outputs
  7. wire [3:0] data;
  8. // Instantiate the Unit Under Test (UUT)
  9. VL23_rom uut (
  10. .clk(clk),
  11. .rst_n(rst_n),
  12. .addr(addr),
  13. .data(data)
  14. );
  15. initial begin
  16. // Initialize Inputs
  17. clk = 0;
  18. rst_n = 0;
  19. addr = 0;
  20. // Wait 100 ns for global reset to finish
  21. #100;
  22. rst_n = 1'b1;
  23. #100;
  24. addr = 4'd6;
  25. #100; //#100 addr = 4'd5;
  26. addr = 4'd5; //代表100ns后addr由上一个状态4'd6 变为此时的状态4'd5
  27. #100;
  28. addr = 4'd4;
  29. #100;
  30. addr = 4'd3;
  31. #100;
  32. addr = 4'd2;
  33. #100;
  34. addr = 4'd1;
  35. // Add stimulus here
  36. end
  37. always #10 clk=~clk;
  38. endmodule

仿真图片:

VL24 边沿检测

题目描述:

        边沿检测:有一个缓慢变化的1bit信号a,编写一个程序检测a信号的上升沿给出指示信号rise,当a信号出现下降沿时给出指示信号down。
        注:rise,down应为单脉冲信号,在相应边沿出现时的下一个时钟为高,之后恢复到0,一直到再一次出现相应的边沿。

代码实现:

  1. module VL24_edge_detect(
  2. input clk,
  3. input rst_n,
  4. input a,
  5. output reg rise,
  6. output reg down
  7. );
  8. reg b,c;
  9. wire posedge_flag;
  10. wire negedge_flag;
  11. always @(posedge clk or negedge rst_n)begin
  12. if(!rst_n)begin
  13. b <= 1'b0;
  14. c <= 1'b0;
  15. end
  16. else begin
  17. b <= a;
  18. c <= b;
  19. end
  20. end
  21. assign posedge_flag = ~c & b;
  22. assign negedge_flag = c & ~b;
  23. always @(*)begin
  24. if(posedge_flag)
  25. rise <= 1'b1;
  26. else if(negedge_flag)
  27. down <= 1'b1;
  28. else begin
  29. rise <= 1'b0;
  30. down <= 1'b0;
  31. end
  32. end
  33. endmodule

仿真文件:

  1. module tb_VL24_edge_detect;
  2. // Inputs
  3. reg clk;
  4. reg rst_n;
  5. reg a;
  6. // Outputs
  7. wire rise;
  8. wire down;
  9. // Instantiate the Unit Under Test (UUT)
  10. VL24_edge_detect uut (
  11. .clk(clk),
  12. .rst_n(rst_n),
  13. .a(a),
  14. .rise(rise),
  15. .down(down)
  16. );
  17. initial begin
  18. // Initialize Inputs
  19. clk = 0;
  20. rst_n = 0;
  21. a = 0;
  22. // Wait 100 ns for global reset to finish
  23. #100;
  24. rst_n = 1'b1;
  25. #100;
  26. a = 1;
  27. #300;
  28. a = 0;
  29. #300;
  30. a = 1;
  31. #300;
  32. a = 4'd0;
  33. // Add stimulus here
  34. end
  35. always #10 clk=~clk;
  36. endmodule

仿真图片:


哈哈哈,结束啦!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/盐析白兔/article/detail/660038
推荐阅读
相关标签
  

闽ICP备14008679号