



# Programcounter

Labor Digital Design

## Contents

|                                                                          |    |
|--------------------------------------------------------------------------|----|
| 1 Goals .....                                                            | 1  |
| 2 Control ROM of the ALU .....                                           | 2  |
| 2.1 Circuit .....                                                        | 2  |
| 2.2 Sequencing of Operations .....                                       | 2  |
| 3 Software implementation of a serial port .....                         | 3  |
| 3.1 Linear algorithm .....                                               | 3  |
| 3.2 Algorithm with loops .....                                           | 5  |
| 3.3 Comparison .....                                                     | 6  |
| 4 Blocks, Components, and <b>For</b> Loops .....                         | 7  |
| 4.1 Creating a Block .....                                               | 7  |
| 4.2 Convert a Block to a Component ( <b>blue</b> to <b>green</b> ) ..... | 9  |
| 4.3 Update the component interface .....                                 | 9  |
| 4.4 <b>For</b> Generate .....                                            | 10 |
| 5 Checkout .....                                                         | 12 |
| Glossary .....                                                           | 13 |

## 1 Goals

This lab presents the development of a program code using a **Read-Only Memory (ROM)** and exercises the creation of a program counter **Program Counter (PC)**.

It also exercises drawing hierarchical circuits.



## 2 | Control ROM of the ALU

### 2.1 Circuit

The Figure 1 shows a simplified view of a processor, with an **Arithmetic and Logical Unit (ALU)**, registers, and a program counter.



Figure 1 - Control **ROM** of the **ALU** and registers

The address of the **ROM** is generated by a program counter, which allows reading the stored code sequentially. The data of the **ROM** contains **Machine Instructions (MIs)** which consist of control signals for the **ALU** and registers.

The input **Output Enable (OE)** controls the high-impedance output of the **ROM**. The input **Chip Select (CS)** is the select signal for the **ROM**. Both must be active for the component to provide its data at the output.

### 2.2 Sequencing of Operations

Instructions are executed in 2 phases, as shown in Figure 2.



Figure 2 - Phases of instructions

The output of the **ALU** is stable at the end of phase 0. The new value of the selected register is latched on the rising edge of the clock at the end of phase 1.



# 3 | Software implementation of a serial port

The Figure 3 shows the timing of the serial transmission of a data word.



Figure 3 - Serial transmission

The serial data is transmitted on the **Least Significant Bit (LSB)** of the processor data bus.

## 3.1 Linear algorithm

The Assembler program code stored in the **ROM** is shown in Listing 1.

```

1 LOAD    s3, FF          ; load stop bit
2 OUTPUT s3              ; output stop bit
3 LOAD    s3, s3          ; no operation
4 LOAD    s3, s3          ; no operation
5 LOAD    s3, s3          ; no operation
6 LOAD    s3, s3          ; no operation
7 LOAD    s0, 00          ; load start bit
8 OUTPUT s0              ; output start bit
9 INPUT   s1              ; load word to send
10 OUTPUT s1              ; output word, LSB is considered
11 SR0    s1              ; shift word, bit 1 -> LSB
12 OUTPUT s1              ; output bit 1
13 SR0    s1              ; bit 2 -> LSB
14 OUTPUT s1              ; output bit 2
15 SR0    s1              ; bit 3 -> LSB
16 OUTPUT s1              ; output bit 3
17 SR0    s1              ; bit 4 -> LSB
18 OUTPUT s1              ; output bit 4
19 SR0    s1              ; bit 5 -> LSB
20 OUTPUT s1              ; output bit 5
21 SR0    s1              ; bit 6 -> LSB
22 OUTPUT s1              ; output bit 6
23 SR0    s1              ; bit 7 -> LSB
24 OUTPUT s1              ; output bit 7
25 LOAD    s3, s3          ; no operation
26 OUTPUT s3              ; output stop bit

```

Listing 1 - Linear algorithm



In the linear algorithm one instruction after the other is read. There are no loops or jump within the program. Our first program counter only needs to be able to increment the address by one.



### 3.1.1 Implementation

An iterative counter must be created. Start by creating the basic component of the iterative counter **CNT/cnt\_1bit**. This counter must increment on the rising edge of the clock, when **incPC** = '1'. Note that in our system, **incPC** is activated every second clock period. Ignore the signals **loadInstrAddr** and **instrAddress** to load a new value into the counter.



- Establish the list of inputs/outputs of the 1-bit counter.
- Create the 1-bit counter block **CNT/cnt\_1bit** (See Section 4.1).
- Implement the circuit of a 1-bit counter inside the created block.
- Convert this block into a component (See Section 4.2).



