

# LAB 2

PIG GAME

**FIGURE 6-28** PIG: (a) Exterior View of PIG, (b) PIG Registers



(a)



**TABLE 6-17**  
**Inputs, Outputs, and Registers of PIG**

| Symbol   | Name/Function                                                     | Type                   |
|----------|-------------------------------------------------------------------|------------------------|
| ROLL     | 1: Starts die rolling, 0: Stops die rolling                       | Control input          |
| HOLD     | 1: Ends player turn, 0: Continues player turn.                    | Control input          |
| NEW_GAME | 1: Starts new game, 0: Continues current game                     | Control input          |
| RESET    | 1: Resets game to INIT state, 0: No action                        | Control input          |
| DDIS     | 7-Bit LED die display array                                       | Data output vector     |
| SUB      | 14-Bit 7-segment pair (a, b, c, d, e, f, g) to Turn Total display | Data output vector     |
| TP1      | 14-Bit 7-segment pair (a, b, c, d, e, f, g) to Player 1 display   | Data output vector     |
| TP2      | 14-Bit 7-segment pair (a, b, c, d, e, f, g) to Player 2 display   | Data output vector     |
| P1       | 1: Player 1 LED on, 0: Player 1 LED off                           | Data output            |
| P2       | 1: Player 2 LED on, 0: Player 2 LED off                           | Data output            |
| DIE      | Die value—specialized counter to count 1,...,6,1,...              | 3-Bit data register    |
| SUR      | Subtotal for active player—parallel load register                 | 7-Bit data register    |
| TR1      | Total for Player 1—parallel load register                         | 7-Bit data register    |
| TR2      | Total for Player 2—parallel load register                         | 7-Bit data register    |
| FP       | First player—flip-flop 0: Player 1, 1: Player 2                   | 1-Bit control register |
| CP       | Current player—flip-flop 0: Player 1, 1: Player 2                 | 1-Bit control register |



**TABLE 6-18**  
**Datapath Output Actions and Control and Status Signals for PIG**

| Action or Status                                                                                    | Control or Status Signals | Meaning for Values 1 and 0                                                                |
|-----------------------------------------------------------------------------------------------------|---------------------------|-------------------------------------------------------------------------------------------|
| $TR1 \leftarrow 0$<br>$TR1 \leftarrow TR1 + SUR$                                                    | RST1<br>LDT1              | 1: Reset TR1 (synchronous reset), 0: No action<br>1: Add SUR to TR1, 0: No action         |
| $TR2 \leftarrow 0$<br>$TR2 \leftarrow TR2 + SUR$                                                    | RST2<br>LDT2              | 1: Reset TR2 (synchronous reset), 0: No action<br>1: Add SUR to TR2, 0: No action         |
| $SUR \leftarrow 0$<br>$SUR \leftarrow SUR + DIE$                                                    | RSSU<br>LDSU              | 1: Reset SUR (synchronous reset), 0: No action<br>1: Add DIE to SUR, 0: No action         |
| $DIE \leftarrow 000$<br>if ( $DIE = 110$ )<br>$DIE \leftarrow 001$<br>else $DIE \leftarrow DIE + 1$ | RESET<br>ENDI             | 1: Reset DIE to 000 (asynchronous reset)<br>1: Enable DIE to increment, 0: Hold DIE value |
| $P1 = BLINK$                                                                                        | BP1                       | 1: Connect P1 to BLINK, 0: Connect P1 to 1                                                |





## What are the inputs and outputs?



|                  |               |               |                                       |
|------------------|---------------|---------------|---------------------------------------|
| <b>std_logic</b> | <b>clock</b>  | <b>CP</b>     | <b>std_logic</b>                      |
| <b>std_logic</b> | <b>reset</b>  | <b>FP</b>     | <b>std_logic</b>                      |
| <b>std_logic</b> | <b>ENADIE</b> | <b>DIGIT0</b> | <b>std_logic_vector( 3 downto 0 )</b> |
| <b>std_logic</b> | <b>LDSU</b>   | <b>DIGIT1</b> | <b>std_logic_vector( 3 downto 0 )</b> |
| <b>std_logic</b> | <b>LDT1</b>   | <b>DIGIT2</b> | <b>std_logic_vector( 3 downto 0 )</b> |
| <b>std_logic</b> | <b>LDT2</b>   | <b>DIGIT3</b> | <b>std_logic_vector( 3 downto 0 )</b> |
| <b>std_logic</b> | <b>RSSU</b>   | <b>LEDDIE</b> | <b>std_logic_vector(2 downto 0)</b>   |
| <b>std_logic</b> | <b>RST1</b>   | <b>DIE1</b>   | <b>std_logic</b>                      |
| <b>std_logic</b> | <b>RST2</b>   | <b>WN</b>     | <b>std_logic</b>                      |



