### Finite State Machine

### **Outline**

- 1. Overview
- 2. FSM representation
- 3. Timing and performance of an FSM
- 4. Moore machine versus Mealy machine
- 5. VHDL description of FSMs
- 6. State assignment
- 7. Moore output buffering
- 8. FSM design examples

#### Overview on FSM

- Contain "random" logic in next-state logic
- Used mainly used as a controller in a large system
- Mealy vs Moore output



## 2. Representation of FSM

#### State diagram



Figure 10.2 Notation for a state.

E.g.
 a memory
 controller



RTL Hardware Design by P. Chu

- ASM (algorithmic state machine) chart
  - Flowchart-like diagram
  - Provide the same info as an FSM
  - More descriptive, better for complex description
  - ASM block
    - One state box
    - One ore more optional decision boxes: with T or F exit path
    - One or more conditional output boxes: for Mealy output



Figure 10.4 ASM block.

### State diagram and ASM chart conversion

• E.g. 1.



### • E.g ?





#### • E.g. 3.



• E.g. 4.





#### • E.g. 6.



we\_me <= 1 read1 write we <= 1 re <= 1 burst=1 read2 re <= 1 read3 · re <= 1 read4 re <= 1

RTL Hardware Design by P. Chu

- Difference between a regular flowchart and ASM chart:
  - Transition governed by clock
  - Transition done between ASM blocks
- Basic rules:
  - For a given input combination, there is one unique exit path from the current ASM block
  - The exit path of an ASM block must always lead to a state box. The state box can be the state box of the current ASM block or a state box of another ASM block.

#### Incorrect ASM charts:





#### 3. Performance of FSM

Similar to regular sequential circuit



$$T_c = T_{cq} + T_{next(max)} + T_{setup}$$

$$T_{co(mo)} = T_{cq} + T_{output(mo)}$$

#### Sample timing diagram







# 4. Moore vs Mealy output

- Moore machine:
  - output is a function of state
- Mealy machine:
  - output function of state and output
- From theoretical point of view
  - Both machines have similar "computation capability"
- Implication of FSM as a controller?

- E.g., edge detection circuit
  - A circuit to detect the rising edge of a slow "strobe" input and generate a "short" (about 1-clock period) output pulse.

### • Three designs:





#### Comparison

- Mealy machine uses fewer states
- Mealy machine responds faster
- Mealy machine may be transparent to glitches
- Which one is better?
- Types of control signal
  - Edge sensitive
    - E.g., enable signal of counter
    - Both can be used but Mealy is faster
  - Level sensitive
    - E.g., write enable signal of SRAM
    - Moore is preferred

## VHDL Description of FSM

- Follow the basic block diagram
- Code the next-state/output logic according to the state diagram/ASM chart
- Use enumerate data type for states

#### • E.g. 6.







```
library ieee;
      use ieee.std_logic_1164.all;
      entity mem_ctrl is
      port (
         clk, reset: in std_logic;
         mem, rw, burst: in std_logic;
         oe, we, we_me: out std_logic);
      end mem_ctrl ;
      architecture mult_seg_arch of mem_ctrl is
         type mc_state_type is
             (idle, read1, read2, read3, read4, write);
         signal state_reg, state_next: mc_state_type;
       begin
         -- state register
          process(clk, reset)
          begin
             if (reset='1') then
                state_reg <= idle;
             elsif (clk'event and clk='1') then
                state_reg <= state_next;
RTL Hardw
                                                        27
             end if;
by P. Chu
          end process;
```

```
-- next-state logic
process(state_reg, mem, rw, burst)
begin
   case state_reg is
      when idle =>
          if mem='1' then
             if rw='1' then
                state_next <= read1;
             else
                state_next <= write;
             end if;
          else
             state_next <= idle;
         end if;
       when write =>
          state_next <= idle;
```

```
when read1 =>
           if (burst='1') then
              state_next <= read2;
           else
              state_next <= idle;
           end if;
       when read2 =>
           state_next <= read3;
       when read3 =>
           state_next <= read4;
       when read4 \Rightarrow
           state_next <= idle;
    end case;
end process;
```

