

# CHAPTER 4

---

## REGULAR SEQUENTIAL CIRCUIT

---

### 4.1 INTRODUCTION

A sequential circuit is a circuit with *memory*, which forms the *internal state* of the circuit. Unlike a combinational circuit, in which the output is a function of input only, the output of a sequential circuit is a function of the input and the internal state. The *synchronous design methodology* is the most commonly used practice in designing a sequential circuit. In this methodology, all storage elements are controlled (i.e., synchronized) by a global clock signal and the data is sampled and stored at the rising or falling edge of the clock signal. It allows designers to separate the storage components from the circuit and greatly simplifies the development process. This methodology is the most important principle in developing a large, complex digital system and is the foundation of most synthesis, verification, and testing algorithms. All of the designs in the book follow this methodology.

#### 4.1.1 D FF and register

The most basic storage component in a sequential circuit is a D-type flip-flop (D FF). The symbol and function table of a positive edge-triggered D FF are shown in Figure 4.1(a). The value of the d signal is sampled at the rising edge of the c1k signal and stored to FF. A D FF may contain an asynchronous reset signal to clear the FF to 0. Its symbol and function table are shown in Figure 4.1(b). Note that the reset operation is independent of the clock signal.

|  <table border="1" style="margin-top: 10px; border-collapse: collapse; text-align: center;"> <thead> <tr> <th>clk</th> <th><math>q^*</math></th> </tr> </thead> <tbody> <tr> <td>0</td> <td>q</td> </tr> <tr> <td>1</td> <td>q</td> </tr> <tr> <td><math>\frac{1}{2}</math></td> <td>d</td> </tr> </tbody> </table> <p>(a) D FF</p>                                                                                                                                                                                                                                                           | clk           | $q^*$ | 0     | q     | 1 | q | $\frac{1}{2}$ | d |  <table border="1" style="margin-top: 10px; border-collapse: collapse; text-align: center;"> <thead> <tr> <th>reset</th> <th>clk</th> <th><math>q^*</math></th> </tr> </thead> <tbody> <tr> <td>1</td> <td>-</td> <td>0</td> </tr> <tr> <td>0</td> <td>0</td> <td>q</td> </tr> <tr> <td>0</td> <td>1</td> <td>q</td> </tr> <tr> <td>0</td> <td><math>\frac{1}{2}</math></td> <td>d</td> </tr> </tbody> </table> <p>(b) D FF with asynchronous reset</p> | reset | clk | $q^*$ | 1 | - | 0 | 0 | 0 | q             | 0 | 1 | q | 0             | $\frac{1}{2}$ | d |  |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------|-------|-------|-------|---|---|---------------|---|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|-----|-------|---|---|---|---|---|---------------|---|---|---|---------------|---------------|---|--|
| clk                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            | $q^*$         |       |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | q             |       |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | q             |       |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| $\frac{1}{2}$                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  | d             |       |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| reset                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          | clk           | $q^*$ |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | -             | 0     |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | 0             | q     |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | 1             | q     |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | $\frac{1}{2}$ | d     |       |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
|  <table border="1" style="margin-top: 10px; border-collapse: collapse; text-align: center;"> <thead> <tr> <th>reset</th> <th>clk</th> <th>en</th> <th><math>q^*</math></th> </tr> </thead> <tbody> <tr> <td>1</td> <td>-</td> <td>-</td> <td>0</td> </tr> <tr> <td>0</td> <td>0</td> <td>-</td> <td>q</td> </tr> <tr> <td>0</td> <td>1</td> <td>-</td> <td>q</td> </tr> <tr> <td>0</td> <td><math>\frac{1}{2}</math></td> <td>0</td> <td>q</td> </tr> <tr> <td>0</td> <td><math>\frac{1}{2}</math></td> <td>1</td> <td>d</td> </tr> </tbody> </table> <p>(c) D FF with synchronous enable</p> | reset         | clk   | en    | $q^*$ | 1 | - | -             | 0 | 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        | 0     | -   | q     | 0 | 1 | - | q | 0 | $\frac{1}{2}$ | 0 | q | 0 | $\frac{1}{2}$ | 1             | d |  |
| reset                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          | clk           | en    | $q^*$ |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | -             | -     | 0     |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | 0             | -     | q     |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | 1             | -     | q     |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | $\frac{1}{2}$ | 0     | q     |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |
| 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | $\frac{1}{2}$ | 1     | d     |       |   |   |               |   |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |       |     |       |   |   |   |   |   |               |   |   |   |               |               |   |  |

**Figure 4.1** Block diagram and functional table of a D FF.**Figure 4.2** Block diagram of a synchronous system.

The three main timing parameters of a D FF are  $T_{cq}$  (clock-to-q delay),  $T_{setup}$  (setup time), and  $T_{hold}$  (hold time).  $T_{cq}$  is the time required to propagate the value of d to q at the rising edge of the clock signal. The d signal must be stable around the sampling edge to prevent the FF from entering the metastable state.  $T_{setup}$  and  $T_{hold}$  specify the time intervals before or after the sampling edge.

A D FF provides 1-bit storage. A collection of D FFs can be grouped together to store multiple bits and is known as a *register*.

### 4.1.2 Synchronous system

**Block diagram** The block diagram of a synchronous system is shown in Figure 4.2. It consists of the following parts:

- *State register*: a collection of D FFs controlled by the same clock signal

- *Next-state logic*: combinational logic that uses the external input and internal state (i.e., the output of register) to determine the new value of the register
- *Output logic*: combinational logic that generates the output signal

**Maximal operating frequency** One of the most difficult design aspects of a sequential circuit is to ensure that the system timing does not violate the setup and hold time constraints. In a synchronous system, the storage components are grouped together and treated as a single register, as shown in Figure 4.2. We need to perform timing analysis on only one memory component.

The timing of a sequential circuit is characterized by  $f_{max}$ , the maximal clock frequency, which specifies how fast the circuit can operate. The reciprocal of  $f_{max}$  specifies  $T_{clock}$ , the minimal clock period, which can be interpreted as the interval between two sampling edges of the clock. To ensure correct operation, the next value must be generated and stabilized within this interval. Assume that the maximal propagation delay of next-state logic is  $T_{comb}$ . The minimal clock period can be obtained by adding the propagation delays and setup time constraint of the closed loop in Figure 4.2:

$$T_{clock} = T_{cq} + T_{comb} + T_{setup}$$

and the maximal clock rate is the reciprocal:

$$f_{max} = \frac{1}{T_{clock}} = \frac{1}{T_{cq} + T_{comb} + T_{setup}}$$

**Timing constraint in Xilinx ISE<sup>Xilinx specific</sup>** During synthesis, Xilinx software will analyze the synthesized circuit and show  $f_{max}$  in a report. We can also specify the desired operating frequency as a synthesis constraint, and the synthesis software will try to obtain a circuit to satisfy this requirement (i.e., a circuit whose  $f_{max}$  is equal to or greater than the desired operating frequency). For example, if we use the 50-MHz (i.e., 20-ns period) oscillator on the prototyping board as the clock source,  $f_{max}$  of a sequential circuit must exceed this frequency (i.e., the period must be smaller than 20 ns). The following lines can be added to the constraint file:

```
NET "clk" TNM_NET = "clk";
TIMESPEC "TS_clk" = PERIOD "clk" 20 ns HIGH 50 %;
```