What are the inputs and outputs?



|                        |                      |                     |                        |
|------------------------|----------------------|---------------------|------------------------|
| <code>std_logic</code> | <code>clock</code>   | <code>ENADIE</code> | <code>std_logic</code> |
| <code>std_logic</code> | <code>reset</code>   | <code>LDSU</code>   | <code>std_logic</code> |
| <code>std_logic</code> | <code>ROLL</code>    | <code>LDT1</code>   | <code>std_logic</code> |
| <code>std_logic</code> | <code>HOLD</code>    | <code>LDT2</code>   | <code>std_logic</code> |
| <code>std_logic</code> | <code>NEWGAME</code> | <code>RSSU</code>   | <code>std_logic</code> |
| <code>std_logic</code> | <code>DIE1</code>    | <code>RST1</code>   | <code>std_logic</code> |
| <code>std_logic</code> | <code>LDPC</code>    | <code>RST2</code>   | <code>std_logic</code> |
|                        | <code>WN</code>      | <code>BP1</code>    | <code>std_logic</code> |
|                        |                      | <code>CP</code>     | <code>std_logic</code> |
|                        |                      | <code>FP</code>     | <code>std_logic</code> |



```

entity datapath is
  port(
    clock : in std_logic; --! Clock
    reset : in std_logic; --! Reset
    ENADIE : in std_logic; --! Enable Die to increment
    LDSU  : in std_logic; --! Add DIE to SUR register
    LDT1  : in std_logic; --! Add SUR to TR1 register
    LDT2  : in std_logic; --! Add SUR to TR2 register
    RSSU  : in std_logic; --! Reset SUR register
    RST1  : in std_logic; --! Reset TR1 register
    RST2  : in std_logic; --! Reset TR2 register
    CP    : inout std_logic; --! current player (register outside)
    FP    : inout std_logic; --! First player (register outside)
    DIGIT0 : out std_logic_vector( 3 downto 0 ); --! digit to the right
    DIGIT1 : out std_logic_vector( 3 downto 0 ); --! 2nd digit to the left
    DIGIT2 : out std_logic_vector( 3 downto 0 ); --! 3rd digit to the left
    DIGIT3 : out std_logic_vector( 3 downto 0 ); --! digit to the left
    LEDDIE : out std_logic_vector(2 downto 0); --! LEDs to display the die value
    DIE1   : out std_logic; --! signal that a one has been obtained
    WN    : out std_logic --! WIN has been achieved by a player
  );
end entity datapath;

```



# DATAPATH

```

Main_process : process(clock,reset) begin
    if reset = '1' then
        -- reset
        DIE <= (others => '0'); --! asynchronous reset
    else
        if rising_edge(clock) then
            if RST1 = '1' then
                TR1 <= (others => '0');
            end if;
            if RST2 = '1' then
                TR2 <= (others => '0');
            end if;
            if RSSU = '1' then
                SUR <= (others => '0');
            end if;
            if LDT1 = '1' then
                TR1 <= TR1 + SUR;
            end if;
            if LDT2 = '1' then
                TR2 <= TR2 + SUR;
            end if;
            if ENADIE = '1' then
                case DIE is
                    when "110" => DIE <= "001";
                    when others => DIE <= DIE +1;
                end case;
            end if;
        end if;

```