D Flip-Flops are available in the “Sequential” library under the name **DFF**

Once the **CNT/cnt\_1bit** component is created, use this component to create the 8-bit counter.



- Copy-paste 8 times the **CNT/cnt\_1bit** component.
- Connect the 1-bit counters together to create the 8-bit counter.

### 3.1.2 Simulation

Simulate the system and verify the correct operation of the counter and processor.



Verify the correct operation of the counter using the testbench **CNT\_test/nanoProcess\_tb** with the simulation file **\$SIMULATION\_DIR/CNT1.do**.



## 3.2 Algorithm with loops

The following algorithm allows a more compact writing of the program but it uses loops or respectively jumps in the program:

```

1 LOAD  s3, FF          ; load stop bit
2 OUTPUT s3              ; output stop bit
3 LOAD  s2, 04          ; initialize loop counter 3
4 SUB   s2, 01          ; decrement loop counter 4
5 JUMP  NZ 03          ; loop back if not end of count 5
6 LOAD  s0, 00          ; load start bit 6
7 OUTPUT s0              ; output start bit 7
8 LOAD  s2, 08          ; initialize loop counter 8
9 INPUT s1              ; load word to send 9
10 LOAD  s3, s3         ; no operation
11 OUTPUT s1             ; output word, LSB is considered
12 SR0   s1              ; next bit -> LSB
13 SUB   s2, 01          ; decrement loop counter
14 JUMP  NZ 0A          ; loop back if not end of count
15 OUTPUT s3              ; output bit 1

```

Listing 2 - Algorithm with loops

### 3.2.1 Implementation

To implement the algorithm with loops, the **PC** must allow loading a new value. The two signals **instrAddr** and **loadInstrAddr** are used to load a new value.

The component **CNT/cnt\_1bit** must be modified to allow loading a new value into the counter.



- Modify the component **CNT/cnt\_1bit** to allow loading a new value.
- As the inputs of the component have been changed, it is necessary to update the component interface (See Section 4.3).

In the first part of the laboratory, the program counter was created by copy-pasting the component **CNT/cnt\_1bit** 8 times. To simplify the implementation, use a **FOR Generate** loop to create the 8-bit counter.



- Use a **FOR Generate** loop around the component **CNT/cnt\_1bit**, to automatically copy-paste it 8 times (See Section 4.4).
- Use the generic parameter **programCounterBitNb** to define the number of bits of the counter and thus create a generic counter.



### 3.2.2 Simulation

Simulate the system and verify the correct operation of the counter and processor. Change the default view of the ROM to select the version with code including loops names **withLoops**.

1. Right-click on the component,
2. Select **Open As**,
3. Click on **withLoops**



 Verify the correct operation of the counter using the test bench available **CNT\_test/nanoProcess\_tb** with the simulation file **\$SIMULATION\_DIR/CNT2.do**.

### 3.3 Comparison

Compare the two algorithms in terms of transmission speed (baudrate) and code size.



- Compare the transmission speed (baudrate) of both algorithms.
- Compare the code size of both algorithms in Listing 1 and Listing 2.



# 4 | Blocks, Components, and For Loops

The following sections will guide you through creating the counter using **FOR Generate** loop and creating a reusable component.

## 4.1 Creating a Block

To create a new block, you need to add an element by clicking on the **Add Block** button (see Figure 4).



Figure 4 - Add a Block

After clicking the button, a new block can be added by clicking on the schematic. Once the block is added, the necessary **Input/Outputs (I/Os)** can be connected:



Figure 5 - New block wired



A **blue block** cannot be copied and pasted as it exists only once. Only the **green block** (components) can be copied and pasted.