This indicates that the `clk` signal has a maximal period of 20 ns (i.e., 50 MHz) and a duty cycle of 50%.

After synthesis, we can check the relevant timing information by invoking the View Design Summary process from the ISE's Processes window. The Timing Constraints section shows whether the imposed constraints are met, and the Static Timing Report section provides more detailed timing information.

### 4.1.3 Code development

Our code development follows the basic block diagram in Figure 4.2. The key is to separate the memory component (i.e., the register) from the system. Once the register is isolated, the remaining portion is a pure combinational circuit, and the coding and analysis schemes discussed in previous chapters can be applied accordingly. While this approach may make the code a bit more cumbersome at times, it helps us to better visualize the circuit architecture and avoid unintended memory and subtle mistakes.

Based on the characteristics of the next-state logic, we divide sequential circuits into three categories:

- *Regular sequential circuit.* The state transitions in the circuit exhibit a “regular” pattern, as in a counter or shift register. The next-state logic is constructed primarily by a predesigned, “regular” component, such as an incrementor or shifter.
- *FSM.* The state transitions in the circuit do not exhibit a simple, repetitive pattern. The next-state logic is constructed by “random logic” and synthesized from scratch. It should be called a random sequential circuit, but is commonly known as an *FSM* (*finite state machine*).
- *FSMD.* The circuit consists of a regular sequential circuit and an *FSM*. The two parts are known as a *data path* and a *control path*, and the complete circuit is known as an *FSMD* (*FSM with data path*). This type of circuit is used to implement an algorithm represented by *register-transfer* (RT) methodology, which describes system operation by a sequence of data transfers and manipulations among registers.

The three types of circuits are discussed in this and two subsequent chapters.

## 4.2 HDL CODE OF THE FF AND REGISTER

Describing storage components in HDL is a subtle procedure, and there are many ways to do it. In fact, one common problem encountered by a new HDL user is the inference of unintended latches and buffers. Instead of covering all possible forms of syntactic descriptions, we introduce the code templates for several commonly used memory components. Since our development process separates the register and the combinational circuit, these components are sufficient for all designs in this book. The components are:

- D FF
- Register
- Register file

All code templates use always blocks. As discussed in Section 3.3.2, nonblocking assignments should be used for the memory elements, whose basic syntax is

```
[variable_name] <= [expression];
```

This type of assignment can avoid potential race condition and eliminate the discrepancy between simulation and synthesis. This topic is explained in detail in Section 7.1.

### 4.2.1 D FF

We consider three types of D FFs:

- D FF without asynchronous reset
- D FF with asynchronous reset
- D FF with synchronous enable

The first two are the most basic memory components and can be found in the library of any device technology. The third can be constructed from a simple D FF. We include the code since it is a frequently used memory component and can be mapped to the FF of the Spartan-3 device’s logic cell.

**D FF without asynchronous reset** The function table of a D FF is shown in Figure 4.1(a) and the code is shown in Listing 4.1.

**Listing 4.1** D FF without asynchronous reset

---

```

module d_ff
(
    input wire clk,
    input wire d,
    output reg q
);

// body
always @(posedge clk)
    q <= d;

endmodule

```

---

The rising edge is expressed by the **posedge** clk event in the sensitivity list. The **posedge** (for “positive edge”) keyword specifies the direction of the clk signal changing toward 1. It indicates that the always block is activated only at the rising edge of the clk signal, a condition reflecting the characteristics of an edge-triggered FF. Note that the d signal is not included in the sensitive list. This is consistent with the fact that the d signal is sampled only at the rising edge of the clk signal, and a change in its value does not trigger any immediate response.

**D FF with asynchronous reset** A D FF may contain an asynchronous reset signal, as shown in the function table of Figure 4.1(b). The signal clears the D FF to 0 any time and is not controlled by the clock signal. It actually has a higher priority than the regularly sampled input. Using an asynchronous reset signal violates the synchronous design methodology and thus should be avoided in normal operation. Its major application is to perform system initialization. For example, we can generate a short reset pulse to force a system to an initial state after turning on the power. The code for a D FF with asynchronous reset is shown in Listing 4.2.

**Listing 4.2** D FF with asynchronous reset

---

```

module d_ff_reset
(
    input wire clk, reset,
    input wire d,
    output reg q
);

// body
always @(posedge clk, posedge reset)
    if (reset)
        q <= 1'b0;
    else
        q <= d;

endmodule

```

---

Note that the **posedge** reset event is also included in the sensitivity list and its value is checked first in the if statement. The q signal is cleared to 0 if it is asserted and its operation is independent of the clk signal.



Figure 4.3 D FF with synchronous enable.

**D FF with synchronous enable** A D FF may include an additional control signal, en, to enable the FF to sample the input value. Its symbol and functional table are shown in Figure 4.1(c). Note that the en signal is examined only at the rising edge of the clock and thus is synchronous. If it is not asserted, the FF keeps its previous value. The code is shown in Listing 4.3.

Listing 4.3 One-segment coding style for a D FF with synchronous enable

---

```

module d_ff_en_1seg
(
    input wire clk, reset,
    input wire en,
    input wire d,
    output reg q
);

// body
10  always @ (posedge clk, posedge reset)
    if (reset)
        q <= 1'b0;
    else if (en)
        q <= d;
15
endmodule

```

---

Note that there is no else branch after the second if statement. According to Verilog definition, a variable keeps its previous value if it is not assigned. If en is 0, q keeps its previous value. Thus, omission of the else branch describes the desired behavior of this FF.

The enabling feature of this D FF is useful in maintaining synchronism between a fast subsystem and a slow subsystem. For example, assume that the operation rates of a fast and a slow subsystem are 50 MHz and 1 MHz. Instead of using a derived 1-MHz clock to drive the slow subsystem, we can generate a periodic enable tick that is asserted one clock cycle every 50 clock cycles. The slow subsystem is disabled (i.e., keeps the previous state) for the remaining 49 clock cycles. The same scheme can also be applied to eliminate a gated clock signal.

Since the enable signal is synchronous, this circuit can be constructed by a regular D FF and simple next-state logic. The code is shown in Listing 4.4, and its block diagram is shown in Figure 4.3.

**Listing 4.4** Two-segment coding style for a D FF with synchronous enable

---

```

module d_ff_en_2seg
(
    input wire clk, reset,
    input wire en,
    input wire d,
    output reg q
);

// signal declaration
reg r_reg, r_next;

// body
// D FF
always @ (posedge clk, posedge reset)
    if (reset)
        r_reg <= 1'b0;
    else
        r_reg <= r_next;

// next-state logic
always @@
    if (en)
        r_next = d;
    else
        r_next = r_reg;

// output logic
always @@
    q = r_reg;

```

---

For clarity, we use suffixes `_next` and `_reg` to emphasize the next input value and the registered output of an FF. They are connected to the `d` and `q` signals of a D FF. The code in Listing 4.3 can be considered as shorthand for this more explicit description.

#### 4.2.2 Register

A register is a collection of D FFs that are controlled by the same clock and reset signals. Like a D FF, a register can have an optional asynchronous reset signal and a synchronous enable signal. The code is identical to that of a D FF except that the array data type is needed for the relevant input and output signals. For example, an 8-bit register with asynchronous reset is shown in Listing 4.5.

**Listing 4.5** Register

---

```

module reg_reset
(
    input wire clk, reset,
    input wire [7:0] d,
    output reg [7:0] q
);

```

---

