



2025/10/02

## 實驗四

序向邏輯練習

姓名：林承羿

學號：01257027

班級：資工 3A

E-mail : IanLin6225@gmail.com

### ※ 注意

1. 繳交時一律轉 PDF 檔
2. 繳交期限為下周上課前
3. 一人繳交一份
4. 檔名請按照作業檔名格式進行填寫，未依照格式不予批改
5. 檔名範例：學號\_姓名\_HW4

### 1、ReLU 函數

#### ● 實驗說明：

設計 1 個 FSM，實現一個 ReLU 函數。

$$ReLU(x) = \begin{cases} x & \text{if } x \geq 0 \\ 0 & \text{if } x < 0 \end{cases}$$

1. Counter\_1 從 -5、-4、…數到 0 後，接著 counter\_2 從 0、1…數到 7
2. 設計 1 個 FSM 控制 2 個 [3:0] counters，2 個 counters 相加後，存入暫存器 w，再代入 ReLU 函數(即 mux)，以暫存器 w 之最高位元做為 s0，並輸出最後結果 y。
3. 負數以二進制表示。

● 系統硬體架構方塊圖（接線圖）：



● 系統架構程式碼、測試資料程式碼與程式碼說明  
截圖請善用 **win+shift+S**

```
④ relu.sv > relu
1  module relu (
2      input logic clk,
3      input logic rst,
4      output logic [3:0] y
5  );
6      typedef enum {state_0 = 0, state_1 = 1} state_t;
7      state_t ps, ns;
8
9      logic cp1, cp2;
10     logic [3:0] counter_1, counter_2, w, sum;
11
12     // -5 ~ 0 計數器
13     always_ff @(posedge clk) begin
14         if (rst) begin
15             counter_1 <= -4'd5;
16         end
17         else if (cp1) begin
18             counter_1 <= counter_1+1;
19         end
20         else begin
21             counter_1 <= counter_1;
22         end
23     end
24
25     // 0 ~ 7 計數器
26     always_ff @(posedge clk) begin
27         if (rst) begin
28             counter_2 <= 4'd0;
29         end
30         else if (cp2) begin
31             counter_2 <= counter_2+1;
32         end
33         else begin
34             counter_2 <= counter_2;
35         end
36     end
```

```
37 // 狀態初始化，讓進入點是正確的(state_0)
38 always_ff @(posedge clk) begin
39     if (rst)
40         ps <= state_0;
41     else
42         ps <= ns;
43 end
44
45 // FSM controller: s0(狀態零)為counter_1遞減至0; s1(狀態一)為counter_2地增至7
46 always_comb begin
47     cp1 = 0; cp2 = 0; ns = ps;
48     case (ps)
49         state_0: begin
50             if (counter_1 == 4'd0) begin
51                 ns = state_1;
52             end
53             else begin
54                 cp1 = 1;
55             end
56         end
57
58         state_1: begin
59             if (counter_2 == 4'd7) begin
60                 ns = state_0;
61             end
62             else begin
63                 cp2 = 1;
64             end
65         end
66     endcase
67 end
68
```

```

69      // 加法器
70      assign sum = counter_1 + counter_2;
71
72      // W 暫存器，正緣給出總和，否則維持
73      always_ff @(posedge clk) begin
74          if (rst) begin
75              w <= 4'd0;
76          end
77          else begin
78              w <= sum;
79          end
80      end
81
82
83      // 令 s0 為 select 線，為 w 之 most-significant-bit
84      assign s0 = w[3];
85
86      // Relu(x) 多功器
87      always_comb begin
88          case (s0)
89              1'b0: y = w;
90              1'b1: y = 4'd0;
91          endcase
92      end
93  endmodule

```

從上圖可看到程式區塊、變數命名基本上依照給定方塊圖施工，其中以 FSM controller 控制兩個 counter，並在 reset 時確認 counter 初值、狀態機的進入點、暫存器 W，而這次學到的是-5 在初始化時要給-4sd' 5(負號寫前面，單位為 signed decimal)。

且確認 counter\_1 + counter\_2 在 bits 為 4 的前提下不可能溢位，邊界為 -5 => 1011，7 => 0111，故不需要用 5 bits 接住總合。

## ●模擬結果與結果說明：



## 2、紅綠燈

### ● 實驗說明：

4. 用 **FSM** 實作紅綠燈
5. 第一組紅綠燈( $R[0]$ ,  $Y[0]$ ,  $G[0]$ ) 由紅燈為起點依序變換為 綠燈→黃燈..
6. 第二組紅綠燈( $R[1]$ ,  $Y[1]$ ,  $G[1]$ ) 根據第一組紅綠燈的狀態顯示 綠燈→黃燈→紅燈..
7. 以 **R** 表示紅燈、**Y** 表示黃燈、**G** 表示綠燈。
8. 1 表示燈亮，0 表示燈滅，最後輸出  $[1:0]R$ 、 $[1:0]Y$ 、 $[1:0]G$ 。