```
-- moore output logic
process (state_reg)
begin
   we <= '0'; -- default value
   oe <= '0'; -- default value
   case state_reg is
      when idle =>
      when write =>
         we <= '1':
      when read1 =>
         oe <= '1':
      when read2 =>
         oe <= '1':
      when read3 \Rightarrow
         oe <= '1':
      when read4 =>
         oe <= '1';
   end case;
end process;
```

```
-- mealy output logic
   process(state_reg, mem, rw)
   begin
      we_me <= '0'; -- default value
      case state_reg is
         when idle =>
             if (mem='1') and (rw='0') then
               we_me <= '1';
             end if;
         when write =>
         when read1 =>
         when read2 =>
         when read3 \Rightarrow
         when read4 \Rightarrow
      end case;
   end process;
end mult_seg_arch;
```

#### Combine next-state/output logic together



```
process(state_reg, mem, rw, burst)
begin
 oe <= '0'; — default values
 we <= '0';
 we_me <= '0';
  case state_reg is
   when idle =>
      if mem='1' then
        if rw='1' then
          state_next <= read1;
        else
          state_next <= write;
         we_me <= '1';
        end if;
      else
        state_next <= idle;
      end if;
    when write =>
      state_next <= idle;
     we <= '1';
```

```
when read1 =>
      if (burst='1') then
        state_next <= read2;
      else
        state_next <= idle;
      end if;
      oe <= '1';
    when read 2 = >
      state_next <= read3;
      oe <= '1';
   when read3 \Rightarrow
      state_next <= read4;
      oe <= '1';
    when read4 = >
      state_next <= idle;
      oe <= '1';
   end case;
end process;
```

# 6. State assignment

- State assignment: assign binary representations to symbolic states
- In a synchronous FSM
  - All assignments work
  - Good assignment reduce the complexity of next-state/output logic
- Typical assignment
  - Binary, Gray, one-hot, almost one-hot

Table 10.1 State assignment example

|               | Binary<br>assignment | Gray code<br>assignment | One-hot<br>assignment | Almost one-hot<br>assignment |
|---------------|----------------------|-------------------------|-----------------------|------------------------------|
| idle          | 000                  | 000                     | 000001                | 00000                        |
| read1         | 001                  | 001                     | 000010                | 00001                        |
| read2         | 010                  | 011                     | 000100                | 00010                        |
| read3         | 011                  | 010                     | 001000                | 00100                        |
| ${\tt read4}$ | 100                  | 110                     | 010000                | 01000                        |
| write         | 101                  | 111                     | 100000                | 10000                        |

# State assignment in VHDL

Implicit: use user attributes enum\_encoding

Explicit: use std\_logic\_vector for the register

```
architecture state_assign_arch of mem_fsm is
    constant idle: std_logic_vector(3 downto 0):="0000";
    constant write: std_logic_vector(3 downto 0):="0100";
    constant read1: std_logic_vector(3 downto 0):="1000";
   constant read2: std_logic_vector(3 downto 0):="1001";
5
    constant read3: std_logic_vector(3 downto 0):="1010";
    constant read4: std_logic_vector(3 downto 0):="1011"
    signal state_reg, state_next: std_logic_vector(3 downto 0);
 begin
   -- state register
    process(clk, reset)
    begin
       if (reset='1') then
          state_reg <= idle;
       elsif (clk'event and clk='1') then
5
          state_reg <= state_next;
       end if:
    end process;
```

```
-- next-state logic
process(state_reg, mem, rw, burst)
begin
   we_me <= '0';
   case state_reg is
      when idle =>
          if mem='1' then
             if rw='1' then
                state_next <= read1;
             else
                state_next <= write
             end if;
          else
             state_next <= idle;
          end if;
       when others =>
          state_next <= idle;
    end case;
 end process;
```

### Handling the unused state

- Many binary representations are not used
- What happens if the FSM enters an unused state?
  - Ignore the condition
  - Safe (Fault-tolerant) FSM: got to an error state or return to the initial state.
- Easy for the explicit state assignment
- No portable code for the enumerated data type

# 6. Moore output buffering

- FSM as control circuit
  - Sometimes fast, glitch-free signal is needed
  - An extra output buffer can be added, but introduce one-clock delay
- Special schemes can be used for Moore output
  - Clever state assignment
  - Look-ahead output circuit