```

    // body
    always @(posedge clk, posedge reset)
10      if (reset)
        q <= 0;
    else
        q <= d;

15 endmodule

```

---

#### 4.2.3 Register file

A register file is a collection of registers with one input port and one or more output ports. The write address signal, `w_addr`, specifies where to store data, and the read address signal, `r_addr`, specifies where to retrieve data. The register file is generally used as fast, temporary storage. The code for a parameterized  $2^W$ -by- $B$  register file is shown in Listing 4.6. Two parameters are defined in this design:  $W$  specifies the number of address bits, which implies that there are  $2^W$  words in the file, and  $B$  specifies the number of bits in a word.

**Listing 4.6** Parameterized register file

```

module reg_file
#
  parameter B = 8, // number of bits
           W = 2 // number of address bits
5   )
(
  input wire clk,
  input wire wr_en,
  input wire [W-1:0] w_addr, r_addr,
10  input wire [B-1:0] w_data,
  output wire [B-1:0] r_data
);

// signal declaration
15 reg [B-1:0] array_reg [2**W-1:0];

// body
// write operation
always @(posedge clk)
20   if (wr_en)
       array_reg[w_addr] <= w_data;
// read operation
  assign r_data = array_reg[r_addr];

25 endmodule

```

---

The code includes several new features. First, a two-dimensional array data type is defined, as in

```
reg [B-1:0] array_reg [2**W-1:0];
```

It indicates that the `array_reg` variable is an array of `[2**W-1:0]` elements and each element is with the data type of `reg [B-1:0]`. Second, a signal is used as an index to access an

element in the array, as in `array_reg[w_addr]`. Although the description is very abstract, Xilinx software recognizes this language construct and can derive the correct implementation accordingly. The `array_reg[...] = ...` and `... = array_reg[...]` statements infer decoding and multiplexing logic, respectively.

Some applications may need to retrieve multiple data words at the same time. This can be done by adding an additional read port:

```
r_data2 = array_reg[r_addr_2];
```

#### 4.2.4 Storage components in a Spartan-3 device *Xilinx specific*

In a Spartan-3 device, each logic cell contains a D FF with asynchronous reset and synchronous enable. These D FFs basically constitute the register of Figure 4.2. Since a logic cell also contains a four-input LUT, it will be wasteful if the cell is used simply as 1 bit of a massive storage. The Spartan-3 device also has distributed RAM (random access memory) and block RAM modules, and they can be used for larger storage requirements. These modules can be configured for synchronous operation, and their characteristics are somewhat like a restricted version of the register file. The configuration and inference of these modules are discussed in Chapter 12.

### 4.3 SIMPLE DESIGN EXAMPLES

We illustrate the construction of several simple, representative sequential circuits in this section.

#### 4.3.1 Shift register

**Free-running shift register** A free-running shift register shifts its content to the left or right by one position in each clock cycle. There is no other control signal. The code for an  $N$ -bit free-running shift-right register is shown in Listing 4.7.

**Listing 4.7** Free-running shift register

---

```
module free_run_shift_reg
#(parameter N=8)
(
    input wire clk, reset,
    input wire s_in,
    output wire s_out
);

// signal declaration
10 reg [N-1:0] r_reg;
wire [N-1:0] r_next;

// body
// register
15 always @ (posedge clk, posedge reset)
    if (reset)
        r_reg <= 0;
    else
```

```

        r_reg <= r_next;
20
    // next-state logic
    assign r_next = {s_in, r_reg[N-1:1]};
    // output logic
    assign s_out = r_reg[0];
25
endmodule

```

---

The next-state logic is a 1-bit shifter, which shifts `r_reg` right one position and inserts the serial input, `s_in`, to the MSB. Since the 1-bit shifter involves only reconnection of the input and output signals, no real logic is needed. Its propagation delay represents the smallest possible  $T_{comb}$ , and the corresponding  $f_{max}$  represents the highest clock rate that can be achieved for a given device technology.

**Universal shift register** A universal shift register can load parallel data, shift its content left or right, or remain in the same state. It can perform parallel-to-serial operation (first loading parallel input and then shifting) or serial-to-parallel operation (first shifting and then retrieving parallel output). The desired operation is specified by a 2-bit control signal, `ctrl`. The code is shown in Listing 4.8.

**Listing 4.8** Universal shift register

```

module univ_shift_reg
#(parameter N=8)
(
    input wire clk, reset,
    5      input wire [1:0] ctrl,
    input wire [N-1:0] d,
    output wire [N-1:0] q
);

10    // signal declaration
    reg [N-1:0] r_reg, r_next;

    // body
    // register
15    always @ (posedge clk, posedge reset)
        if (reset)
            r_reg <= 0;
        else
            r_reg <= r_next;

20    // next-state logic
    always @*
        case (ctrl)
            2'b00: r_next = r_reg;                      // no op
            2'b01: r_next = {r_reg[N-2:0], d[0]};       // shift left
            2'b10: r_next = {d[N-1], r_reg[N-1:1]};     // shift right
            default: r_next = d;                         // load
        endcase
    // output logic
30    assign q = r_reg;

```

---

**endmodule**

The next-state logic uses a 4-to-1 multiplexer to select the desired next value of the register. Note that the LSB and MSB of d (i.e., d[0] and d[N-1]) are used as serial input for the shift-left and shift-right operations.

In a Xilinx Spartan-3 device, a logic cell's 4-input LUT is implemented by a 16-by-1 SRAM. The same SRAM can also be configured as a cascading chain of sixteen 1-bit SRAM cells, which resembles a 16-bit shift register. This can be used to construct certain forms specific of shift register and leads to very efficient implementation.

#### 4.3.2 Binary counter and variant

**Free-running binary counter** A free-running binary counter circulates through a binary sequence repeatedly. For example, a 4-bit binary counter counts from "0000", "0001", ..., to "1111" and wraps around. The code for a parameterized  $N$ -bit free-running binary counter is shown in Listing 4.9.

**Listing 4.9** Free-running binary counter

---

```

module free_run_bin_counter
  #(parameter N=8)
  (
    input wire clk, reset,
    output wire max_tick,
    output wire [N-1:0] q
  );

  // signal declaration
  reg [N-1:0] r_reg;
  wire [N-1:0] r_next;

  // body
  // register
  always @(posedge clk, posedge reset)
    if (reset)
      r_reg <= 0; // {N{1'b'0}}
    else
      r_reg <= r_next;

  // next-state logic
  assign r_next = r_reg + 1;
  // output logic
  assign q = r_reg;
  assign max_tick = (r_reg==2**N-1) ? 1'b1 : 1'b0;
  // can also use (r_reg=={N{1'b1}})

endmodule

```

---

The next-state logic is an incrementor, which adds 1 to the register's current value. By definition of the + operator, the addition implicitly wraps around after the r\_reg reaches "1...1". The circuit also consists of an output status signal, max\_tick, which is asserted when the counter reaches the maximal value, "1...1" (which is equal to  $2^N - 1$ ).

**Table 4.1** Function table of a universal binary counter

| syn_clr | load | en | up | q*        | Operation         |
|---------|------|----|----|-----------|-------------------|
| 1       | —    | —  | —  | 00 ··· 00 | synchronous clear |
| 0       | 1    | —  | —  | d         | parallel load     |
| 0       | 0    | 1  | 1  | q+1       | count up          |
| 0       | 0    | 1  | 0  | q-1       | count down        |
| 0       | 0    | 0  | —  | q         | pause             |

