# Digital electronics 1 - 08 traffic lights

## Traffic light controller

| Input P  | 0        | 0        | 1        | 1          | 0        | 1        | 0        | 1          | 1          | 1          | 1          | 0          | 0          | 1          | 1          | 1          |
|----------|----------|----------|----------|------------|----------|----------|----------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| Clock    | <b>↑</b> | <b>↑</b> | <b>↑</b> | $\uparrow$ | <b>↑</b> | <b>↑</b> | <b>↑</b> | $\uparrow$ |
| State    | Α        | Α        | В        | С          |          |          |          |            |            |            |            |            |            |            |            |            |
| Output R | 0        | 0        | 0        |            |          |          |          |            |            |            |            |            |            |            |            |            |

## State diagram



## Source code of process p\_traffic\_fsm

```
p_traffic_fsm : process(clk)
   begin
       if rising_edge(clk) then
          s_cnt <= c_ZERO;</pre>
                                    -- Clear all bits
           elsif (s en = '1') then
              -- Every 250 ms, CASE checks the value of the s_state
              -- variable and changes to the next state according
              -- to the delay value.
              case s_state is
                  -- If the current state is STOP1, then wait 1 sec
                  -- and move to the next GO_WAIT state.
                  when STOP1 =>
                      -- Count up to c_DELAY_1SEC
                      if (s_cnt < c_DELAY_1SEC) then
```

```
s_cnt <= s_cnt + 1;
    else
        -- Move to the next state
        s_state <= WEST_GO;</pre>
        -- Reset local counter value
        s_cnt <= c_ZERO;</pre>
    end if;
when WEST_GO =>
    if (s_cnt < c_DELAY_4SEC) then
        s_cnt <= s_cnt + 1;</pre>
    else
        -- Move to the next state
        s_state <= WEST_WAIT;</pre>
        -- Reset local counter value
        s_cnt <= c_ZERO;</pre>
    end if;
when WEST_WAIT =>
    if (s_cnt < c_DELAY_2SEC) then
        s_cnt <= s_cnt + 1;
    else
        -- Move to the next state
        s_state <= STOP2;</pre>
        -- Reset local counter value
        s_cnt <= c_ZERO;</pre>
    end if;
when STOP2 =>
    if (s_cnt < c_DELAY_1SEC) then
        s_cnt <= s_cnt + 1;
    else
        -- Move to the next state
        s_state <= SOUTH_GO;</pre>
        -- Reset local counter value
        s_cnt <= c_ZERO;</pre>
    end if;
when SOUTH GO =>
    if (s_cnt < c_DELAY_4SEC) then
        s_cnt <= s_cnt + 1;
    else
        -- Move to the next state
        s_state <= SOUTH_WAIT;</pre>
        -- Reset local counter value
        s_cnt <= c_ZERO;</pre>
    end if;
when SOUTH_WAIT =>
```

```
if (s_cnt < c_DELAY_2SEC) then
                         s_cnt <= s_cnt + 1;
                     else
                         -- Move to the next state
                         s state <= STOP1;
                         -- Reset local counter value
                         s_cnt <= c_ZERO;</pre>
                     end if;
                -- It is a good programming practice to use the
                -- OTHERS clause, even if all CASE choices have
                -- been made.
                when others =>
                    s_state <= STOP1;</pre>
            end case;
        end if; -- Synchronous reset
    end if; -- Rising edge
end process p_traffic_fsm;
```

### Source code of process p\_output\_fsm

```
p_output_fsm : process(s_state)
    begin
        case s_state is
             when STOP1 =>
                 south_o <= "100"; -- Red (RGB = 100)
                 west o <= "100"; -- Red (RGB = 100)
             when WEST GO =>
                 south_o <= red;</pre>
                 west_o <= green;</pre>
             when WEST_WAIT =>
                 south_o <= red;</pre>
                 west_o <= orange;</pre>
             when STOP2 =>
                 south o <= red;
                 west_o <= red;</pre>
             when SOUTH_GO =>
                 south o <= green;
                 west_o <= red;</pre>
             when SOUTH_WAIT =>
                 south_o <= orange;</pre>
                 west_o <= red;</pre>
             when others =>
                 south_o <= "100"; -- Red
                 west_o <= "100"; -- Red
```

end case;
end process p\_output\_fsm;

## Simulated waveforms of traffic lights



## State table