```

if DIE = "001" then
    DIE1 <= '1';
else
    DIE1 <= '0';
end if;
if LDSU = '1' then
    SUR <= SUR + (frontbits & DIE);
end if;
if CP ='1' then
    D <= TR2;
else
    D <= TR1;
end if;

if (D > "1100011") then
    WN <= '1';
else
    WN <= '0';
end if;

end if;
end if;
--! connection to displays
LEDDIE <= DIE;
DIGIT0 <= bcd1;
DIGIT1 <= bcd2;
DIGIT2 <= bcd3;
DIGIT3 <= bcd4;
end process;

```



Copyright ©2016 Pearson Education, All Rights Reserved

```

Main_process : process(clock,reset) begin
    if reset = '1' then
        -- reset
        DIE <= (others => '0'); --! asynchronous reset
    else
        if rising_edge(clock) then
            if RST1 = '1' then
                TR1 <= (others => '0');
            end if;
            if RST2 = '1' then
                TR2 <= (others => '0');
            end if;
            if RSSU = '1' then
                SUR <= (others => '0');
            end if;
            if LDT1 = '1' then
                TR1 <= TR1 + SUR;
            end if;
            if LDT2 = '1' then
                TR2 <= TR2 + SUR;
            end if;
            if ENADIE = '1' then
                case DIE is
                    when "110" => DIE <= "001";
                    when others => DIE <= DIE +1;
                end case;
            end if;
        end if;

```

```

if DIE = "001" then
    DIE1 <= '1';
else
    DIE1 <= '0';
end if;
if LDSU = '1' then
    SUR <= SUR + (frontbits & DIE);
end if;
if CP = '1' then
    D <= TR2;
else
    D <= TR1;
end if;

if (D > "1100011") then
    WN <= '1';
else
    WN <= '0';
end if;

end if;
end if;
--! connection to displays
LEDDIE <= DIE;
DIGIT0 <= bcd1;
DIGIT1 <= bcd2;
DIGIT2 <= bcd3;
DIGIT3 <= bcd4;
end process;

```



```

Main_process : process(clock,reset) begin
    if reset = '1' then
        -- reset
        DIE <= (others => '0'); --! asynchronous reset
    else
        if rising_edge(clock) then
            if RST1 = '1' then
                TR1 <= (others => '0');
            end if;
            if RST2 = '1' then
                TR2 <= (others => '0');
            end if;
            if RSSU = '1' then
                SUR <= (others => '0');
            end if;
            if LDT1 = '1' then
                TR1 <= TR1 + SUR;
            end if;
            if LDT2 = '1' then
                TR2 <= TR2 + SUR;
            end if;
            if ENADIE = '1' then
                case DIE is
                    when "110" => DIE <= "001";
                    when others => DIE <= DIE +1;
                end case;
            end if;
        end if;

```



```

if DIE = "001" then
    DIE1 <= '1';
else
    DIE1 <= '0';
end if;
if LDSU = '1' then
    SUR <= SUR + (frontbits & DIE);
end if;
if CP = '1' then
    D <= TR2;
else
    D <= TR1;
end if;

if (D > "1100011") then
    WN <= '1';
else
    WN <= '0';
end if;

end if;
end if;
--! connection to displays
LEDDIE <= DIE;
DIGIT0 <= bcd1;
DIGIT1 <= bcd2;
DIGIT2 <= bcd3;
DIGIT3 <= bcd4;
end process;

```



```

Main_process : process(clock,reset) begin
    if reset = '1' then
        -- reset
        DIE <= (others => '0'); --! asynchronous reset
    else
        if rising_edge(clock) then
            if RST1 = '1' then
                TR1 <= (others => '0');
            end if;
            if RST2 = '1' then
                TR2 <= (others => '0');
            end if;
            if RSSU = '1' then
                SUR <= (others => '0');
            end if;
            if LDT1 = '1' then
                TR1 <= TR1 + SUR;
            end if;
            if LDT2 = '1' then
                TR2 <= TR2 + SUR;
            end if;
            if ENADIE = '1' then
                case DIE is
                    when "110" => DIE <= "001";
                    when others => DIE <= DIE +1;
                end case;
            end if;
        end if;

```