- Potential problems of the Moore output logic:
  - Potential hazards introduce glitches
  - Increase the Tco delay (Tco = Tcq + Toutput)
- Can we get control signals directly from the register?



# Clever state assignment

- Assigning state according to output signal patterns
- Output can be obtained from register directly
- Extra register bits may be needed
- Must use explicit state assignment in VHDL code to access individual register bit
- Difficult to revise and maintain



 Table 10.2
 Clever assignment

|       | $q_3q_2 \ (	exttt{oe})  (	exttt{we})$ | $q_1q_0$ | $q_3q_2q_1q_0$ |
|-------|---------------------------------------|----------|----------------|
| idle  | 00                                    | 00       | 0000           |
| read1 | 10                                    | 00       | 1000           |
| read2 | 10                                    | 01       | 1001           |
| read3 | 10                                    | 10       | 1010           |
| read4 | 10                                    | 11       | 1011           |
| write | 01                                    | 00       | 0100           |

RTL Hardware Design by P. Chu

#### VHDL code

```
architecture state_assign_arch of mem_fsm is
   constant idle: std_logic_vector(3 downto 0):="0000";
   constant write: std_logic_vector(3 downto 0):="0100";
   constant read1: std_logic_vector(3 downto 0):="1000";
   constant read2: std_logic_vector(3 downto 0):="1001";
   constant read3: std_logic_vector(3 downto 0):="1010";
   constant read4: std_logic_vector(3 downto 0):="1011"
   signal state_reg,state_next: std_logic_vector(3 downto 0);
```

```
-- Moore output logic
oe <= state_reg(3);
we <= state_reg(2);</pre>
```

# Look-ahead output circuit

- Output buffer introduces one-clock delay
- The "next" value of Moore output can be obtained by using state\_next signal
- Buffer the next value cancel out the one-clock delay
- More systematic and easier to revise and maintain



(a) Moore output with a regular output buffer



(b) Moore output with a look-ahead buffer

- Modification over original VHDL code:
  - Add output buffer
  - Use state\_next to replace state\_reg in Moore output logic

```
-- output buffer
process(clk, reset)
begin
  if (reset='1') then
     oe_buf_reg <= '0';
    we_buf_reg <= '0';
    elsif (clk'event and clk='1') then
     oe_buf_reg <= oe_next;
    we_buf_reg <= we_next;
    end if;
end process;</pre>
```

```
-- moore output logic
process(state_next)
begin
   we_next <= '0'; -- default value
   oe_next <= '0'; -- default value
   case state_next is
      when idle =>
      when write =>
         we_next <= '1';
      when read1 =>
         oe_next <= '1';
      when read2 =>
         oe_next <= '1';
      when read3 =>
         oe_next <= '1';
      when read4 =>
         oe_next <= '1';
   end case;
end process;
```

# 7. FSM design examples

- Edge detector circuit
- Arbitrator (read)
- DRAM strobe signal generation
- Manchester encoding/decoding (read)
- FSM base binary counter

#### Edge detecting circuit (Moore)



RTL Hardware Design by P. Chu

```
library ieee;
use ieee.std_logic_1164.all;
entity edge_detector1 is
port (
   clk, reset: in std_logic;
   strobe: in std_logic;
  p1: out std_logic);
end edge_detector1;
architecture moore_arch of edge_detector1 is
   type state_type is (zero, edge, one);
   signal state_reg, state_next: state_type;
begin
  -- state register
   process(clk, reset)
   begin
      if (reset='1') then
         state_reg <= zero;
      elsif (clk'event and clk='1') then
         state_reg <= state_next;
      end if:
   end process;
```

```
-- next-state logic
   process(state_reg, strobe)
   begin
      case state_reg is
         when zero=>
            if strobe= '1' then
               state_next <= edge;
            else
               state_next <= zero;
            end if;
         when edge =>
            if strobe= '1' then
               state_next <= one;
            else
               state_next <= zero;
            end if;
         when one =>
            if strobe= '1' then
               state_next <= one;
            else
               state_next <= zero;
            end if;
      end case;
   end process;
  -- moore output logic
   p1 <= '1' when state_reg=edge else
        '0':
end moore_arch;
```

#### Use clever state assignment

|      | state_reg(1)<br>(p1) | state_reg(0) |
|------|----------------------|--------------|
| zero | 0                    | 0            |
| edge | 1                    | 0            |
| one  | 0                    | 1            |