| Current state | Direction<br>South | Direction<br>West | Delay | No cars  | Cars WEST  | Cars<br>SOUTH | Both<br>Directions |  |
|---------------|--------------------|-------------------|-------|----------|------------|---------------|--------------------|--|
| STOP1         | red                | red               | 1 sec | WAIT1    | WAIT1      | WAIT1         | WAIT1              |  |
| WAIT1         | red                | red               | 0 sec | WEST_GO  | WEST_GO    | SOUTH_GO      | WEST_GO            |  |
| WEST_GO       | red                | green             | 4 sec | WEST_GO  | WEST_GO    | WEST_WAIT     | WEST_WAIT          |  |
| WEST_WAIT     | red                | yellow            | 2 sec | STOP2    | STOP2      | STOP2         | STOP2              |  |
| STOP2         | red                | red               | 1 sec | WAIT2    | WAIT2      | WAIT2         | WAIT2              |  |
| WAIT2         | red                | red               | 0 sec | SOUTH_GO | WEST_GO    | SOUTH_GO      | SOUTH_GO           |  |
| SOUTH_GO      | green              | red               | 4 sec | SOUTH_GO | SOUTH_WAIT | SOUTH_GO      | SOUTH_WAIT         |  |
| SOUTH_WAIT    | yellow             | red               | 2 sec | STOP1    | STOP1      | STOP1         | STOP1              |  |

## State diagram



### Simulated waveforms of smart traffic lights



### Listing of VHDL code of sequential process p\_smart\_traffic\_fsm

```
p_smart_traffic_fsm : process(clk)
    begin
        if rising_edge(clk) then
                                      -- Synchronous reset
            if (reset = '1') then
                s_state <= STOP1 ;</pre>
                                        -- Set initial state
                       <= c_ZERO;
                                         -- Clear all bits
                s_cnt
            elsif (s_en = '1') then
                -- Every 250 ms, CASE checks the value of the s_state
                 -- variable and changes to the next state according
                -- to the delay value.
                case s_state is
                     -- If the current state is STOP1, then wait 1 sec
                     -- and move to the next GO_WAIT state.
                     when STOP1 =>
                         -- Count up to c_DELAY_1SEC
                         if (s_cnt < c_DELAY_1SEC) then
                             s_cnt <= s_cnt + 1;
                         else
                             -- Move to the next state
                             s_state <= WAIT1;</pre>
                             -- Reset local counter value
                             s_cnt <= c_ZERO;</pre>
                         end if;
                     when WAIT1 =>
                         if (SW = "10") then s_state <= SOUTH_GO;</pre>
                         else s state <= WEST GO;
                         end if;
                     when WEST_GO =>
                         if (s_cnt < c_DELAY_4SEC) then
                             s_cnt <= s_cnt + 1;
                         elsif (SW = "10" or SW = "11") then
                             -- Move to the next state
                             s_state <= WEST_WAIT;</pre>
                             -- Reset local counter value
                             s_cnt <= c_ZERO;</pre>
                         else
                             s_state <= WEST_GO;</pre>
```

```
s_cnt <= c_ZERO;</pre>
    end if;
when WEST_WAIT =>
    if (s_cnt < c_DELAY_2SEC) then
        s_cnt <= s_cnt + 1;
    else
        -- Move to the next state
        s_state <= STOP2;</pre>
        -- Reset local counter value
        s_cnt <= c_ZERO;</pre>
    end if;
when STOP2 =>
    if (s_cnt < c_DELAY_1SEC) then
        s_cnt <= s_cnt + 1;
    else
        -- Move to the next state
        s_state <= WAIT2;</pre>
        -- Reset local counter value
        s_cnt <= c_ZERO;</pre>
    end if;
when WAIT2 =>
    if (SW = "01") then s_state <= WEST_GO;</pre>
    else s_state <= SOUTH_GO;</pre>
    end if;
when SOUTH_GO =>
    if (s_cnt < c_DELAY_4SEC) then
        s_cnt <= s_cnt + 1;
    elsif (SW = "01" or SW = "11") then
        -- Move to the next state
        s_state <= SOUTH_WAIT;</pre>
        -- Reset local counter value
        s cnt <= c ZERO;
    else
        s_state <= SOUTH_GO;</pre>
        s cnt <= c ZERO;
    end if;
when SOUTH_WAIT =>
    if (s_cnt < c_DELAY_2SEC) then
        s_cnt <= s_cnt + 1;
    else
        -- Move to the next state
        s_state <= STOP1;</pre>
        -- Reset local counter value
```

GitHub repository