# MICS 実験第一 J2 課題レポート

#### 学籍番号 2210342, 鈴木謙太郎

#### 2024年7月27日

## 1 問題1

### 1.1 モジュール eq の動作検証

実験資料で示されたコード 1 のようなモジュール eq を, 同じく示された eqSim を用いてシミュレーションした.このモジュール eq は, 入力 a,b に対して  $(a \wedge b) \vee (\bar{a} \wedge \bar{b})$  を出力するもので,a=b のときだけ s=1 となる.

Code 1: モジュール eq の Verilog コード

```
1 module eq (
2     s,a,b
3 );
4     input a, b;
5     output s;
6     wire na, nb, s1, s2;
7     assign na = ~a, nb = ~b;
8     assign s1 = a & b, s2 = na & nb;
9     assign s = s1 | s2;
10 endmodule
```

このモジュール eq をシミュレーションするモジュール eqSim をコード 2 のように作成した.このモジュールは,上のモジュール eq に対して予想される 4 通りの入力をすべて与えて,それぞれの出力を確認するものである.

Code 2: モジュール eqSim の Verilog コード

```
1 module eqSim; /* 一致検出回路の */
2 wire s; /* シミュレーション */
3 reg x, y;
4 eq g1(s, x, y);
5 initial
6 begin
7 $dumpfile("eq.vcd");
8 $dumpvars(0, eqSim);
```

```
9  $monitor(" %b %b %b %b", x, y, g1.s1, g1.s2,s, $stime);
10  $display(" x y s1 s2 s time");
11  x=0; y=0;
12  #50 y=1;
13  #50 x=1; y=0;
14  #50 y=1;
15  #50 $finish;
16  end
17 endmodule
```

その結果は下のようになった.

また、その波形をgtkwaveで表示した結果は図1のようになった.



図 1: eqSim の波形

このシミュレーションはすべての入力例をカバーしており,x=y のときのみ s=1 となることが確認できている.

## 1.2 モジュール eqSim の改良

モジュール eqSim をコード 3 のように改良し,eq 内部の s1, s2 を表示するようにした.

Code 3: 改良後のモジュール eqSim の Verilog コード

```
1 module eqSim; /* 一致検出回路の */
2 wire s; /* シミュレーション */
3 reg x, y;
```

```
eq g1(s, x, y);
    initial
5
      begin
       $dumpfile("eq.vcd");
       $dumpvars(0, eqSim);
       monitor(" \%b \%b \%b \%b \%b", x, y, g1.s1, g1.s2,s, $stime);
       $display(" x y s1 s2 s
                                  time");
10
      x=0; y=0;
      #50 y=1;
12
      #50 x=1; y=0;
13
       #50 y=1;
14
       #50 $finish;
15
       end
16
17 endmodule
```

このときの出力は下のようになった.

```
    x y s1 s2 s
    time

    0 0 0 1 1
    0

    0 1 0 0 0
    50

    1 0 0 0 0
    100

    1 1 1 0 1
    150
```

eqSim.v:15: \$finish called at 200 (1s)

これにより、内部的にも s1 と s2 が正しく計算され, $s=s1 \lor s2$  により正しい s が出力されていることが確認できた.

## 2 問題 2

入力信号 a,b,c,d を受け取り, $(a=b) \land (c=d)$  のとき出力信号 s を 1 に,それ以外のとき s を 0 にする モジュール double Eq を作成する.

このモジュールの簡易的な回路図は図 2 のようになっている.eq と表記した部分は 1 節で扱ったモジュール eq を用いる.



図 2: モジュール doubleEq の回路図

これをもとに、コード 4 のようにモジュール double Eq を作成した.

Code 4: モジュール doubleEq の Verilog コード

```
1 module doubleEq(
2     s,a,b,c,d
3 );
4     input a, b, c, d;
5     output s;
6     wire w1, w2;
7     eq m1(w1, a, b);
8     eq m2(w2, c, d);
9     assign s = w1 & w2;
10 endmodule
```

## 3 問題3

2節で作成したモジュール doubleEq をシミュレーションするモジュール doubleEqSim をコード 5 のように作成した. 今回は入力ケースがたかだか 16 通りほどだったので, 想定されるすべての入力についてテストを行った.

Code 5: モジュール doubleEqSim の Verilog コード