The `max_tick` signal represents a special type of signal that is asserted for a single clock cycle. In this book, we call this type of signal a *tick* and use the suffix `_tick` to indicate a signal with this property. It is commonly used to interface with the enable signal of other sequential circuits.

**Universal binary counter** A universal binary counter is more versatile. It can count up or down, pause, be loaded with a specific value, or be synchronously cleared. Its functions are summarized in Table 4.1. Note the difference between the `reset` and `syn_clr` signals. The former is asynchronous and should only be used for system initialization. The latter is sampled at the rising edge of the clock and can be used in normal synchronous design. The code for this counter is shown in Listing 4.10.

**Listing 4.10** Universal binary counter

---

```

module univ_bin_counter
  #(parameter N=8)
  (
    input wire clk, reset,
    input wire syn_clr, load, en, up,
    input wire [N-1:0] d,
    output wire max_tick, min_tick,
    output wire [N-1:0] q
  );
  // signal declaration
  reg [N-1:0] r_reg, r_next;
  // body
  // register
  always @ (posedge clk, posedge reset)
    if (reset)
      r_reg <= 0; // 
    else
      r_reg <= r_next;
  // next-state logic
  always @*
    if (syn_clr)
      r_next = 0;
    else if (load)
      r_next = d;
    else if (en & up)

```

---

```

      r_next = r_reg + 1;
30    else if (en & ~up)
        r_next = r_reg - 1;
    else
        r_next = r_reg;

35    // output logic
assign q = r_reg;
assign max_tick = (r_reg==2**N-1) ? 1'b1 : 1'b0;
assign min_tick = (r_reg==0) ? 1'b1 : 1'b0;

40 endmodule

```

---

The next-state logic follows the functional table and is described by an always block, which contains an if statement to prioritize the desired operations.

**Mod- $m$  counter** A mod- $m$  counter counts from 0 to  $m - 1$  and wraps around. A parameterized mod- $m$  counter is shown in Listing 4.11. It has two parameters: M, which specifies the limit,  $m$ ; and N, which specifies the number of bits needed and should be equal to  $\lceil \log_2 M \rceil$ . The code is shown in Listing 4.11, and the default value is for a mod-10 counter.

**Listing 4.11** Mod- $m$  counter

```

module mod_m_counter
#(
    parameter N=4, // number of bits in counter
              M=10 // mod-M
5      )
(
    input wire clk, reset,
    output wire max_tick,
    output wire [N-1:0] q
10 );
// signal declaration
reg [N-1:0] r_reg;
wire [N-1:0] r_next;
15
// body
// register
always @ (posedge clk, posedge reset)
    if (reset)
        r_reg <= 0;
    else
        r_reg <= r_next;

// next-state logic
20 assign r_next = (r_reg==(M-1)) ? 0 : r_reg + 1;
// output logic
assign q = r_reg;
assign max_tick = (r_reg==(M-1)) ? 1'b1 : 1'b0;

25
30 endmodule

```

---

The next-state logic is constructed by a conditional operator. If the counter reaches  $M-1$ , the new value is cleared to 0. Otherwise, it is incremented by 1.

Inclusion of the  $N$  parameter in the code is somewhat redundant since its value depends on  $M$ . A more elegant way is to define a function that calculates  $N$  from  $M$  automatically. This scheme is discussed in Section 7.4.

#### 4.4 TESTBENCH FOR SEQUENTIAL CIRCUITS

A testbench is a program that mimics a physical lab bench, as discussed in Section 1.7. In this section, we illustrate the construction of a simple testbench for the previous universal binary counter. It can serve as a template for other sequential circuits. Development of a more sophisticated testbench is discussed in Section 7.5. The code for the simple testbench is shown in Listing 4.12.

**Listing 4.12** Testbench for a universal binary counter

---

```

'timescale 1 ns/10 ps

// The 'timescale directive specifies that
// the simulation time unit is 1 ns and
// the simulator timestep is 10 ps

module bin_counter_tb();

    // declaration
10    localparam T=20; // clock period
    reg clk, reset;
    reg syn_clr, load, en, up;
    reg [2:0] d;
    wire max_tick, min_tick;
    wire [2:0] q;

    // uut instantiation
    univ_bin_counter #(N(3)) uut
        (.clk(clk), .reset(reset), .syn_clr(syn_clr),
20        .load(load), .en(en), .up(up), .d(d),
        .max_tick(max_tick), .min_tick(min_tick), .q(q));

    // clock
    // 20 ns clock running forever
25    always
        begin
            clk = 1'b1;
            #(T/2);
            clk = 1'b0;
            #(T/2);
        end

    // reset for the first half cycle
30    initial
        begin
            reset = 1'b1;

```

```

#(T/2);
reset = 1'b0;
end
40
// other stimulus
initial
begin
    // ===== initial input =====
45    syn_clr = 1'b0;
    load = 1'b0;
    en = 1'b0;
    up = 1'b1; // count up
    d = 3'b000;
50    @(negedge reset); // wait reset to deassert
    @(negedge clk); // wait for one clock
    // ===== test load =====
    load = 1'b1;
    d = 3'b011;
55    @(negedge clk); // wait for one clock
    load = 1'b0;
    repeat(2) @(negedge clk);
    // ===== test syn_clear =====
    syn_clr = 1'b1; // assert clear
60    @(negedge clk);
    syn_clr = 1'b0;
    // ===== test up counter and pause =====
    en = 1'b1; // count
    up = 1'b1;
65    repeat(10) @(negedge clk);
    en = 1'b0; // pause
    repeat(2) @(negedge clk);
    en = 1'b1;
    repeat(2) @(negedge clk);
70    // ===== test down counter =====
    up = 1'b0;
    repeat(10) @(negedge clk);
    // ===== wait statement =====
    // continue until q=2
75    wait(q==2);
    @(negedge clk);
    up = 1'b1;
    // continue until min_tick becomes 1
    @(negedge clk);
    wait(min_tick);
80    @(negedge clk);
    up = 1'b0;
    // ===== absolute delay =====
    #(4*T); // wait for 80 ns
85    en = 1'b0; // pause
    #(4*T); // wait for 80 ns
    // ===== stop simulation =====
    // return to interactive simulation mode
$stop;

```

```
90      end  
endmodule
```

---

The code consists of a component instantiation statement, which creates an instance of a 3-bit counter, and three segments, which generate a stimulus for clock, reset, and regular inputs.

The clock generation is specified by an always block:

```
always  
  begin  
    clk = 1'b1;  
    #(T/2);  
    clk = 1'b0;  
    #(T/2);  
  end
```

The T term is a constant that represents the number of time units in a clock period. It is defined as

```
localparam T=20; // clock period
```

Note that the always block has no sensitivity list and repeats itself forever. The **clk** signal is assigned between 0 and 1 alternately, and each value lasts for half a period.

The reset stimulus is specified by an initial block:

```
initial  
  begin  
    reset = 1'b1;  
    #(T/2);  
    reset = 1'b0;  
  end
```

An initial block is executed *once* at the beginning of a simulation. It indicates that the **reset** signal is set to 1 initially and changed to 0 after half a period. The block represents the “power-on” condition, in which the **reset** signal is asserted momentarily to clear the system to the initial state. Note that, by default, the x value (for unknown), not 0, is assigned to a variable. Using a short reset pulse is a good mechanism for performing system initialization.

