Verilog Circuits-Sequential Logic -More Circuits

Problem 115 :Rule90

一个具有特殊规则的一维序列

规则很简单。一维序列中元素有 1,0 两种状态,分别对应开,关状态。

在每个时钟边沿到来时刻,元素的下一个状态为元素相邻两个元素的异或。

下表更详细地给出了跳变的规则,(可以视为状态转移表),元素下一个状态可以视作输出,输入为元素本身的状态与相应两个相邻元素的当前状态。

Verilog Circuits-Sequential Logic -More Circuits

对于需要实现的电路,创建一个拥有 512 个元素的序列 (q[511:0]),每个时钟周期按照上述规则变换。load 信号有效时,序列更新 data 信号值为初始值。另外假设所有边界的值为 0 (q[-1] 以及 q[512])

module top_module(
    input clk,
    input load,
    input [511:0] data,
    output [511:0] q ); 
    always @(posedge clk)
        begin
            if(load)
                q<=data;
            else 
                begin
                    q<={1'b0,q[511:1]}^{q[510:0],1'b0};//异或左右邻居矩阵,左右邻居矩阵分别是原矩阵右移或者左移并补零得到
                end  
        end  
endmodule
  • 网友的 solution
//Problem 115 Rule90解题太麻烦了,如下写法即可:

module top_module(
input clk,
input load,
input [511:0] data,
output [511:0] q );

always @(posedge clk) begin
if (load)
q <= data;
else
q <= (q<<1)^(q>>1); //for太麻烦,参考答案也太麻烦了。
end

endmodule

Problem 116:Rule 110

依然是一维序列题

规则:一维序列中元素有 1,0 两种状态,分别对应开,关状态。

在每个时钟边沿到来时刻,元素的下一个状态取决于元素本身的状态与前后两个相邻元素的当前状态。下表详细地给出了跳变的规则

Left Center Right Center's next state
1 1 1 0
1 1 0 1
1 0 1 1
1 0 0 0
0 1 1 1
0 1 0 1
0 0 1 1
0 0 0 0

对于需要实现的电路,创建一个拥有 512 个元素的序列 (q[511:0]),每个时钟周期按照上述规则变换。load 信号有效时,序列更新 data 信号值为初始值。另外假设所有边界的值为 0 (q[-1] q[512])

  • 与上题的区别就是没有给出关系式(上题是异或),所以我们首先需要找出状态转移规则。于是把上表转换成真值表来寻找规则,输入为三个信号,left,center,right,输出信号为:next_state

  • 这题用卡诺图化简,得逻辑表达式为:\(OUT = Center\^{} Right + (Center · \~{}Left )\)

module top_module(
    input clk,
    input load,
    input [511:0] data,
    output [511:0] q
); 
    int i;
    always @(posedge clk)
        begin
            if(load)
                q<=data;
            else
                begin
                    q[0]<=q[0];
                    q[511] <= (q[511]^q[510])||(q[511]);
                    for(i = 1; i<511;i=i+1)
                        q[i] <= (q[i]^q[i-1])|(q[i]&!q[i+1]);
                end
        end  
endmodule

Problem 117:Conway's Game of life 16*16

升级版,二维序列,即二维矩阵

规则:元素的下一个状态取决于当前状态九宫格中的 8 个邻居元素中 1 的个数,当邻居有 n 个 1 时,

  • 0-1 ,元素变为 0
  • 2 ,元素保持不变
  • 3 ,元素变为 1
  • 4+ ,元素变为 0

方便做题起见,本题中的这个二维矩阵设定为 16x16,广义上可以是无限的。

为了让事情变得更加有趣,这个 16x16 矩阵的边界进行循环处理,回卷到对边,打个比方,上边界的上一行为下边界,左边界的左一列为右边界。

Verilog Circuits-Sequential Logic -More Circuits

所以对元素 (0,0) 来说,共有 8 个邻居 : (15,1), (15,0), (15,15), (0,1), (0,15), (1,1), (1,0) 以及 (1,15)。

这个 16x16 矩阵表示为 256bit 长度的向量 q,其中 q[15:0] 代表第一行,q[31:16] 代表第二行,以此类推。

load 信号有效时,更新 q 信号值为初始值 data, q 每个周期变换一次。

module top_module(
    input clk,
    input load,
    input [255:0] data,
    output [255:0] q ); 
//抄一遍代码试试
    int idx_up,idx_down,idx_right,idx_left,idx_ul,idx_ur,idx_dl,idx_dr;//
    reg [3:0]u,d,l,r;
    reg [3:0]qtemp[255:0];
    reg [255:0]qcomp;
    always @(posedge clk)//每个周期赋值
        begin
            if(load)
                q<=data;
            else 
                q<=qcomp;
        end  
    always @(*)
        begin
            for(int i=0;i<16;i++)
                for(int j=0;j<16;j++)
                    begin
                        u=i[3:0]+1;//行数
                        d=i[3:0]-1;
                        l=j[3:0]-1;
                        r=j[3:0]+1;//列数
                        idx_up=u*16+j;//第几个
                        idx_down=d*16+j;
                        idx_right=i*16+r;//当前行的某一列
                        idx_left=i*16+l;
                        idx_ul=u*16+l;
                        idx_ur=u*16+r;
                        idx_dl=d*16+l;
                        idx_dr=d*16+r;
                        qtemp[i*16+j]=q[idx_up]+q[idx_down]+q[idx_left]+q[idx_right]+q[idx_ul]+q[idx_ur]+q[idx_dl]+q[idx_dr];
                        qcomp[i*16+j] = qtemp[i*16+j]==2? q[i*16+j]:( qtemp[i*16+j]==3?1:0);
                    end 
        end  
endmodule
上一篇:TOJ 3488 Game Dice


下一篇:http反向代理之haproxy详解