```

if DIE = "001" then
    DIE1 <= '1';
else
    DIE1 <= '0';
end if;
if LDSU = '1' then
    SUR <= SUR + (frontbits & DIE);
end if;
if CP = '1' then
    D <= TR2;
else
    D <= TR1;
end if;

if (D > "1100011") then
    WN <= '1';
else
    WN <= '0';
end if;

end if;
end if;
--! connection to displays
LEDDIE <= DIE;
DIGIT0 <= bcd1;
DIGIT1 <= bcd2;
DIGIT2 <= bcd3;
DIGIT3 <= bcd4;
end process;

```



```

Main_process : process(clock,reset) begin
    if reset = '1' then
        -- reset
        DIE <= (others => '0'); --! asynchronous reset
    else
        if rising_edge(clock) then
            if RST1 = '1' then
                TR1 <= (others => '0');
            end if;
            if RST2 = '1' then
                TR2 <= (others => '0');
            end if;
            if RSSU = '1' then
                SUR <= (others => '0');
            end if;
            if LDT1 = '1' then
                TR1 <= TR1 + SUR;
            end if;
            if LDT2 = '1' then
                TR2 <= TR2 + SUR;
            end if;
            if ENADIE = '1' then
                case DIE is
                    when "110" => DIE <= "001";
                    when others => DIE <= DIE +1;
                end case;
            end if;
        end if;

```



```

if DIE = "001" then
    DIE1 <= '1';
else
    DIE1 <= '0';
end if;
if LDSU = '1' then
    SUR <= SUR + (frontbits & DIE);
end if;
if CP = '1' then
    D <= TR2;
else
    D <= TR1;
end if;

if (D > "1100011") then
    WN <= '1';
else
    WN <= '0';
end if;

end if;
end if;
--! connection to displays
LEDDIE <= DIE;
DIGIT0 <= bcd1;
DIGIT1 <= bcd2;
DIGIT2 <= bcd3;
DIGIT3 <= bcd4;
end process;

```



Copyright ©2016 Pearson Education, All Rights Reserved

```

Main_process : process(clock,reset) begin
  if reset = '1' then
    -- reset
    DIE <= (others => '0'); --! asynchronous reset
  else
    if rising_edge(clock) then
      if RST1 = '1' then
        TR1 <= (others => '0');
      end if;
      if RST2 = '1' then
        TR2 <= (others => '0');
      end if;
      if RSSU = '1' then
        SUR <= (others => '0');
      end if;
      if LDT1 = '1' then
        TR1 <= TR1 + SUR;
      end if;
      if LDT2 = '1' then
        TR2 <= TR2 + SUR;
      end if;
      if ENADIE = '1' then
        case DIE is
          when "110" => DIE <= "001";
          when others => DIE <= DIE +1;
        end case;
      end if;
    end if;

```



```

if DIE = "001" then
  DIE1 <= '1';
else
  DIE1 <= '0';
end if;
if LDSU = '1' then
  SUR <= SUR + (frontbits & DIE);
end if;
if CP ='1' then
  D <= TR2;
else
  D <= TR1;
end if;
if (D > "1100011") then
  WN <= '1';
else
  WN <= '0';
end if;
end if;
--! connection to displays
LEDDIE <= DIE;
DIGIT0 <= bcd1;
DIGIT1 <= bcd2;
DIGIT2 <= bcd3;
DIGIT3 <= bcd4;
end process;

```



Copyright ©2016 Pearson Education, All Rights Reserved

```

Main_process : process(clock,reset) begin
  if reset = '1' then
    -- reset
    DIE <= (others => '0'); --! asynchronous reset
  else
    if rising_edge(clock) then
      if RST1 = '1' then
        TR1 <= (others => '0');
      end if;
      if RST2 = '1' then
        TR2 <= (others => '0');
      end if;
      if RSSU = '1' then
        SUR <= (others => '0');
      end if;
      if LDT1 = '1' then
        TR1 <= TR1 + SUR;
      end if;
      if LDT2 = '1' then
        TR2 <= TR2 + SUR;
      end if;
      if ENADIE = '1' then
        case DIE is
          when "110" => DIE <= "001";
          when others => DIE <= DIE +1;
        end case;
      end if;
    end if;

```