The second initial block generates a stimulus for other input signals. We first test the load and clear operations and then exercise counting in both directions. The final **\$stop** function forces the simulator to stop simulation.

For a synchronous system with positive edge-triggered FFs, an input signal must be stable around the rising edge of the clock signal to satisfy the setup and hold time constraints. One easy way to achieve this is to change an input signal’s value during the 1-to-0 transition of the **clk** signal. We can wait for this transition edge by using

```
@(negedge clk);
```

The **negedge** **clk** event specifies the condition that the **clk** signal changes to 0 (i.e., negative edge). Note that each statement represents a new falling edge, which corresponds to the advancement of one clock cycle. In our template, we generally use this statement to specify the progress of time. For multiple clock cycles, we can use a repeat statement, as in

```
repeat(10) @(negedge clk); // repeat 10 times
```

Several additional timing control constructs are shown at the end of the initial block. We can wait until a special condition, such as “when q is equal to 2”



**Figure 4.4** Testbench waveform.

```
wait(q==2);
```

or wait until a signal changes, such as

```
wait(min_tick);
```

or wait for an absolute time, such as

```
#(4*T); // wait for 4T
```

If an input signal is modified after these statements, we need to make sure that the input change does not occur at the rising edge of the clock. An additional

```
@(negedge clk);
```

statement should be added when needed.

We can compile the code and perform simulation. Part of the simulated waveform is shown in Figure 4.4.

## 4.5 CASE STUDY

After examining several simple circuits, we discuss the design of more sophisticated examples in this section.

### 4.5.1 LED time-multiplexing circuit

The S3 board has four seven-segment LED displays, each containing seven bars and one small round dot. To reduce the use of FPGA's I/O pins, the S3 board uses a time-multiplexing sharing scheme. In this scheme, the four displays have their individual enable signals but share eight common signals to light the segments. All signals are active low (i.e., enabled when a signal is 0). The schematic of displaying a "3" on the rightmost LED is shown in Figure 4.5. Note that the enable signal (i.e., an) is "1110". This configuration clearly can enable only one display at a time. We can *time-multiplex* the four LED patterns by enabling the four displays in turn, as shown in the simplified timing diagram in Figure 4.6. If the refreshing rate of the enable signal is fast enough, the human eye cannot distinguish the on and off intervals of the LEDs and perceives that all four displays are lit simultaneously. This scheme reduces the number of I/O pins from 32 to 12 (i.e., eight LED segments plus four enable signals) but requires a time-multiplexing circuit. Two variations of the circuit are discussed in the following subsections.



**Figure 4.5** Time-multiplexed seven-segment LED display.



**Figure 4.6** Timing diagram of a time-multiplexed seven-segment LED display.



**Figure 4.7** Symbol and block diagram of a time-multiplexing circuit.

**Time multiplexing with LED patterns** The symbol and block diagram of the time-multiplexing circuit are shown in Figure 4.7. It takes four seven-segment LED patterns,  $\text{in}3$ ,  $\text{in}2$ ,  $\text{in}1$ , and  $\text{in}0$ , and passes them to the output,  $\text{sseg}$ , in accordance with the enable signal.

The refresh rate of the enable signal has to be fast enough to fool our eyes but should be slow enough so that the LEDs can be turned on and off completely. The rate around the range 1000 Hz should work properly. In our design, we use an 18-bit binary counter for this purpose. The two MSBs are decoded to generate the enable signal and are used as the selection signal for multiplexing. The refreshing rate of an individual bit, such as  $\text{an}[0]$ , becomes  $\frac{50M}{2^{16}}$  Hz, which is about 800 Hz. The code is shown in Listing 4.13.

**Listing 4.13** LED time-multiplexing circuit with LED patterns

---

```

module disp_mux
(
    input wire clk, reset,
    input [7:0] in3, in2, in1, in0,
    output reg [3:0] an,      // enable. 1-out-of-4 asserted low
    output reg [7:0] sseg // led segments
);

    // constant declaration
    // refreshing rate around 800 Hz (50 MHz/2^16)
    localparam N = 18;

```

```

    // signal declaration
    reg [N-1:0] q_reg;
15   wire [N-1:0] q_next;

    // N-bit counter
    // register
    always @(posedge clk, posedge reset)
20      if (reset)
        q_reg <= 0;
      else
        q_reg <= q_next;

25   // next-state logic
   assign q_next = q_reg + 1;

    // 2 MSBs of counter to control 4-to-1 multiplexing
    // and to generate active-low enable signal
30   always @*
      case (q_reg[N-1:N-2])
        2'b00:
          begin
            an = 4'b1110;
35            sseg = in0;
          end
        2'b01:
          begin
            an = 4'b1101;
40            sseg = in1;
          end
        2'b10:
          begin
            an = 4'b1011;
45            sseg = in2;
          end
        default:
          begin
            an = 4'b0111;
50            sseg = in3;
          end
      endcase

    endmodule

```

We use the testing circuit in Figure 4.8 to verify operation of the LED time-multiplexing circuit. It uses four 8-bit registers to store the LED patterns. The registers use the same 8-bit switch as input but are controlled by individual enable signals. When we press a pushbutton, the corresponding register is enabled and the switch pattern is loaded to that register. The code is shown in Listing 4.14.

**Listing 4.14** Testing circuit for time multiplexing with LED patterns

---

```

module disp_mux_test
(
  input wire clk,

```



**Figure 4.8** LED time-multiplexing testing circuit.

```

5      input wire [3:0] btn,
6      input wire [7:0] sw,
7      output wire [3:0] an,
8      output wire [7:0] sseg
9      );

10     // signal declaration
11    reg [7:0] d3_reg, d2_reg, d1_reg, d0_reg;

12    // instantiate 7-seg LED display time-multiplexing module
13    disp_mux disp_unit
14        (.clk(clk), .reset(1'b0), .in0(d0_reg), .in1(d1_reg),
15         .in2(d2_reg), .in3(d3_reg), .an(an), .sseg(sseg));

16    // registers for 4 led patterns
17    always @(posedge clk)
18    begin
19        if (btn[3])
20            d3_reg <= sw;
21        if (btn[2])
22            d2_reg <= sw;
23        if (btn[1])
24            d1_reg <= sw;
25        if (btn[0])
26            d0_reg <= sw;
27    end

28
29 endmodule

```



**Figure 4.9** Block diagram of a hexadecimal time-multiplexing circuit.

**Time multiplexing with hexadecimal digits** The most common application of a seven-segment LED is to display a hexadecimal digit. The decoding circuit is discussed in Section 3.9.1. To display four hexadecimal digits with the previous time-multiplexing circuit, four decoding circuits are needed. A better alternative is first to multiplex the hexadecimal digits and then decode the result, as shown in Figure 4.9.

This scheme requires only one decoding circuit and reduces the width of the 4-to-1 multiplexer from 8 bits to 5 bits (i.e., 4 bits for the hexadecimal digit and 1 bit for the decimal point). The code is shown in Listing 4.15. In addition to clock and reset, the input consists of four 4-bit hexadecimal digits, hex3, hex2, hex1, and hex0, and four decimal points, which are grouped as one signal, dp\_in.

**Listing 4.15** LED time-multiplexing circuit with hexadecimal digits

---