### ● 系統硬體架構方塊圖 (接線圖)：



### ● 系統架構程式碼、測試資料程式碼與程式碼說明 截圖請善用 **win+shift+S**

```
① traffic_light.sv > ...
1  module traffic_light (
2      input logic clk, rst,
3      output logic [1:0] R, Y, G
4  );
5      // 計數器，0 ~ 15 加到溢位即回到初始state
6      logic [3:0] cnt;
7      always_ff @(posedge clk) begin
8          if (rst) begin
9              cnt <= 4'd0;
10         end
11         else begin
12             cnt = cnt+1;
13         end
14     end
15
16     // 令出 RYG 紅綠燈，0=>第一組 / 1=>第二組
17     typedef enum {r = 0, y = 1, g = 2} state_t;
18     state_t ps0, ns0;
19     state_t ps1, ns1;
20
21     // 初始化狀態機進入點，0=>r / 1=>g
22     always_ff @(posedge clk) begin
23         if (rst) begin
24             ps0 <= r;
25             ps1 <= g;
26         end
27         else begin
28             ps0 <= ns0;
29             ps1 <= ns1;
30         end
31     end
```

```
32      // 第一組 FSM controller: r(8)->g(14)->y(0)
33      always_comb begin
34          R[0] = 1'd0; G[0] = 1'd0; Y[0] = 1'd0; ns0 = ps0;
35          case (ps0)
36              r: begin
37                  if (cnt == 4'd8) begin
38                      G[0] = 1'd1;
39                      ns0 = g;
40                  end
41                  else begin
42                      R[0] = 1'd1;
43                  end
44              end
45
46              g: begin
47                  if (cnt == 4'd14) begin
48                      Y[0] = 1'd1;
49                      ns0 = y;
50                  end
51                  else begin
52                      G[0] = 1'd1;
53                  end
54              end
55
56              y: begin
57                  if (cnt == 4'd0) begin
58                      R[0] = 1'd1;
59                      ns0 = r;
60                  end
61                  else begin
62                      Y[0] = 1'd1;
63                  end
64              end
65          endcase
66      end
67  
```

```

68
69 // 第二組 FSM controller: g(6)->y(8)->r(0)
70 always_comb begin
71     G[1] = 1'd0; Y[1] = 1'd0; R[1] = 1'd0; ns1 = ps1;
72     case (ps1)
73         g: begin
74             if (cnt == 4'd6) begin
75                 Y[1] = 1'd1;
76                 ns1 = y;
77             end
78             else begin
79                 G[1] = 1'd1;
80             end
81         end
82
83         y: begin
84             if (cnt == 4'd8) begin
85                 R[1] = 1'd1;
86                 ns1 = r;
87             end
88             else begin
89                 Y[1] = 1'd1;
90             end
91         end
92
93         r: begin;
94             if (cnt == 4'd0) begin
95                 G[1] = 1'd1;
96                 ns1 = g;
97             end
98             else begin
99                 R[1] = 1'd1;
100            end
101        end
102    endcase
103 end
104 endmodule

```

與第一題不同的是，我這題採用在 FSM controller 中不將個別狀態要做的事以 flag 分開，而是寫在裡面，以下說明原因。

從提供的波形圖已知 RYG 其中一者為 low，必有另一者馬上為 high，如果只在 else 寫出要 output 會慢一個 clock，意思是我要在 if 中提前一個 clock 坐下個狀態的結果以保證必有一者為 high，因此如果合在一起寫可能會多寫另外兩個 always\_comb，沒啥必要。

另外可以注意到的是計數器為 0~15，為 4bits 溢位後當歸零，即從頭開始。因此在 reset 後沒有將計數器歸零的必要。

## ●模擬結果與結果說明：



此為說明用波形圖，可以看到最底下的 `mycnt` 是一個計數器，方便用來快速確認 RYG 是否正確，本人放上兩段的波形圖以證明可以循環相同的結果。





### ● 結論與心得：

經過此次的實驗，我更熟悉如何撰寫 system verilog 中的狀態機，並更深刻的了解波形圖所呈現的結果。第一題印象深刻的是學會如何初始化負數。從第二題中我學習到如何撰寫多個狀態基於同一份 sv 檔案中。除此之外最多的是透過擴充 tb 查看自己想要的輸出，讓除錯更有邏輯與循序漸進的感覺而非一昧通靈毫無頭緒的憑猜想嘗試。