```
architecture clever_assign_buf_arch of edge_detector1 is
   constant zero: std_logic_vector(1 downto 0):= "00";
   constant edge: std_logic_vector(1 downto 0):= "10";
   constant one: std_logic_vector(1 downto 0) := "01";
   signal state_reg, state_next: std_logic_vector(1 downto 0);
```

```
-- moore output logic
p1 <= state_reg(1);</pre>
```

#### Use look-ahead output

```
process(clk, reset)
begin
   if (reset='1') then
      p1_reg <= '0';
elsif (clk'event and clk='1') then
      p1_reg <= p1_next;
end if;
end process;</pre>
```

#### Edge detecting circuit (Mealy)



```
architecture mealy_arch of edge_detector2 is
   type state_type is (zero, one);
   signal state_reg, state_next: state_type;
begin
    -- state register
   process(clk, reset)
   begin
        if (reset='1') then
            state_reg <= zero;
        elsif (clk'event and clk='1') then
            state_reg <= state_next;
        end if;</pre>
```

```
process(state_reg, strobe)
 begin
    case state_reg is
       when zero=>
          if strobe= '1' then
             state_next <= one;
          else
             state_next <= zero;
          end if;
       when one =>
          if strobe= '1' then
             state_next <= one;
          else
             state_next <= zero;
          end if;
   end case;
end process;
-- mealy output logic
p2 <= '1' when (state_reg=zero) and (strobe='1') else
       , 0 , :
```

-- next-state logic

- Edge detecting circuit (direct implementation):
  - edge occurs when previous value is 0 and new value is 1
  - Same as Mealy design with state assignment: zero => 0, one => 1



Figure 10.19 Direct implementation of an edge detector.

```
architecture direct_arch of edge_detector2 is
   signal delay_reg: std_logic;
begin
  -- delay register
  process (clk, reset)
  begin
      if (reset='1') then
         delay_reg <= '0';
      elsif (clk'event and clk='1') then
         delay_reg <= strobe;
      end if;
  end process;
  -- decoding logic
  p2 <= (not delay_reg) and strobe;
end direct_arch;
```

#### DRAM strobe signal generation

– E.g.,120ns DRAM (Trc=120ns):Tras=85ns, Tcas=20ns, Tpr=35ns



(a) Simplified timing of a DRAM read cycle

- 3 intervals has to be at least 65ns, 20 ns, and 35 ns
- A slow design: use a 65ns clock period
  - 195 ns (3\*65ns) read cycle
- The control signal is level-sensitive





```
library ieee;
use ieee.std_logic_1164.all;
entity dram_strobe is
   port (
      clk, reset: in std_logic;
      mem: in std_logic;
      cas_n, ras_n: out std_logic
      );
end dram_strobe;
architecture fsm_slow_clk_arch of dram_strobe is
   type fsm_state_type is (idle,r,c,p);
   signal state_reg, state_next: fsm_state_type;
begin
  -- state register
   process(clk, reset)
   begin
      if (reset='1') then
         state_reg <= idle;
      elsif (clk'event and clk='1') then
         state_reg <= state_next;
      end if:
   end process;
```

```
-- output logic
-- next-state logic
                                    process(state_reg)
process(state_reg, mem)
                                    begin
begin
                                       ras_n <= '1';
   case state_reg is
                                       cas_n <= '1';
      when idle =>
         if mem='1' then
                                       case state_reg is
            state_next <= r;
                                          when idle =>
         else
                                          when r =>
            state_next <= idle;
                                              ras_n <= '0';
         end if;
                                          when c =>
      when r = >
                                              ras_n <= '0';
         state_next <=c;
                                              cas_n <= '0';
      when c = >
                                          when p = >
         state_next <=p;
      when p =>
                                        end case;
         state_next <=idle;
                                    end process;
    end case;
                                end fsm_slow_clk_arch;
end process;
```

Should revise the code to obtain glitch-free output

- A faster design: use a 20ns clock period
  - 140 ns (7\*20ns) read cycle



- FSM-based binary counter:
  - Free-running mod-16 counter



- 4-bit binary counter with features:
  - Synchronous clear, load, enable







RTL Hard by P. Chu