```

if DIE = "001" then
  DIE1 <= '1';
else
  DIE1 <= '0';
end if;
if LDSU = '1' then
  SUR <= SUR + (frontbits & DIE);
end if;
if CP ='1' then
  D <= TR2;
else
  D <= TR1;
end if;
if (D > "1100011") then
  WN <= '1';
else
  WN <= '0';
end if;
end if;
--! connection to displays
LEDDIE <= DIE;
DIGIT0 <= bcd1;
DIGIT1 <= bcd2;
DIGIT2 <= bcd3;
DIGIT3 <= bcd4;
end process;

```



Copyright ©2016 Pearson Education, All Rights Reserved

```

entity controlunit is
  port(
    clock : in std_logic; --! Clock
    reset : in std_logic; --! Reset
    ROLL : in std_logic; --! button for the roll
    HOLD : in std_logic; --! button for hold
    NEWGAME: in std_logic; --! button for new game
    ENADIE : out std_logic; --! Enable Die to increment
    LDSU : out std_logic; --! Add DIE to SUR register
    LDT1 : out std_logic; --! Add SUR to TR1 register
    LDT2 : out std_logic; --! Add SUR to TR2 register
    RSSU : out std_logic; --! Reset SUR register
    RST1 : out std_logic; --! Reset TR1 register
    RST2 : out std_logic; --! Reset TR2 register
    BP1 : out std_logic; --! enables blinking
    CP : inout std_logic; --! current player (register outside)
    FP : inout std_logic; --! First player (register outside)
    DIE1 : in std_logic; --! signal that the die is at one
    WN : in std_logic --! WIN has been achieved by a player
  );
end entity controlunit;

```

## CONTROL UNIT



```

architecture rtl of controlunit is
  --type definition
  type PIG_STATE is (INITIAL, BEGINNING, ROLLING,
ONE, ROLLHOLD, TEST, WIN);
  --Signal definition
  signal PIG_CURRENT_STATE : PIG_STATE;
  begin
FSM : process(clock)
  begin
if rising_edge(clock) then
  if reset = '1' then --chosen this one but could
be anything to reset
    -- reset
    PIG_CURRENT_STATE <= INITIAL;
    FP <= '0';
  else
    --default values
    ENADIE <= '0';
    LDSU <= '0';
    LDT1 <= '0';
    LDT2 <= '0';
    RSSU <= '0';
    RST1 <= '0';
    RST2 <= '0';
    BP1 <= '0';
  end if;
  case PIG_CURRENT_STATE is
    when INITIAL =>
      PIG_CURRENT_STATE <= BEGINNING;
      RST1 <= '1'; --! reset TR1
      RST2 <= '1'; --! reset TR2
      CP <= FP; --! behavioural implmentation
    when BEGINNING =>
      RSSU <= '1';
      if ROLL = '1' then --! Press the button to roll
        PIG_CURRENT_STATE <= ROLLING;
      end if;
    when ROLLING =>
      if ROLL = '1' then
        ENADIE <= '1'; --! enables die increment
      else
        PIG_CURRENT_STATE <= ONE;
      end if;
    when ONE =>
      if DIE1 ='1' then
        CP <= not CP;
        PIG_CURRENT_STATE <= BEGINNING;
      else
        PIG_CURRENT_STATE <= ROLLHOLD;
        LDSU <= '1';
      end if;
    end case;
  end if;
end process;
end architecture;

```

```
when ROLLHOLD =>
  if HOLD = '1' then
    PIG_CURRENT_STATE <= TEST;
    CP <= not CP;
    if CP = '1' then
      LDT1 <= '1';
    else
      LDT2 <= '1';
    end if;
  else
    if ROLL = '1' then
      PIG_CURRENT_STATE <= ROLLING;
    else
      PIG_CURRENT_STATE <= ROLLHOLD;
    end if;
  end if;
```

```
when TEST =>
  if WN = '1' then
    PIG_CURRENT_STATE <= WIN;
  else
    PIG_CURRENT_STATE <= BEGINNING;
  end if;
when WIN =>
  BP1 <= '1';
  if NEWGAME = '1' then
    FP <= not FP;
    PIG_CURRENT_STATE <= INITIAL;
  else
    PIG_CURRENT_STATE <= WIN;
  end if;
end case;
end if;
end process;
end architecture ;
```

# Main



What's inside?