Once the **I/Os** are wired, the type of block content must be selected (**Block Diagram** /**State Diagram** /**VHDL file** ).



The following window opens, allowing you to choose the type of block to create. Select the desired block type and click **Finish**.



| Link                                | Block type    | Icon |
|-------------------------------------|---------------|------|
| <b>Graphical View/Block Diagram</b> | block diagram |      |
| <b>Graphical View/State Diagram</b> | state machine |      |
| <b>VHDL File/Architecture</b>       | VHDL code     |      |

#### 4.1.1 Block diagram

- Select **Graphical View/Block Diagram**
- Click on **Next**
- Enter the name of the block under **Design unit**
- Fill in the **I/O** table
  - Ensure that the types are correct
  - Define the limits for multi-bit types
- Click on “Finish”



The **I/Os** can still be added, removed, and modified when editing the schematic.



## 4.2 Convert a Block to a Component (blue to green)

Converting the block to a component (from **blue** to **green**) allows copying and pasting the block and makes it available in the project library.

**Blue** blocks are ideal for quickly creating a block interface, while **green** components are essential for reusing the elements elsewhere in the project.



1. Right-click on the **blue block**
2. Click on **Convert Block to component**



## 4.3 Update the component interface

The interface of a component consists of several parts: **Inputs** and **Outputs**, **Generics**, and **Symbol**. These different parameters can be added, removed, and modified.

**Once the component interface has been modified**

1. Right-click on the component  
2. Select **Update**  
3. Click on **Interface and Graphics**

1. Right-click on the component  
2. Select **Update**  
3. Click on **From Symbol**



## 4.4 For Generate

The **FOR Generate** frame allows you to create multiple iterations of the same structure. To add a new frame, click on the **add frame** button and select **FOR**.



When you add a **FOR** frame, the following element is drawn on the schematic.



All signals and components inside the **FOR** frame will be copied **n+1** times. The index **i** goes from **0** to **n**. *Change n to a constant or generic.*

All component outputs must use the index **i** in their **Slice/Index**. Otherwise, short-circuits will occur between the outputs.

Component inputs that are common to all instances (like a clock or reset) should not use the index **i**. However, inputs that vary between instances (like a data signal) should use the index **i**.



For example: the **clock** should be the same everywhere; but a counter should output all bits. **Bit[0]**, then **bit[1]**, then **bit[2]** ... **bit[i]**.



- Only signals and components (green) elements can be placed in a **FOR** loop
- No **In Port** or **Out Port** should reside inside the frame.
- No blocks (blue) can be placed inside.



#### 4.4.1 Example

The example demonstrate an iterative counter circuit. The generic **counter\_nBits** allows the selection of the number of bits for the counter.



| Part                                                                            | Description                                                                                                                                                                 |
|---------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <b>0 TO (counter_nBits-1)</b>                                                   | Repeat the loop <b>counter_nBits</b> times                                                                                                                                  |
| <b>counter(i) : unsigned...</b>                                                 | Write the counter output.<br>The <b>1st block</b> writes the <b>bit0</b> of the counter signal.<br>The <b>2nd block</b> writes the <b>bit1</b> of the counter signal, etc.. |
| <b>cin(i) : std_ulogic_vector ...</b><br><b>cin(i+1) : std_ulogic_vector...</b> | Transmit information to the next block.<br>The <b>1st block cout</b> is connected to the <b>2nd block cin</b> , etc..                                                       |



As we have **cin(i+1)**, when **i = (counter\_nBits - 1)  $\Rightarrow$  cin(counter\_nBits)**, therefore the size of the **cin** bus must be **(counter\_nBits DOWNTO 0)**.



Do not forget to connect **cin(0)**, OUTSIDE of the frame.



## 5 | Checkout

This is the end of the lab, you have successfully implemented a program counter that supports all instructions of the  $\mu$ Processor. Before leaving the lab, make sure you have completed the following tasks:

- Circuit Design
  - The 1-bit counter block **CNT/cnt\_1bit** has been created.
  - The **FOR** frame has been used to duplicate the 1-bit counter block.
  - The **n**-bit counter has been created.
  - The counter has been adjusted to allow loading a new value.
- Simulations
  - Both counters have been successfully tested with the testbench **CNT\_test/nanoProcess\_tb** and simulation files **\$SIMULATION\_DIR/CNT1.do** and **\$SIMULATION\_DIR/CNT2.do**.
- Comparison
  - The baud rate of both algorithms has been compared.
  - The code size of both algorithms has been compared.
- Documentation and Project Files
  - Ensure that all steps (design, simulations, comparison) are well documented in your lab report.
  - Save the project on a USB stick or in the shared network folder (**\filer01.hevs.ch**).
  - Share the files with your lab partner to ensure continuity of work.



# Glossary

**ALU – Arithmetic and Logical Unit** [2](#), [2](#), [2](#), [2](#)

**CS – Chip Select** [2](#)

**I/O – Input/Output** [7](#), [7](#), [8](#), [8](#)

**LSB – Least Significant Bit** [3](#)

**MI – Machine Instruction:** A machine instruction is a binary-coded operation that a processor can execute directly. It typically consists of an opcode, operands and immediates. [2](#)

**OE – Output Enable** [2](#)

**PC – Program Counter** [1](#), [5](#)

**ROM – Read-Only Memory** [1](#), [2](#), [2](#), [2](#), [2](#), [2](#), [3](#), [6](#)