```

module disp_hex_mux
(
    input wire clk, reset,
    input wire [3:0] hex3, hex2, hex1, hex0, // hex digits
    input wire [3:0] dp_in, // 4 decimal points
    output reg [3:0] an, // enable 1-out-of-4 asserted low
    output reg [7:0] sseg // led segments
);

// constant declaration
// refreshing rate around 800 Hz (50 MHz/2^16)
localparam N = 18;
// internal signal declaration
reg [N-1:0] q_reg;
wire [N-1:0] q_next;
reg [3:0] hex_in;
reg dp;

// N-bit counter
// register
always @(posedge clk, posedge reset)

```

```

if (reset)
    q_reg <= 0;
else
    q_reg <= q_next;

// next-state logic
assign q_next = q_reg + 1;

// 2 MSBs of counter to control 4-to-1 multiplexing
// and to generate active-low enable signal
always @*
    case (q_reg[N-1:N-2])
        2'b00:
            begin
                an = 4'b1110;
                hex_in = hex0;
                dp = dp_in[0];
            end
        2'b01:
            begin
                an = 4'b1101;
                hex_in = hex1;
                dp = dp_in[1];
            end
        2'b10:
            begin
                an = 4'b1011;
                hex_in = hex2;
                dp = dp_in[2];
            end
        default:
            begin
                an = 4'b0111;
                hex_in = hex3;
                dp = dp_in[3];
            end
    endcase

// hex to seven-segment led display
always @*
begin
    case(hex_in)
        4'h0: sseg[6:0] = 7'b0000001;
        4'h1: sseg[6:0] = 7'b1001111;
        4'h2: sseg[6:0] = 7'b0010010;
        4'h3: sseg[6:0] = 7'b0000110;
        4'h4: sseg[6:0] = 7'b1001100;
        4'h5: sseg[6:0] = 7'b0100100;
        4'h6: sseg[6:0] = 7'b0100000;
        4'h7: sseg[6:0] = 7'b0001111;
        4'h8: sseg[6:0] = 7'b0000000;
        4'h9: sseg[6:0] = 7'b0000100;
        4'ha: sseg[6:0] = 7'b0001000;
    endcase
end

```

```

75      4'hb: sseg[6:0] = 7'b1100000;
    4'hc: sseg[6:0] = 7'b0110001;
    4'hd: sseg[6:0] = 7'b1000010;
    4'he: sseg[6:0] = 7'b0110000;
    default: sseg[6:0] = 7'b0111000; //4'hf
80  endcase
    sseg[7] = dp;
end

endmodule

```

---

To verify operation of this circuit, we define the 8-bit switch as two 4-bit unsigned numbers, add the two numbers, and show the two numbers and their sum on the four-digit seven-segment LED display. The code is shown in Listing 4.16.

**Listing 4.16** Testing circuit for time multiplexing with hexadecimal digits

```

module hex_mux_test
(
    input wire clk,
    input wire [7:0] sw,
    output wire [3:0] an,
    output wire [7:0] sseg
);

// signal declaration
10   wire [3:0] a, b;
    wire [7:0] sum;

// instantiate 7-seg LED display module
disp_hex_mux disp_unit
15   (.clk(clk), .reset(1'b0),
    .hex3(sum[7:4]), .hex2(sum[3:0]), .hex1(b), .hex0(a),
    .dp_in(4'b1011), .an(an), .sseg(sseg));

// adder
20   assign a = sw[3:0];
    assign b = sw[7:4];
    assign sum = {4'b0,a} + {4'b0,b};

endmodule

```

---

**Simulation consideration** Many sequential circuit examples in the book operate at a relatively slow rate, as does the enable pulse of the LED time-multiplexing circuit. This can be done by generating a single-clock enable tick from a counter. An 18-bit counter is used in this circuit:

```

localparam N = 18;
reg [N-1:0] q_reg;
wire [N-1:0] q_next;
. .
assign q_next = q_reg + 1;

```

Because of the counter's size, simulating this type of circuit consumes a significant amount of computation time (i.e.,  $2^{18}$  clock cycles for one iteration). Since our main interest is in

the multiplexing part of the code, most simulation time is wasted. It is more efficient to use a smaller counter in simulation. We can do this by modifying the constant statement

```
localparam N = 4;
```

when constructing the testbench. This requires only  $2^4$  clock cycles for one iteration and allows us to better exercise and observe the key operations.

Instead of using a constant and modifying code between simulation and synthesis, an alternative is to define N as a parameter. During instantiation, we can assign different values for simulation and synthesis.

#### 4.5.2 Stopwatch

We consider the design of a stopwatch in this subsection. The watch displays the time in three decimal digits, and counts from 00.0 to 99.9 seconds and wraps around. It contains a synchronous clear signal, `clr`, which returns the count to 00.0, and an enable signal, `go`, which enables and suspends the counting. This design is basically a BCD (binary-coded decimal) counter, which counts in BCD format. In this format, a decimal number is represented by a sequence of 4-bit BCD digits. For example,  $139_{10}$  is represented as "0001 0011 1001" and the next number in sequence is  $140_{10}$ , which is represented as "0001 0100 0000".

Since the S3 board has a 50-MHz clock, we first need a mod-5,000,000 counter that generates a one-clock-cycle tick every 0.1 second. The tick is then used to enable counting of the three-digit BCD counter.

**Design I** Our first design of the BCD counter uses a cascading structure of three decade (i.e., mod-10) counters, representing counts of 0.1, 1, and 10 seconds, respectively. The decade counter has an enable signal and generates a one-clock-cycle tick when it reaches 9. We can use these signals to “hook” the three counters. For example, the 10-second counter is enabled only when the enable tick of the mod-5,000,000 counter is asserted and both the 0.1- and 1-second counters are 9. The code is shown in Listing 4.17.

**Listing 4.17** Cascading description for a stopwatch

---

```
module stop_watch_cascade
(
    input wire clk,
    input wire go, clr,
    output wire [3:0] d2, d1, d0
);

// declaration
localparam DVSR = 5000000;
10 reg [22:0] ms_reg;
wire [22:0] ms_next;
reg [3:0] d2_reg, d1_reg, d0_reg;
wire [3:0] d2_next, d1_next, d0_next;
wire d1_en, d2_en, d0_en;
15 wire ms_tick, d0_tick, d1_tick;

// body
// register
always @(posedge clk)
```

```

20   begin
21     ms_reg <= ms_next;
22     d2_reg <= d2_next;
23     d1_reg <= d1_next;
24     d0_reg <= d0_next;
25   end

26
27   // next-state logic
28   // 0.1 sec tick generator: mod=5000000
29   assign ms_next = (clr || (ms_reg==DVSR && go)) ? 4'b0 :
30           (go) ? ms_reg + 1 :
31             ms_reg;
32   assign ms_tick = (ms_reg==DVSR) ? 1'b1 : 1'b0;
33   // 0.1 sec counter
34   assign d0_en = ms_tick;
35   assign d0_next = (clr || (d0_en && d0_reg==9)) ? 4'b0 :
36           (d0_en) ? d0_reg + 1 :
37             d0_reg;
38   assign d0_tick = (d0_reg==9) ? 1'b1 : 1'b0;
39   // 1 sec counter
40   assign d1_en = ms_tick & d0_tick;
41   assign d1_next = (clr || (d1_en && d0_reg==9)) ? 4'b0 :
42           (d1_en) ? d1_reg + 1 :
43             d1_reg;
44   assign d1_tick = (d1_reg==9) ? 1'b1 : 1'b0;
45
46   // 10 sec counter
47   assign d2_en = ms_tick & d0_tick & d1_tick;
48   assign d2_next = (clr || (d2_en && d2_reg==9)) ? 4'b0 :
49           (d2_en) ? d2_reg + 1 :
50             d2_reg;

51
52   // output logic
53   assign d0 = d0_reg;
54   assign d1 = d1_reg;
55   assign d2 = d2_reg;

endmodule

```