```
1 module doubleEqSim; /* 一致検出回路の */
    wire s; /* シミュレーション */
    reg a, b, c, d;
3
    doubleEq g1(s, a, b, c, d);
    initial
5
      begin
6
       $dumpfile("doubleEq.vcd");
       $dumpvars(0, doubleEqSim);
8
       $monitor(" %b %b %b %b %b %b %b %b", a, b, c, d, s, g1.w1, g1.w2, $stime);
9
       $display(" a b c d w1 w2 s
                                        time");
10
       /* test all case */
11
12
       a=0; b=0; c=0; d=0;
       #50 a=0; b=0; c=0; d=1;
13
       #50 a=0; b=0; c=1; d=0;
14
       #50 a=0; b=0; c=1; d=1;
15
       #50 a=0; b=1; c=0; d=0;
16
       #50 a=0; b=1; c=0; d=1;
17
       #50 a=0; b=1; c=1; d=0;
18
       #50 a=0; b=1; c=1; d=1;
19
       #50 a=1; b=0; c=0; d=0;
20
       #50 a=1; b=0; c=0; d=1;
21
       #50 a=1; b=0; c=1; d=0;
22
       #50 a=1; b=0; c=1; d=1;
23
       #50 a=1; b=1; c=0; d=0;
24
       #50 a=1; b=1; c=0; d=1;
25
       #50 a=1; b=1; c=1; d=0;
26
       #50 a=1; b=1; c=1; d=1;
27
       end
29 endmodule
```

このテストを実行し,gtkwaveで波形を表示した結果は図3のようになった.



図 3: doubleEqSim の波形

この波形から,a = b かつ c = d のときだけ s = 1 となることが確認できた.

## 4 問題 4

実験資料で示された状態遷移表 m2 に従って, 図 4 のような状態遷移図と真理値表を作成した.



図 4: 作成した状態遷移図および真理値表

この結果,b,s1',s2'の論理式は式(1)のようになった.

$$b = as_1 s_2$$

$$s1' = s_1 \bar{s_2} + \bar{a}s_1 + a\bar{s_1}s_2$$

$$s2' = \bar{a}s_2 + a\bar{s_2}$$
(1)

この論理式を下に、コード6のようなモジュール count を作成した.