Note that all registers are controlled by the same clock signal. This example illustrates how to use a one-clock-cycle enable tick to maintain synchronicity. An inferior approach is to use the output of the lower counter as the clock signal for the next stage. Although it may appear to be simpler, it violates the synchronous design principle and is a very poor practice.

**Design II** An alternative for the three-digit BCD counter is to describe the entire structure in a nested if statement. The nested conditions indicate that the counter reaches .9, 9.9, and 99.9 seconds. The code is shown in Listing 4.18.

**Listing 4.18** Nested if-statement description for a stopwatch

---

```

module stop_watch_if
(
  input wire clk,

```

```

      input wire go, clr,
      output wire [3:0] d2, d1, d0
    );

    // declaration
localparam DVSR = 5000000;
reg [22:0] ms_reg;
wire [22:0] ms_next;
reg [3:0] d2_reg, d1_reg, d0_reg;
reg [3:0] d2_next, d1_next, d0_next;
wire ms_tick;

// body
// register
always @(posedge clk)
begin
  ms_reg <= ms_next;
  d2_reg <= d2_next;
  d1_reg <= d1_next;
  d0_reg <= d0_next;
end

// next-state logic
// 0.1 sec tick generator: mod=5000000
assign ms_next = (clr || (ms_reg==DVSR && go)) ? 4'b0 :
  (go) ? ms_reg + 1 :
  ms_reg;

assign ms_tick = (ms_reg==DVSR) ? 1'b1 : 1'b0;
// 3-digit bcd counter
always @*
begin
  // default: keep the previous value
  d0_next = d0_reg;
  d1_next = d1_reg;
  d2_next = d2_reg;
  if (clr)
    begin
      d0_next = 4'b0;
      d1_next = 4'b0;
      d2_next = 4'b0;
    end
  else if (ms_tick)
    if (d0_reg != 9)
      d0_next = d0_reg + 1;
    else // reach XX9
      begin
        d0_next = 4'b0;
        if (d1_reg != 9)
          d1_next = d1_reg + 1;
        else // reach X99
          begin
            d1_next = 4'b0;
            if (d2_reg != 9)

```

```

d2_next = d2_reg + 1;
else // reach 999
    d2_next = 4'b0;
end
end

// output logic
assign d0 = d0_reg;
assign d1 = d1_reg;
assign d2 = d2_reg;

endmodule

```

---

**Verification circuit** To verify operation of the stopwatch, we can combine it with the previous hexadecimal LED time-multiplexing circuit to display the output of the watch. The code is shown in Listing 4.19. Note that the first digit of the LED is assigned to 0 and the go and clr signals are mapped to two pushbuttons of the S3 board.

**Listing 4.19** Testing circuit for a stopwatch

```

module stop_watch_test
(
    input wire clk,
    input wire [1:0] btn,
    5      output wire [3:0] an,
    output wire [7:0] sseg
);

// signal declaration
10     wire [3:0] d2, d1, d0;

// instantiate 7-seg LED display module
disp_hex_mux disp_unit
(.clk(clk), .reset(1'b0),
15     .hex3(4'b0), .hex2(d2), .hex1(d1), .hex0(d0),
     .dp_in(4'b1101), .an(an), .sseg(sseg));

// instantiate stopwatch
stop_watch_if counter_unit
20     (.clk(clk), .go(btn[1]), .clr(btn[0]),
     .d2(d2), .d1(d1), .d0(d0));

endmodule

```

---

### 4.5.3 FIFO buffer

A FIFO (first-in-first-out) buffer is an “elastic” storage between two subsystems, as shown in the conceptual diagram of Figure 4.10. It has two control signals, wr and rd, for write and read operations. When wr is asserted, the input data is written into the buffer. The read operation is somewhat misleading. The head of the FIFO buffer is normally always available and thus can be read at any time. The rd signal actually acts like a “remove”



**Figure 4.10** Conceptual diagram of a FIFO buffer.

signal. When it is asserted, the first item (i.e., head) of the FIFO buffer is removed and the next item becomes available.

FIFO buffer is a critical component in many applications and the optimized implementation can be quite complex. In this subsection, we introduce a simple, genuine circular-queue-based design. More efficient, device-specific implementation can be found in the Xilinx literature.

**Circular-queue-based implementation** One way to implement a FIFO buffer is to add a control circuit to a register file. The registers in the register file are arranged as a circular queue with two pointers. The *write pointer* points to the head of the queue, and the *read pointer* points to the tail of the queue. The pointer advances one position for each write or read operation. The operation of an eight-word circular queue is shown in Figure 4.11.

A FIFO buffer usually contains two status signals, *full* and *empty*, to indicate that the FIFO is full (i.e., cannot be written) and empty (i.e., cannot be read), respectively. One of the two conditions occurs when the read pointer is equal to the write pointer, as shown in Figure 4.11(a), (f), and (i). The most difficult design task of the controller is to derive a mechanism to distinguish the two conditions. One scheme is to use two FFs to keep track of the empty and full statuses. The FFs are set to 1 and 0 during system initialization and then modified in each clock cycle according to the values of the *wr* and *rd* signals. The code is shown in Listing 4.20.

**Listing 4.20** FIFO buffer

---

```

module fifo
  #(
    parameter B=8, // number of bits in a word
               W=4 // number of address bits
  )
  (
    input wire clk, reset,
    input wire rd, wr,
    input wire [B-1:0] w_data,
    output wire empty, full,
    output wire [B-1:0] r_data
  );

  // signal declaration
  reg [B-1:0] array_reg [2**W-1:0]; // register array
  reg [W-1:0] w_ptr_reg, w_ptr_next, w_ptr_succ;
  reg [W-1:0] r_ptr_reg, r_ptr_next, r_ptr_succ;
  reg full_reg, empty_reg, full_next, empty_next;

```



Figure 4.11 FIFO buffer based on a circular queue.

```

wire wr_en;

20
    // body
    // register file write operation
    always @(posedge clk)
        if (wr_en)
            array_reg[w_ptr_reg] <= w_data;
    // register file read operation
    assign r_data = array_reg[r_ptr_reg];
    // write enabled only when FIFO is not full
    assign wr_en = wr & ~full_reg;

30
    // fifo control logic
    // register for read and write pointers
    always @(posedge clk, posedge reset)
        if (reset)
            begin
                w_ptr_reg <= 0;
                r_ptr_reg <= 0;
                full_reg <= 1'b0;
                empty_reg <= 1'b1;
            end
        else
            begin
                w_ptr_reg <= w_ptr_next;
                r_ptr_reg <= r_ptr_next;
45
                full_reg <= full_next;
                empty_reg <= empty_next;
            end

// next-state logic for read and write pointers
50
always @*
begin
    // successive pointer values
    w_ptr_succ = w_ptr_reg + 1;
    r_ptr_succ = r_ptr_reg + 1;
55
    // default: keep old values
    w_ptr_next = w_ptr_reg;
    r_ptr_next = r_ptr_reg;
    full_next = full_reg;
    empty_next = empty_reg;
    case ({wr, rd})
        // 2'b00: no op
        2'b01: // read
            if (~empty_reg) // not empty
                begin
65
                    r_ptr_next = r_ptr_succ;
                    full_next = 1'b0;
                    if (r_ptr_succ==w_ptr_reg)
                        empty_next = 1'b1;
                end
        2'b10: // write
            if (~full_reg) // not full

```

```

    begin
        w_ptr_next = w_ptr_succ;
        empty_next = 1'b0;
75      if (w_ptr_succ==r_ptr_reg)
            full_next = 1'b1;
        end
2'b11: // write and read
    begin
80      w_ptr_next = w_ptr_succ;
      r_ptr_next = r_ptr_succ;
    end
    endcase
end
85
// output
assign full = full_reg;
assign empty = empty_reg;

90 endmodule

```

---

The code is divided into a register file and a FIFO controller. The controller consists of two pointers and two status FFs. Its next-state logic examines the wr and rd signals and takes actions accordingly. For example, let us consider the "10" case, which implies that only a write operation occurs. The status FF is checked first to ensure that the buffer is not full. If this condition is met, we advance the write pointer by one position and clear the empty status FF. Storing one extra word to the buffer may make it full. This happens if the new write pointer "catches" the read pointer, which is expressed by the `w_ptr_succ==r_ptr_reg` expression.

**Verification circuit** The verification circuit examines the operation of a  $2^4$ -by-3 FIFO buffer. We use three switches to generate the input data and use two buttons for the wr and rd signals. The 3-bit readout and the full and empty status signals are displayed in five discrete LEDs. Because of bounces of the mechanical contact, a debouncing circuit is needed to generate a clean one-clock-cycle tick. The debouncing module, named `debounce`, is discussed in Section 6.2.1 but for now can be treated as a predesigned module. The original button inputs are `btn[0]` and `btn[1]`, and the debounced signals are `db_btn[0]` and `db_btn[1]`. The code is shown in Listing 4.21.

**Listing 4.21** Testing circuit for a FIFO buffer

---

```

module fifo_test
(
    input wire clk, reset,
    input wire [1:0] btn,
5     input wire [2:0] sw,
    output wire [7:0] led
);

// signal declaration
10    wire [1:0] db_btn;

// debounce circuit for btn[0]
debounce btn_db_unit0

```

```

15      (.clk(clk), .reset(reset), .sw(btn[0]),
16          .db_level(), .db_tick(db_btn[0]));
17  // debounce circuit for btn[1]
18  debounce btn_db_unit1
19      (.clk(clk), .reset(reset), .sw(btn[1]),
20          .db_level(), .db_tick(db_btn[1]));
21  // instantiate a 2^2-by-3 fifo
22  fifo #(.B(3), .W(2)) fifo_unit
23      (.clk(clk), .reset(reset),
24          .rd(db_btn[0]), .wr(db_btn[1]), .w_data(sw),
25          .r_data(led[2:0]), .full(led[7]), .empty(led[6]));
26  // disable unused leds
27  assign led[5:3] = 3'b000;

endmodule

```

---

## 4.6 BIBLIOGRAPHIC NOTES

The bibliographic information for this chapter is similar to that for Chapter 3.

## 4.7 SUGGESTED EXPERIMENTS

### 4.7.1 Programmable square-wave generator

A programmable square-wave generator is a circuit that can generate a square wave with variable on (i.e., logic 1) and off (i.e., logic 0) intervals. The durations of the intervals are specified by two 4-bit control signals,  $m$  and  $n$ , which are interpreted as unsigned integers. The on and off intervals are  $m \times 100$  ns and  $n \times 100$  ns, respectively (recall that the period of the S3 onboard oscillator is 20 ns). Design a programmable square-wave generator circuit. The circuit should be completely synchronous. We need a logic analyzer or oscilloscope to verify its operation.

### 4.7.2 PWM and LED dimmer

The duty cycle of a square wave is defined as the percentage of the on interval (i.e., logic 1) in a period. A PWM (pulse width modulation) circuit can generate an output with variable duty cycles. For a PWM with 4-bit resolution, a 4-bit control signal,  $w$ , specifies the duty cycle. The  $w$  signal is interpreted as an unsigned integer and the duty cycle is  $\frac{w}{16}$ .

1. Design a PWM circuit with 4-bit resolution and verify its operation using a logic analyzer or oscilloscope.
2. Modify the LED time-multiplexing circuit to include the PWM circuit for the  $a_n$  signal. The PWM circuit specifies the percentage of time that the LED display is on. We can control the perceived brightness by changing the duty cycle. Verify the circuit's operation by observing 1 bit of  $a_n$  on a logic analyzer or oscilloscope.
3. Replace the LED time-multiplexing circuit of Listing 4.19 with the new design and use the lower 4 bits of the 8-bit switch to control the duty cycle. Verify operation of the circuit. It may be necessary to go to a dark area to see the effect of dimming.

**Figure 4.12** Pattern for Experiment 4.7.3.**Figure 4.13** Pattern for Experiment 4.7.4.

### 4.7.3 Rotating square circuit

In a seven-segment LED display, a square pattern can be created by enabling the a, b, f, and g segments or the c, d, e, and g segments. We want to design a circuit that circulates the square patterns in the four-digit seven-segment LED display. The clockwise circulating pattern is shown in Figure 4.12. The circuit should have an input, en, which enables or pauses the circulation, and an input, cw, which specifies the direction (i.e., clockwise or counterclockwise) of the circulation.

Design the circuit and verify its operation on the prototyping board. Make sure that the circulation rate is slow enough for visual inspection.

### 4.7.4 Heartbeat circuit

We want to create a “heartbeat” for the prototyping board. It repeats the simple pattern in the four-digit seven-segment display, as shown in Figure 4.13, at a rate of 72 Hz. Design the circuit and verify its operation on the prototyping board.

### 4.7.5 Rotating LED banner circuit

The prototyping board has a four-digit seven-segment LED display, and thus only four symbols can be displayed at a time. We can show more information if the data is rotated and moved continuously. For example, assume that the message is 10 digits (i.e., “0123456789”). The display can show the message as “0123”, “1234”, “2345”, . . . , “6789”, “7890”, . . . , “0123”. The circuit should have an input, en, which enables or pauses the rotation, and an input, dir, which specifies the direction (i.e., rotate left or right).

Design the circuit and verify its operation on the prototyping board. Make sure that the rotation rate is slow enough for visual inspection.

### 4.7.6 Enhanced stopwatch

Modify the stopwatch with the following extensions:

- Add an additional signal, up, to control the direction of counting. The stopwatch counts up when the up signal is asserted and counts down otherwise.
- Add a minute digit to the display. The LED display format should be like M.SS.D, where D represents 0.1 second and its range is between 0 and 9, SS represents seconds and its range is between 00 and 59, and M represents minutes and its range is between 0 and 9.

Design the new stopwatch and verify its operation with a testing circuit.

#### 4.7.7 Stack

A stack is a last-in-first-out buffer in which the last stored data is retrieved first. Storing a data word to a stack is known as a *push* operation, and retrieving a data word from a stack is known as a *pop* operation. The I/O signals of a stack are similar to those of a FIFO buffer except that we generally use the push and pop signals in place of the wr and rd signals. Design a stack using a register file and verify its operation with a testing circuit similar to the one in Listing 4.21.

This Page Intentionally Left Blank