1 module count (

```
2
       a,ck,b
3 );
    input a, ck;
4
    output b;
    wire na;
    wire s1, s2, t;
    wire d1, d2, d3, d4;
    wire e1, e2, e3;
    dffn f1(s1,d1,ck);
10
    dffn f2(s2,e1,ck);
11
    assign na = ~a;
    assign ns1 = ~s1;
13
    assign ns2 = ~s2;
14
15
     assign d4 = s1 & ns2, d3 = na & s1, d2 = a & ns1 & s2, d1 = d4 | d3 | d2;
16
     assign e3 = na & s2, e2 = a & ns2, e1 = e3 | e2;
     assign b = a & s1 & s2;
19 endmodule
```

また, このモジュール count をシミュレーションするモジュール countSim をコード 7 のように作成した. clk が発するクロック信号は 50 サイクル周期で切り替わる. ここで a の値を切り替える間隔を 100 サイクルにすることで, a を 1 回切り替える間に clk が 0,1 両方の場合をテストできた. これにより, 少ない行数でより多くのテストケースをカバーできた.

Code 7: モジュール countSim の Verilog コード

```
1 module countSim;
    reg a;
    wire b;
4
     clk clk1(ck);
5
6
     count dut (a,ck,b);
     initial
9
      begin
       $dumpfile("countSim.vcd");
10
11
       $dumpvars(0, countSim);
       $monitor("%b %b %b %b", a, ck, b, dut.s1, dut.s2, $stime);
12
       $display("a ck b s1 s2
                                   time");
13
14
       a = 0;
15
       #100 a = 1;
16
17
       #100 a = 0;
```

```
#100 a = 1;
18
       #100 a = 0;
19
       #100 a = 1;
20
       #100 a = 0;
21
       #100 a = 1;
^{22}
       #100 a = 0;
23
       $finish;
24
       end
25
26 endmodule
```

このテストの出力から、状態 s1 と s2 が正しく遷移していることが確認できた.また,gtkwave を用いて波形を表示すると図 5 のようになった.



図 5: countSim の波形

この波形から, 正しい状態遷移の結果 s1s2 が 11 から 00 に遷移するときに b=1 となっていることが確認できた.

## 5 問題 5

実験資料で示された 1 ビット加算器を参考にして,N ビット加算器モジュール addN をコード 8 のように作成した.

Code 8: モジュール addN の Verilog コード

```
1 module addN (
       a, b, sum, ci, cu
3);
    parameter N = 8;
4
    input [N-1:0] a, b;
6
     input ci;
7
     output [N-1:0] sum;
    output cu;
9
10
    wire [N-1:0] sum;
11
    wire [N:0] c; // carries
12
```

```
13   assign c[0] = ci;
14
15
16   assign sum = a ^ b ^ c[N-1:0];
17   assign c[N:1] = (a & b) | (b & c[N-1:0]) | (a & c[N-1:0]);
18
19
20   assign cu = c[N];
21 endmodule
```

また、モジュール addN をシミュレーションするモジュール addNSim をコード 9 のように作成した.

Code 9: モジュール addNSim の Verilog コード

```
1 module addNSim;
    reg [7:0] a, b;
    reg ci;
    wire [7:0] sum;
    wire cu;
    addN #8 g1(a, b, sum, ci, cu);
    initial begin
8
      $dumpfile("addN.vcd");
      $dumpvars(0, addNSim);
10
      $monitor(" %b %b %b %b", a, b,ci, sum, cu, $stime);
11
                   a
12
      $display("
                            b ci sum cu time");
13
      // test normal
14
      a = 8'b00000011;
15
      b = 8'b00000011;
16
      ci = 0;
17
18
      // test carry in
19
      #10;
      a = 8'b00000011;
21
      b = 8'b00000011;
22
      ci = 1;
23
24
      #10;
25
      // test overflow
26
      a = 8'b11111111;
27
      b = 8,600000001;
      ci = 0;
```

```
30
31 #10 $finish;
32 end
33 endmodule
```

このテストの出力は下のようになった.

| a        | b       | ci | sum      | cu | time |
|----------|---------|----|----------|----|------|
| 00000011 | 0000011 | 0  | 00000110 | 0  | 0    |
| 00000011 | 0000011 | 1  | 00000111 | 0  | 10   |
| 11111111 | 0000001 | 0  | 00000000 | 1  | 20   |

また,gtkwave を用いて波形を表示すると図 6 のようになった.



図 6: addNSim の波形

これらの結果から、通常の演算と桁あふれが発生する演算の両方でモジュール  $\operatorname{addN}$  が正しく動作していることが確認できた。 ビットベクトルを用いることで、 $\operatorname{1}$  ビット加算器の実装を大きく変えることなく  $\operatorname{N}$  ビット加算器を実装できることがわかった。

## 6 問題 6

実験資料で示された 1 ビットレジスタを参考にして,N ビットレジスタ regN を作成した.

#### 6.1 モジュール dffn の改良

まず, モジュール dffn を N ビットの入出力に対応させる必要があると考え, コード 10 のように改良した. なお, 入出力ビット数 N のデフォルト値を 1 にすることで既にモジュール dffn を使用している他のモジュールに影響を与えないようにした.

Code 10: 改良後のモジュール dffn の Verilog コード

```
1 module dffn (
2    Q,D,ck
3 );
4    parameter N = 1;
5
6    input [N-1:0] D;
```

```
7  input ck;
8  output [N-1:0] Q;
9  reg [N-1:0] Q;
10  initial
11     Q = {N{1'bo}};
12  always @(negedge ck)
13     Q = D;
14  endmodule
```

あわせて、モジュール dffn をシミュレーションするモジュール dffnSim もコード 11 のように改良した.

Code 11: 改良後のモジュール dffnSim の Verilog コード

```
1 module dffnSim;
     reg[1:0] i;
    wire[1:0] o;
    clk clk1(ck);
     dffn #2 dffn1(o, i, ck);
     initial
      begin
       $dumpfile("dffnSim.vcd");
       $dumpvars(0, dffnSim);
9
       $monitor(" %b %b %b", ck,i,o,$stime);
10
       $display("ck i o time");
11
       i = 2,600;
12
       #100 i = 2'b01;
       #200 i = 2'b10;
14
       #100 $finish;
15
       end
17 endmodule
```

このテストの出力は下のようになった.

```
ck i o
               time
 0 00 00
                 0
 1 00 00
                50
0 01 01
               100
1 01 01
               150
0 01 01
               200
 1 01 01
               250
0 10 10
               300
               350
 1 10 10
dffnSim.v:15: $finish called at 400 (1s)
0 10 10
               400
```

gtkwave を用いて波形を表示すると図7のようになった.



図 7: dffnSim の波形

このテストでは 2 ビットの信号の入出力をテストしている。クロックの立ち下がりに同期して 2 ビットの値を更新・保持できているので, 仕様を崩すことなくモジュール dffn を N ビット入出力に対応させることができたと判断した。

### 6.2 モジュール regN の作成

次に,N ビットレジスタのモジュール regN をコード 12 のように作成した.

Code 12: モジュール regN の Verilog コード

```
1 module regN (
2
       1, d, ck, q
3);
    parameter N = 8;
4
5
    input 1;
     input [N-1:0] d;
    input ck;
    output [N-1:0] q;
    wire [N-1:0] d1;
10
11
12
    dffn #8 f1(q,d1,ck);
     assign d1 = 1 ? d : q;
14 endmodule
```

実験資料で示された conditional assign を用いることで、少ない記述量でも load 信号の値に応じて d1 に送られる信号を制御することができた.

また, モジュール regN をシミュレーションするモジュール regNSim をコード 13 のように作成した.

Code 13: モジュール regNSim の Verilog コード

```
1  module regNSim ();
2  reg 1;
3  reg [7:0] d;
```

```
clk c1(ck);
4
    wire [7:0] q;
5
7
    regN #8 g1(1, d, ck, q);
    initial begin
9
    $dumpfile("regN.vcd");
10
    $dumpvars(0, regNSim);
    12
    $display("load
                      data ck
                                           time");
                                  q
13
14
    1 = 0;
15
    d = 8,600000000;
16
    #50;
17
18
    // pass first data
20
    1 = 1;
    d = 8'b00000011;
21
    #50;
23
    // get first data
24
    1 = 0;
25
    #50;
26
    // pass second data
28
    1 = 1;
    d = 8'b00001111;
    #50;
31
    // get second data
    1 = 0;
34
    #50;
36
    #50 $finish;
37
    end
39 endmodule
```

#### このテストの出力は下のようになった.

```
      0
      00000011
      0 00000011
      100

      1
      00001111
      1 00000011
      150

      0
      00001111
      0 00001111
      200

      0
      00001111
      1 00001111
      250

      0
      00001111
      0 00001111
      300
```

regNSim.v:39: \$finish called at 300 (1s)

gtkwave を用いて波形を表示すると図8のようになった.



図 8: regNSim の波形

クロックが立ち下がる際に load に 1 が入力されているときだけ, 出力 q が d の値に更新されることが視覚的に確認できた.

## 7 問題7

実験資料で示された N ビット加減算器の図解を参考にして,N ビット加減算器 calcN をコード 14 のように作成した.

Code 14: モジュール calcN の Verilog コード

```
1 module calcN (
    a, b, sum, k, cu
3);
    parameter N = 8;
4
    input [N-1:0] a, b;
6
    input k;
7
    output [N-1:0] sum;
    output cu;
9
10
    wire [N-1:0] B = k ? ~b : b;
11
    addN #(
12
         .N(N)
13
     ) g1(a, B, sum, k, cu);
14
15
```

また、モジュール calcN をシミュレーションするモジュール calcNSim をコード 15 のように作成した.

Code 15: モジュール calc<br/>NSim の Verilog コード

```
module calcNSim;
    reg [7:0] a, b;
    reg k;
    wire [7:0] sum;
    wire cu;
    calcN #8 g1(a, b, sum, k, cu);
7
    initial begin
8
       $dumpfile("calcN.vcd");
       $dumpvars(0, calcNSim);
10
       $monitor(" %b %b %b %b", a, b, k, sum, cu, $stime);
11
                             b k sum cu
       $display("
                   a
12
13
       // test add
14
       a = 8'b00000011;
15
      b = 8'b00000011;
16
      k = 0;
18
       // test sub
19
       #10;
       a = 8'b00000011;
21
      b = 8'b00000011;
22
      k = 1;
23
       #10;
24
25
      // test add overflow
26
       a = 8'b11111111;
27
      b = 8,600000001;
      k = 0;
29
       #10;
30
31
      // test sub overflow
32
       a = 8'b00000000;
      b = 8,600000001;
34
      k = 1;
35
      #10 $finish;
    end
37
```

このテストの出力は下のようになった.

| a          | b            | k  | sum       | Cι | 1    | time |
|------------|--------------|----|-----------|----|------|------|
| 00000011   | 00000011     | 0  | 00000110  | 0  |      | 0    |
| 00000011   | 00000011     | 1  | 0000001   | 1  |      | 10   |
| 11111111   | 0000001      | 0  | 00000000  | 1  |      | 20   |
| 00000000   | 0000001      | 1  | 11111111  | 0  |      | 30   |
| calcNSim.v | 7:41: \$fini | sh | called at | 40 | (1s) |      |

k=0 のときに加算,k=1 のときに減算を行うことが確認できた。また、加算時は cu=1 ならオーバーフロー、減算時は cu=0 ならアンダーフローとなることも確認できた。

gtkwave を用いて波形を表示すると図 9 のようになった. a と b に同じ入力をしていても,k を切り替えることで加算減算が切り替わり異なる値を正しく出力していることが確認できた.



図 9: calcNSim の波形