



# **Programming an Embedded MicroBlaze Processor (XD131)**

## **Programming an Embedded MicroBlaze Processor (XD131)**

Programming an Embedded MicroBlaze Processor

Programming an Embedded MicroBlaze V Processor

# Programming an Embedded MicroBlaze Processor (XD131)

## Important

Released with AMD Vitis™ Unified Software Platform and AMD Vivado™ Design Suite 2025.1 without changes from 2024.1.

Create an AMD MicroBlaze™ system for a Spartan-7 FPGA using Vivado IP integrator.

- Programming an Embedded MicroBlaze Processor
  - Introduction
  - Step 1: Start the Vivado IDE and Create a Project
  - Step 2: Create an IP Integrator Design
  - Step 3: Memory-Mapping the Peripherals in IP Integrator
  - Step 4: Validate Block Design
  - Step 5: Generate Output Products
  - Step 6: Create a Top-Level Wrapper
  - Step 7: Take the Design through Implementation
  - Step 8: Export the Design to the Vitis software platform
  - Step 9: Create a “Platform Component”
  - Step 10: Create a “Peripheral Test” Application
  - Step 11: Execute the Software Application on a SP701 Board
  - Step 12: Connect to Vivado Logic Analyzer
  - Step 13: Set the MicroBlaze to Logic Cross Trigger
  - Step 14: Set the Logic to Processor Cross- Trigger
  - Conclusion
  - Lab Files
- Programming an Embedded MicroBlaze V Processor
  - Introduction
  - Step 1: Start the Vivado IDE and Create a Project
  - Step 2: Create an IP Integrator Design
  - Step 3: Memory-Mapping the Peripherals in IP Integrator
  - Step 4: Validate Block Design
  - Step 5: Generate Output Products
  - Step 6: Create a Top-Level Wrapper
  - Step 7: Take the Design through Implementation
  - Step 8: Export the Design to the Vitis software platform
  - Step 9: Create a “Platform Component”
  - Step 10: Create a “Peripheral Test” Application
  - Step 11: Execute the Software Application on a SP701 Board
  - Step 12: Connect to Vivado Logic Analyzer
  - Step 13: Set the Logic to Processor Cross- Trigger
  - Conclusion
  - Lab Files

# Programming an Embedded MicroBlaze Processor

In this tutorial, you will create a simple AMD soft-processor system for a Spartan-7 FPGA using AMD Vivado™ IP integrator.

This tutorial uses the classic AMD MicroBlaze™ soft-processor.

## Introduction

The MicroBlaze system includes native AMD IP including:

- MicroBlaze processor
- AXI block RAM
- Double Data Rate 3 (DDR3) memory
- UARTLite
- AXI GPIO
- MicroBlaze Debug Module (MDM)
- Proc Sys Reset
- Local memory bus (LMB)

Parts of the block design are constructed using the Platform Board Flow feature.

This lab also shows the cross-trigger capability of the MicroBlaze processor.

The feature is demonstrated using a software application code developed in the Vitis software platform in a stand-alone application mode.

This lab targets the AMD SP701 FPGA Evaluation Kit.

## Step 1: Start the Vivado IDE and Create a Project

1. Start the Vivado IDE by clicking the Vivado desktop icon or by typing `vivado` at a command prompt.
2. From the Quick Start page, select **Create Project**.
3. In the New Project dialog box, use the following settings:
  - a. In the Project Name dialog box, type the project name and location.
  - b. Make sure that the **Create project subdirectory** check box is selected.  
Click **Next**.
  - c. In the Project Type dialog box, select **RTL project**. Ensure that the **Do not specify sources at this time** check box is cleared. Click **Next**.
  - d. In the Add Sources dialog box, set the Target language to either **VHDL** or **Verilog**. You can leave the Simulator language selection to **Mixed**.

- e. Click **Next**.
- f. In Add Constraints dialog box, click **Next**.
- g. In the Default Part dialog box, select **Boards** and choose **Spartan-7 SP701 Evaluation Platform**. Click **Next**.
- h. Review the project summary in the New Project Summary dialog box and click **Finish** to create the project.

Because you selected the SP701 board when you created the Vivado IDE project, you see the following message in the Tcl Console:

```
set_property board_part xilinx.com:sp701:part0:1.1 [current_project]
```

Although Tcl commands are available for many of the actions performed in the Vivado IDE, they are not explained in this tutorial. Instead, a Tcl script is provided that can be used to recreate this entire project. See the Tcl Console for more information. You can also refer to the *Vivado Design Suite Tcl Command Reference Guide* (UG835) for information about the `write_bd_tcl` commands.

## Step 2: Create an IP Integrator Design

1. From Flow Navigator, under IP integrator, select **Create Block Design**.
2. Specify the IP subsystem design name. For this step, you can use `mb_subsystem` as the Design name. Leave the Directory field set to its default value of <Local to Project>. Leave the Specify source set drop-down list set to its default value of Design Sources.
3. Click **OK** in the Create Block Design dialog box, shown in the following figure.



4. In the IP integrator diagram area, right-click and select **Add IP**. The IP integrator Catalog opens. Alternatively, you can also select the Add IP icon in the middle of the canvas.

This design is empty. Press the button to add IP.

5. Type **mig** in the Search field to find the MIG core, then select **Memory Interface Generator (MIG 7 Series)**, and press **Enter**.



The Designer Assistance link becomes active in the block design banner.

6. Click **Run Block Automation**.



The Run Block Automation dialog box opens.

## Programming an Embedded MicroBlaze Processor (XD131)



7. Click **OK**. This instantiates the MIG core and connects the I/O interfaces to the I/O interfaces for the DDR memory on the SP701 board.



8. Right-click anywhere in the block design canvas, and select **Add IP**. The IP catalog opens.
  9. In the Search field, type **micr** to find the MicroBlaze IP, then select **MicroBlaze**, and press **Enter**.
- Note:** If not displayed by default, the IP Details window can be displayed by clicking **CTRL+Q** on the keyboard while searching for IP.



## Use the Board Window to Connect to Board Interfaces

There are several ways to use an existing interface in IP integrator. Use the Board window to instantiate some of the interfaces that are present on the SP701 board.

1. Navigate to the **Windows>Board**.
2. Select the **Board** window to see the interfaces present on the SP701 board.



In the Board window, notice that the DDR3 SDRAM interface is connected as shown by the circle in the following figure. This is because you used the Block Automation feature in the previous steps to connect the MIG core to the board interfaces for DDR3 SDRAM memory.



3. From the Board window, select **UART** under the Miscellaneous folder, and drag and drop it into the block design canvas.  
This instantiates the AXI Uartlite IP on the block design.
4. From the Board window, select **LED** under the General Purpose Input or Output folder, and drag and drop it into the block design canvas.  
This instantiates the GPIO IP on the block design and connects it to the on-board LEDs.
5. Next, from the Board window, select **FPGA Reset** under the Reset folder, and drag and drop it into the block design canvas.  
This connects the CPU push button reset to the MIG core IP.

The block design now should look like the following figure.



## Add Peripheral: AXI block RAM Controller

1. Add the AXI block RAM Controller, shown in the following figure, by right-clicking the IP integrator canvas and selecting **Add IP**.

# Programming an Embedded MicroBlaze Processor (XD131)



The block design now should look like the following figure.



Run Block Automation

1. Click **Run Block Automation**, as shown below.



The **Run Block Automation** dialog box opens.

2. On the **Run Block Automation** dialog box:
  - Leave Preset as the default value, **None**.
  - Set Local Memory to **64 KB**.
  - Leave the Local Memory ECC as the default value, **None**.
  - Set Cache Configuration to **32 KB**.
  - Set Debug Module to **Extended Debug**.
  - Leave the Peripheral AXI Port option as the default value, **Enabled**.
  - Leave the Interrupt Controller option unchecked.
  - Leave The Clock source option set to **/mig\_7series\_0/ui\_addn\_clk\_0 (100 MHz)**.



3. Click **OK**.

This generates a basic MicroBlaze system in the IP integrator diagram area, as shown in the following figure.

# Programming an Embedded MicroBlaze Processor (XD131)



## 4. Click Run Block Automation again.

The **Run Block Automation** dialog box opens.

## 5. On the **Run Block Automation** dialog box:

### a. Set Keep Classic MicroBlaze



With the introduction of the MicroBlaze V, AMD provide the ability to automate the conversion of a classic MicroBlaze processors to the RISC-V MicroBlaze V processor. In this tutorial, the classic MicroBlaze is kept.

## Use Connection Automation

Run Connection Automation provides several options that you can select to make connections. This section will walk you through the first connection, and then you

will use the same procedure to make the rest of the required connections for this tutorial.

1. Click **Run Connection Automation** as shown in the following figure.



The Run Connection Automation dialog box opens.

2. Check the All Automation check box in the left pane of the dialog box as shown in the following figure. This selects interfaces to run Connection Automation for.



3. Use the following table to set options in the Run Connection Automation dialog box.

**Table 1: Run Connection Automation Options**

| <b>Connection</b>                                                                 | <b>More Information</b>                                                                                                                                                                                                                                                                                                                                                                                        | <b>Setting</b>                                                                                                                                                                       |
|-----------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| axi_bram_ctrl_0<br><ul style="list-style-type: none"> <li>• BRAM_PORTA</li> </ul> | <p>The only option for this automation is to instantiate a new Block Memory Generator as shown under options.</p>                                                                                                                                                                                                                                                                                              | Leave the Blk_Mem_Gen to its default option of Auto.                                                                                                                                 |
| axi_bram_ctrl_0<br><ul style="list-style-type: none"> <li>• BRAM_PORTB</li> </ul> | <p>The Run Connection Automation dialog box opens and gives you two choices:</p> <ul style="list-style-type: none"> <li>• Instantiate a new BMG and connect the PORTB of the AXI block RAM Controller to the new BMG IP</li> <li>• Use the previously instantiated BMG core and automatically configure it to be a true dual- ported memory and connected to PORTB of the AXI block RAM Controller.</li> </ul> | Leave the Blk_Mem_Gen option to its default value of Auto.                                                                                                                           |
| axi_bram_ctrl_0<br><ul style="list-style-type: none"> <li>• S_AXI</li> </ul>      | <p>Two options are presented in this case. The Master field can be set for either cached or non-cached accesses.</p>                                                                                                                                                                                                                                                                                           | The Run Connection Automation dialog box offers to connect this to the /microblaze_0 (Cached). Leave it to its default value. In case, cached accesses are not desired this could be |

| Connection                | More Information                                                                                                                                               | Setting                                                                                                                      |
|---------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------|
|                           |                                                                                                                                                                | changed to /microblaze_0 (Periph). Leave the Clock Connection (for unconnected clks) field set to its default value of Auto. |
| axi_gpio_0<br>• S_AXI     | The Master field is set to its default value of /microblaze_0 (Periph). The Clock Connection (for unconnected clks) field is set to its default value of Auto. | Keep these default settings.                                                                                                 |
| axi_uartlite_0<br>• S_AXI | The Master field is set to its default value of /microblaze_0 (Periph). The Clock Connection (for unconnected clks) field is set to its default value of Auto. | Keep these default settings.                                                                                                 |
| mdm_1<br>• TRIG_IN_0      | This will be connected to a new System ILA core's TRIG_OUT pin.                                                                                                | Leave the ILA Connection settings to its default value of Auto.                                                              |
| mdm_1<br>• TRIG_OUT_0     | This will be connected to the System ILA core's TRIG_IN pin.                                                                                                   | Leave the ILA Connections settings to its default value of Auto.                                                             |
| mig_7series_0<br>• S_AXI  | The Master field is set to microblaze_0 (Cached). Leave it to this value so the accesses to the DDR3 memory are cached accesses.                               | Keep these default settings.                                                                                                 |

| Connection                                                                                  | More Information                                                                       | Setting                   |
|---------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------|---------------------------|
|                                                                                             | The Clock Connection (for unconnected clks) field is set to its default value of Auto. |                           |
| Rst_mig_7_series_0_100M<br><ul style="list-style-type: none"> <li>• ext_reset_in</li> </ul> | The reset pin of the Processor Sys Reset IP will be connected to the board reset pin.  | Keep the default setting. |

1. After setting the appropriate options, as shown in the table above, click **OK**.

At this point, your IP integrator diagram area should look like the following figure.



**Note:** The relative placement of your IP might be slightly different.

# Mark Nets for Debugging

1. To monitor the AXI transactions taking place between the MicroBlaze and the GPIO, select the interface net connecting M00\_AXI interface pin of the microblaze\_0\_axi\_periph instance and the S\_AXI interface pin of the axi\_gpio\_0 instance.
  2. Right-click and select **Debug** from the context menu.  
**Note:** The Designer Assistance is available as indicated by the Run Connection Automation link in the banner of the block design.
  3. Click **Run Connection Automation**.

4. In the Run Connection Automation dialog box, go with the default setting as shown in the following figure.



5. Click **OK**.

6. Click the Regenerate Layout button  in the IP integrator toolbar to generate an optimum layout for the block design. The block diagram should look like the following figure.



**Note:** The relative placement of your IP might be slightly different.

Add Connection Between MDM and AXI SmartConnect

In order to enable JTAG-based debugging of the AXI BRAM Controller and the DDR3 RAM, a connection between the MicroBlaze Debug Module (MDM) and AXI SmartConnect must be made.

1. Click **Run Connection Automation**.
2. In the Run Connection Automation dialog box set the Slave interface option to either **/axi\_bram\_ctrl\_0/S\_AXI** or **/mig\_7series\_0/S\_AXI**.



Either option will connect to the same AXI SmartConnect instance allowing for JTAG memory access. 3. Click the Regenerate Layout button  in the IP integrator toolbar to generate an optimum layout for the block design. The block diagram should look like the following figure.



**Note:** The relative placement of your IP might be slightly different.

This connection connects the AXI4 master port of the MicroBlaze Debug Module (MDM) to the AXI SmartConnect for direct access to memory from JTAG. This allows fast program download, as well as transparent memory access when the connected MicroBlaze processors are executing.

## Step 3: Memory-Mapping the Peripherals in IP Integrator

1. Click the **Address Editor** window.
2. In the Address Editor, do the following:
  - a. Expand the **microblaze\_0** instance by clicking on the **Expand All** icon  in the toolbar to the top of the Address Editor window.
  - b. Ensure the range of **microblaze\_0/mig\_7\_series\_0/memmap** IP in both the Data and the Instruction section are **512 MB**, and **mig\_7series\_0/memmap** also is **512 MB**, as shown in the following figure.



- c. The top of the Address Editor window should show Assigned (11), indicating all 11 interfaces were assigned addresses. If Unassigned shows any interfaces unassigned, click on the **Assign All** arrow .

You must also ensure that the memory in which you are going to run and store your software is within the cacheable address range. This occurs when you enable Instruction Cache and Data Cache, while running the Block Automation for the MicroBlaze processor.

To use either Memory IP DDR or AXI block RAM, those IP must be in the cacheable area; otherwise, the MicroBlaze processor cannot read from or write to them. Validating the design will automatically re-configure the MicroBlaze processor's cacheable address range.

## Step 4: Validate Block Design

To run design rule checks on the design:

1. Click the Validate Design button on the toolbar, or select **Tools > Validate Design**.

The Validate Design dialog box informs you that there are no critical warnings or errors in the design.

2. Click **OK**.
3. Save your design by pressing **Ctrl+S**, or select **File > Save Block Design**.

## Step 5: Generate Output Products

1. In the Sources window, select the block design, then right-click it and select **Generate Output Products**. Alternatively, you can click **Generate Block**

**Design in the Flow Navigator.**

The Generate Output Products dialog box opens.

2. Click **Generate**.



The Generate Output Products dialog box informs you that Out-of-context module runs were launched.

3. Click **OK**.
4. Wait a few minutes for all the Out-of-Context module runs to finish as shown in the Design Runs windows.

| Name                       | Constraints  | Status                  | WNS | TNS | WHS | THS | WBSS | TPWS | Total Power | Failed Routes | Methodology | RQA Score | QoR Suggestions | LUT | FF   | BRAM | URAM | DSP |
|----------------------------|--------------|-------------------------|-----|-----|-----|-----|------|------|-------------|---------------|-------------|-----------|-----------------|-----|------|------|------|-----|
| synth_1 (active)           | constrs_1    | Not started             |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| impl_1                     | constrs_1    | Not started             |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| Out-of-Context Module Runs |              |                         |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| mb_subsystem               |              | Submodule Runs Complete |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 6462 | 578  | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 3421 | 309  | 22   | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 48   | 64   | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 108  | 114  | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 283  | 266  | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 0    | 1    | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 0    | 1    | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 83   | 74   | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 4    | 2    | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 10   | 12   | 2    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 443  | 473  | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 19   | 40   | 0    | 0   |
| mb_subsystem               | mb_subsystem | synth_design Complete!  |     |     |     |     |      |      |             |               |             |           |                 |     | 10   | 12   | 2    | 0   |
| mb_subsystem               | mb_subsystem | Using cached IP results |     |     |     |     |      |      |             |               |             |           |                 |     | 7335 | 894  | 0    | 0   |
|                            |              |                         |     |     |     |     |      |      |             |               |             |           |                 |     | 134  | 130  | 0    | 0   |
|                            |              |                         |     |     |     |     |      |      |             |               |             |           |                 |     | 1    | 0    | 0    | 0   |

## Step 6: Create a Top-Level Wrapper

- Under Design Sources, right-click the block design `mb_subsystem` and click **Create HDL Wrapper**.

In the Create HDL Wrapper dialog box, Let Vivado manage wrapper and auto-update is selected by default.

- Click **OK**.



## Step 7: Take the Design through Implementation

- In the Flow Navigator, click **Generate Bitstream**.

The No implementation Results Available dialog box opens.

- Click **Yes**.

The Launch Runs dialog box opens.

- Make the appropriate choices and click **OK**.

Bitstream generation can take several minutes to complete. Once it finishes, the Bitstream Generation Completed dialog box asks you to select what to do next.

4. Keep the default selection of Open Implemented Design and click **OK**.
5. Verify that all timing constraints have been met by looking at the Timing - Design Timing Summary window, as shown in the following figure.



**Note:** The timing summary for your design might be slightly different.

## Step 8: Export the Design to the Vitis software platform



**IMPORTANT!** *For the usb driver to install, you must power on and connect the board to the host PC before launching the Vitis software platform.*

Next, open the design and export to the Vitis software platform.

1. From the Vivado File menu, select **File > Export > Export Hardware**. The Export Hardware Platform dialog box opens.
2. Click **Next**.
3. Select the **Include bitstream** option using the radio button in the Output view and click **Next**.



4. Leave the XSA file name field at its default value and click **Next**. (The following figure shows Windows-specific settings.)

# Programming an Embedded MicroBlaze Processor (XD131)



5. Click **Finish**. This will export the hardware XSA File in the project directory.



6. To launch the Vitis software platform, select **Tools > Launch Vitis IDE**. The Eclipse Launcher dialog box opens.

Step 9: Create a “Platform Component”

The Vitis software platform launches in a separate window.



1. Select **Open Workspace** and select an new folder for the desired Workspace location, such as C:\Projects\Vitis\_Worksspaces\microblaze-system (Windows-specific).
2. Select **File > New Component > Platform** or under **Embedded Development** click **Create Platform Component**.



3. Click **Next**.
4. In the Platform Creation Flow window, select the **Hardware Design** option and Click **Browse** to open the Select Hardware Design (XSA) window. Navigate to

the directory where the XSA file was created in Vivado and click **Open**.



5. Click **Next**.

6. In the Operating System and Processor window, select the **standalone** for the operating system and **microblaze\_0** for the processor.



7. Click **Next**.

## 8. Review the Platform Component Creation Summary and click **Finish**.



## Step 10: Create a “Peripheral Test” Application

### 1. Select **File > New Component > From Examples** or under **Get Started** click **Examples**.



## 2. Select Peripheral Tests.



## 3. Click on Create Application Component from Template.



## 4. Click Next.

## Programming an Embedded MicroBlaze Processor (XD131)

5. In the Platform page, select the previously created platform.



6. Click **Next**.

7. In the Domain page leave all the fields at their default values and click **Next**.



8. Review the Summary and click **Finish**.



9. A new `peripheral_tests` application is created. If the `testperiph.c` file is not already open, select **MICROBLAZE-SYSTEM/peripheral\_tests** [Application]/Sources/src/testperiph.c, and double-click to open the source file. Modify the source file by inserting a while statement at approximately line 18. Press **Ctrl + S** to save the file.

- a. In line 18, add `while(1)` above the curly brace as shown in the following figure.

```

C:\testperiph.c
peripheral_tests > src > testperiph.c > main
1
2 #include <stdio.h>
3 #include "xparameters.h"
4 #include "xil_printf.h"
5
6 #include "xgpio.h"
7 #include "gpio_header.h"
8
9 #include "xuartlite.h"
10 #include "uartlite_header.h"
11 int main()
12 {
13
14     static XGpio axi_gpio_0;
15     static Xuartlite axi_uartlite_0;
16
17     printf("---Entering main---\n\r");
18     while(1)
19     {
20         int status;
21         printf("\n\rRunning GpioOutputExample for axi_gpio_0 ... \r\n");
22         status = GpioOutputExample(&axi_gpio_0, XPAR_AXI_GPIO_0_BASEADDR);
23         if (status == 0) {
24             printf("GpioOutputExample PASSED \r\n");
25         } else {
26             printf("GpioOutputExample FAILED \r\n");
27         }
28     }
29 }
```

10. To build the application select **peripheral\_tests** under **Flow > Component** and click on **Build**. Click **OK** on the pop-up window to build the associated platform.



11. Wait for the application to finish compiling.

## Step 11: Execute the Software Application on a SP701 Board



**IMPORTANT!** Make sure that you have connected the target board to the host computer and it is turned on.

1. To start the debug session, click on the Debug icon under **Flow > Component [peripheral\_tests] > Debug**.



2. The Debug perspective window opens, if the testperiph.c file is not already open, select .../src/testperiph.c, and double-click to open the source file.



3. Add a breakpoint in the code so that the processor stops code execution when the breakpoint is encountered. To do so, scroll down to line 24 and click on the left pane red dot, which adds a breakpoint on that line of code, as shown in the following figure.



```

testperiph.c x
peripheral_tests > src > testperiph.c > main
1
2 #include <stdio.h>
3 #include "xparameters.h"
4 #include "xil_printf.h"
5
6 #include "xgpio.h"
7 #include "gpio_header.h"
8
9 #include "xuartlite.h"
10 #include "uartlite_header.h"
11 int main()
12 {
13     static XGpio axi_gpio_0;
14     static Xuartlite axi_uartlite_0;
15
16     print("---Entering main---\n");
17     while (1)
18     {
19         int status;
20         print("\nRunning GpioOutputExample for axi_gpio_0 ... \r\n");
21         status = GpioOutputExample(&axi_gpio_0, XPAR_AXI_GPIO_0_BASEADDR);
22         if (status == 0) {
23             print("GpioOutputExample PASSED \r\n");
24         } else {
25             print("GpioOutputExample FAILED \r\n");
26         }
27     }
28 }

```

4. Configure your terminal program (e.g., TeraTerm, Putty) to connect to the SP701 Serial Port with the settings shown below.

- Baud rate = 9600
- Data bits = 8
- Stop bits = 1
- Flow control = None
- Parity = None

## Step 12: Connect to Vivado Logic Analyzer

Connect to the SP701 board using the Vivado Logic Analyzer.

1. In the Vivado IDE session, from the Program and Debug drop-down list of the Vivado Flow Navigator, select **Open Hardware Manager**.
2. In the Hardware Manager window, click **Open target > Open New Target**.



The Open New Hardware Target dialog box opens, shown in the following figure.



3. Click **Next**.
4. On the Hardware Server Settings page, ensure that the Connect to field is set to **Local server (target is on local machine)** as shown in the following figure, and click **Next**.



5. On the Select Hardware Target page, click **Next**.

6. Ensure that all the settings are correct on the Open Hardware Target Summary dialog box and click **Finish**.

**Note:** You can also use the Auto Connect option to connect to the target hardware.

### Step 13: Set the MicroBlaze to Logic Cross Trigger

When the Vivado Hardware Session successfully connects to the SP701 board, you see the information shown in the following figure:



1. Select the **Settings - hw\_il\_1** tab and set the Trigger Mode Settings as follows:
  1. Set Trigger mode to **TRIG\_IN\_ONLY**.
  2. Set TRIG\_OUT mode to **TRIG\_IN\_ONLY**.
  3. Under Capture Mode Settings, ensure that Trigger position in window is set to **512**.



2. Arm the ILA core by clicking the Run Trigger ➤ button.

This arms the ILA. You should see the status “Waiting for Trigger” in the **Status - hw\_il\_1** tab as shown in the following figure.



3. In the Vitis software platform Debug window, click **MicroBlaze #0** and then click the **Continue** button ➤.

The code will execute until the breakpoint set on line 24 in testperiph.c file is reached. As the breakpoint is reached, this triggers the ILA, as shown in the following figure.



This demonstrates that when the breakpoint is encountered during code execution, the MicroBlaze triggers the ILA that is set up to trigger. This way you can monitor the state of the hardware at a certain point of code execution.

## Step 14: Set the Logic to Processor Cross- Trigger

Now try the logic to processor side of the cross-trigger mechanism. In other words, remove the breakpoint that you set earlier on line 24 to have the ILA trigger the

processor and stop code execution.

1. Select the **Breakpoints** tab towards the bottom left corner of the window, and clear the **testperiph.c [line: 24]** check box. This removes the breakpoint that you set up earlier.

Alternatively, you can also right click on the breakpoint in the testperiph.c file, and select **Disable Breakpoint**.

2. In the Debug window, right-click the **MicroBlaze #0 target** and select **Continue** button .

The code runs continuously because it has an infinite loop.

You can see the code executing in the Terminal Window.

3. In Vivado, select the **Settings - hw\_il\_1** tab. Change the Trigger Mode to **BASIC\_OR\_TRIG\_IN** and the TRIG\_OUT mode to **TRIGGER\_OR\_TRIG\_IN**.

4. Click on the (+) sign in the Trigger Setup window to add the slot\_0 :

`microblaze_0_axi_periph_M00_AXI : AwVALID` signal from the Add Probes window.

5. In the Trigger Setup window, for slot\_0 :

`microblaze_0_axi_periph_M00_AXI : AwVALID` signal, ensure that the Operator field is set to **==**, the Radix field to **[B] (Binary)** and the Value field to **1 (logical one)**.

This essentially sets up the ILA to trigger when the awvalid transitions to a value of 1.



6. Click the Run Trigger button to 'arm' the ILA in the Status - hw\_il\_1 window.

The ILA immediately triggers as the application software is continuously performing a write to the GPIO thereby toggling the `net_slot_0\axi_awvalid` signal, which causes the ILA to trigger. The ILA in turn, toggles the TRIG\_OUT signal, which signals the processor to stop code execution.

This is seen in Vitis in the highlighted area of the debug window.



## Conclusion

In this tutorial, you:

- Stitched together a design in the Vivado IP integrator
- Took the design through implementation and bitstream generation
- Exported the hardware to Vitis
- Created and modified application code that runs on a Standalone Operating System
- Modified the linker script so that the code executes from the DDR3 memory
- Verified cross-trigger functionality between the MicroBlaze processor executing code and the design logic

## Lab Files

The Tcl script `lab_classic_mb.tcl` is included with the design files to perform all the tasks in Vivado. The Vitis software platform operations must be done in the Vitis GUI. You will need to modify the Tcl script to match the desired project path and project name on your machine.

# Programming an Embedded MicroBlaze V Processor

In this tutorial, you will create a simple AMD soft-processor system for a Spartan-7 FPGA using AMD Vivado™ IP integrator.

This tutorial uses the new AMD MicroBlaze™ V soft-core RISC-V processor.

## Introduction

The MicroBlaze V system includes native AMD IP including:

- MicroBlaze V processor
- AXI block RAM
- Double Data Rate 3 (DDR3) memory
- UARTLite
- AXI GPIO
- MicroBlaze Debug Module (MDM) V
- Proc Sys Reset
- Local memory bus (LMB)

Parts of the block design are constructed using the Platform Board Flow feature.

This lab also shows the cross-trigger capability of the MicroBlaze V processor.

The feature is demonstrated using a software application code developed in the Vitis software platform in a stand-alone application mode.

This lab targets the AMD SP701 FPGA Evaluation Kit.

## Step 1: Start the Vivado IDE and Create a Project

1. Start the Vivado IDE by clicking the Vivado desktop icon or by typing `vivado` at a command prompt.
2. From the Quick Start page, select **Create Project**.
3. In the New Project dialog box, use the following settings:
  - a. In the Project Name dialog box, type the project name and location.
  - b. Make sure that the **Create project subdirectory** check box is selected.  
Click **Next**.
  - c. In the Project Type dialog box, select **RTL project**. Ensure that the **Do not specify sources at this time** check box is cleared. Click **Next**.
  - d. In the Add Sources dialog box, set the Target language to either **VHDL** or **Verilog**. You can leave the Simulator language selection to **Mixed**.

- e. Click **Next**.
- f. In Add Constraints dialog box, click **Next**.
- g. In the Default Part dialog box, select **Boards** and choose **Spartan-7 SP701 Evaluation Platform**. Click **Next**.
- h. Review the project summary in the New Project Summary dialog box and click **Finish** to create the project.

Because you selected the SP701 board when you created the Vivado IDE project, you see the following message in the Tcl Console:

```
set_property board_part xilinx.com:sp701:part0:1.1
[current_project]
```

Although Tcl commands are available for many of the actions performed in the Vivado IDE, they are not explained in this tutorial. Instead, a Tcl script is provided that can be used to recreate this entire project. See the Tcl Console for more information. You can also refer to the *Vivado Design Suite Tcl Command Reference Guide* (UG835) for information about the `write_bd_tcl` commands.

## Step 2: Create an IP Integrator Design

1. From Flow Navigator, under IP integrator, select **Create Block Design**.
2. Specify the IP subsystem design name. For this step, you can use `mbv_subsystem` as the Design name. Leave the Directory field set to its default value of <Local to Project>. Leave the Specify source set drop-down list set to its default value of Design Sources.
3. Click **OK** in the Create Block Design dialog box, shown in the following figure.



4. In the IP integrator diagram area, right-click and select **Add IP**. The IP integrator Catalog opens. Alternatively, you can also select the Add IP icon in the middle of the canvas.

This design is empty. Press the button to add IP.

5. Type **mig** in the Search field to find the MIG core, then select **Memory Interface Generator (MIG 7 Series)**, and press **Enter**.



The Designer Assistance link becomes active in the block design banner.

6. Click **Run Block Automation**.



The Run Block Automation dialog box opens.

## Programming an Embedded MicroBlaze Processor (XD131)



7. Click **OK**. This instantiates the MIG core and connects the I/O interfaces to the I/O interfaces for the DDR memory on the SP701 board.



8. Right-click anywhere in the block design canvas, and select **Add IP**. The IP catalog opens.
9. In the Search field, type **micr** to find the MicroBlaze V IP, then select **MicroBlaze V**, and press **Enter**.

**Note:** If not displayed by default, the IP Details window can be displayed by clicking **CTRL+Q** on the keyboard while searching for IP.

The screenshot shows the IP Integrator interface. At the top, there is a search bar with the text "Search: micr" and a result count "(6 matches)". Below the search bar is a list of search results, each preceded by a small icon:

- MicroBlaze
- MicroBlaze Debug Module (MDM)
- MicroBlaze Debug Module (MDM) V
- MicroBlaze MCS
- MicroBlaze MCS V
- MicroBlaze V

Below the search results, a message box displays the instruction: "ENTER to select, ESC to cancel, Ctrl+Q for IP details".

A modal dialog box titled "IP Details" is open, containing the following information:

| Name:        | MicroBlaze V                                                                                                     |
|--------------|------------------------------------------------------------------------------------------------------------------|
| Version:     | 1.0 (Rev. 1)                                                                                                     |
| Interfaces:  | AXI4, AXI4-Stream                                                                                                |
| Description: | The Xilinx MicroBlaze V soft processor core, implementing the free and open RISC-V Instruction Set Architecture. |
| Status:      | Production                                                                                                       |
| License:     | Included                                                                                                         |
| Change Log:  | <a href="#">View Change Log</a>                                                                                  |
| Vendor:      | Xilinx, Inc.                                                                                                     |
| VLAN:        | xilinx.com:ip:microblaze_riscv:1.0                                                                               |
| Repository:  | C:/Xilinx/Vivado/2024.1/data/ip                                                                                  |

## Use the Board Window to Connect to Board Interfaces

There are several ways to use an existing interface in IP integrator. Use the Board window to instantiate some of the interfaces that are present on the SP701 board.

1. Navigate to the **Windows>Board**.
2. Select the **Board** window to see the interfaces present on the SP701 board.



In the Board window, notice that the DDR3 SDRAM interface is connected as shown by the circle in the following figure. This is because you used the Block Automation feature in the previous steps to connect the MIG core to the board interfaces for DDR3 SDRAM memory.



3. From the Board window, select **UART** under the Miscellaneous folder, and drag and drop it into the block design canvas.  
This instantiates the AXI Uartlite IP on the block design.
4. From the Board window, select **LED** under the General Purpose Input or Output folder, and drag and drop it into the block design canvas.  
This instantiates the GPIO IP on the block design and connects it to the on-board LEDs.
5. Next, from the Board window, select **FPGA Reset** under the Reset folder, and drag and drop it into the block design canvas.  
This connects the CPU push button reset to the MIG core IP.  
The block design now should look like the following figure.



## Add Peripheral: AXI block RAM Controller

1. Add the AXI block RAM Controller, shown in the following figure, by right-clicking the IP integrator canvas and selecting **Add IP**.



The block design now should look like the following figure.



## Run Block Automation

1. Click **Run Block Automation**, as shown below.



The **Run Block Automation** dialog box opens.

2. On the **Run Block Automation** dialog box:
  - Leave Preset as the default value, **None**.
  - Set Local Memory to **64 KB**.
  - Leave the Local Memory ECC as the default value, **None**.
  - Set Cache Configuration to **32 KB**.
  - Set Debug Module to **Debug Enabled**.
  - Leave the Peripheral AXI Port option as the default value, **Enabled**.
  - Leave the Interrupt Controller option unchecked.
  - Leave The Clock source option set to **/mig\_7series\_0/ui\_addn\_clk\_0 (100 MHz)**.

# Programming an Embedded MicroBlaze Processor (XD131)



### 3. Click OK.

This generates a basic MicroBlaze V system in the IP integrator diagram area, as shown in the following figure.



## Use Connection Automation

Run Connection Automation provides several options that you can select to make connections. This section will walk you through the first connection, and then you will use the same procedure to make the rest of the required connections for this tutorial.

1. Click **Run Connection Automation** as shown in the following figure.



The Run Connection Automation dialog box opens.

2. Check the All Automation check box in the left pane of the dialog box as shown in the following figure. This selects interfaces to run Connection Automation for.



3. Use the following table to set options in the Run Connection Automation dialog box.

**Table 1: Run Connection Automation Options**

| Connection                                                                     | More Information                                                                                           | Setting                                              |
|--------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------|------------------------------------------------------|
| axi_bram_ctrl_0 <ul style="list-style-type: none"> <li>• BRAM_PORTA</li> </ul> | The only option for this automation is to instantiate a new Block Memory Generator as shown under options. | Leave the Blk_Mem_Gen to its default option of Auto. |

| Connection                                                                        | More Information                                                                                                                                                                                                                                                                                                                                                                                               | Setting                                                                                                                                                                                                                                                                           |
|-----------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| axi_bram_ctrl_0<br><ul style="list-style-type: none"> <li>• BRAM_PORTB</li> </ul> | <p>The Run Connection Automation dialog box opens and gives you two choices:</p> <ul style="list-style-type: none"> <li>• Instantiate a new BMG and connect the PORTB of the AXI block RAM Controller to the new BMG IP</li> <li>• Use the previously instantiated BMG core and automatically configure it to be a true dual- ported memory and connected to PORTB of the AXI block RAM Controller.</li> </ul> | Leave the Blk_Mem_Gen option to its default value of Auto.                                                                                                                                                                                                                        |
| axi_bram_ctrl_0<br><ul style="list-style-type: none"> <li>• S_AXI</li> </ul>      | <p>Two options are presented in this case. The Master field can be set for either cached or non-cached accesses.</p>                                                                                                                                                                                                                                                                                           | <p>The Run Connection Automation dialog box offers to connect this to the /microblaze_riscv_0 (Cached). Leave it to its default value. In case, cached accesses are not desired this could be changed to /microblaze_riscv_0 (Periph).</p> <p>Leave the Clock Connection (for</p> |

| Connection                                | More Information                                                                                                                                                                                                           | Setting                                                   |
|-------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------|
|                                           |                                                                                                                                                                                                                            | unconnected clks) field set to its default value of Auto. |
| axi_gpio_0<br>• S_AXI                     | The Master field is set to its default value of /microblaze_0 (Periph).<br>The Clock Connection (for unconnected clks) field is set to its default value of Auto.                                                          | Keep these default settings.                              |
| axi_uartlite_0<br>• S_AXI                 | The Master field is set to its default value of /microblaze_0 (Periph).<br>The Clock Connection (for unconnected clks) field is set to its default value of Auto.                                                          | Keep these default settings.                              |
| mig_7series_0<br>• S_AXI                  | The Master field is set to microblaze_0 (Cached). Leave it to this value so the accesses to the DDR3 memory are cached accesses.<br>The Clock Connection (for unconnected clks) field is set to its default value of Auto. | Keep these default settings.                              |
| Rst_mig_7_series_0_100M<br>• ext_reset_in | The reset pin of the Processor Sys Reset IP will be connected to the board reset pin.                                                                                                                                      | Keep the default setting.                                 |

- After setting the appropriate options, as shown in the table above, click **OK**.

At this point, your IP integrator diagram area should look like the following figure.



**Note:** The relative placement of your IP might be slightly different.

## Mark Nets for Debugging

1. To monitor the AXI transactions taking place between the MicroBlaze and the GPIO, select the interface net connecting M00\_AXI interface pin of the microblaze\_riscv\_0\_axi\_periph instance and the S\_AXI interface pin of the axi\_gpio\_0 instance.
2. Right-click and select **Debug** from the context menu.  
**Note:** The Designer Assistance is available as indicated by the Run Connection Automation link in the banner of the block design.
3. Click **Run Connection Automation**.
4. In the Run Connection Automation dialog box, go with the default setting as shown in the following figure.

# Programming an Embedded MicroBlaze Processor (XD131)



5. Click **OK**.
6. Click the Regenerate Layout button in the IP integrator toolbar to generate an optimum layout for the block design. The block diagram should look like the following figure.



**Note:** The relative placement of your IP might be slightly different.

## Step 3: Memory-Mapping the Peripherals in IP Integrator

1. Click the **Address Editor** window.
2. In the Address Editor, do the following:
  - a. Expand the microblaze\_0 instance by clicking on the Expand All icon in the toolbar to the top of the Address Editor window.

- b. Ensure the range of microblaze\_riscv\_0/mig\_7\_series\_0/memmap IP in both the Data and the Instruction section are **512 MB**, as shown in the following figure.



The screenshot shows the Address Editor window with the following details:

| Name                                                     | Interface | Slave Segment | Master Base Address | Range | Master High Address |
|----------------------------------------------------------|-----------|---------------|---------------------|-------|---------------------|
| Network 0                                                |           |               |                     |       |                     |
| /microblaze_riscv_0                                      |           |               |                     |       |                     |
| /microblaze_riscv_0/Data (32 address bits : 4G)          |           |               |                     |       |                     |
| /axi_bram_ctrl_0/S_AXI                                   | S_AXI     | Mem0          | 0xC000_0000         | 8K    | 0xC000_1FFF         |
| /axi_gpio_0/S_AXI                                        | S_AXI     | Reg           | 0x4000_0000         | 64K   | 0x4000_FFFF         |
| /axi_uartlite_0/S_AXI                                    | S_AXI     | Reg           | 0x4060_0000         | 64K   | 0x4060_FFFF         |
| /microblaze_riscv_0_local_memory/dlmb_bram_if_cntlr/SLMB |           | Mem           | 0x0                 | 64K   | 0xFFFF              |
| /mig_7series_0/memmap                                    | S_AXI     | memaddr       | 0x8000_0000         | 512M  | 0x9FFF_FFFF         |
| /microblaze_riscv_0/Instruction (32 address bits : 4G)   |           |               |                     |       |                     |
| /axi_bram_ctrl_0/S_AXI                                   | S_AXI     | Mem0          | 0xC000_0000         | 8K    | 0xC000_1FFF         |
| /microblaze_riscv_0_local_memory/limb_bram_if_cntlr/SLMB |           | Mem           | 0x0                 | 64K   | 0xFFFF              |
| /mig_7series_0/memmap                                    | S_AXI     | memaddr       | 0x8000_0000         | 512M  | 0x9FFF_FFFF         |

- c. The top of the Address Editor window should show Assigned (11), indicating all 11 interfaces were assigned addresses. If Unassigned shows any interfaces unassigned, click on the **Assign All** arrow .

You must also ensure that the memory in which you are going to run and store your software is within the cacheable address range. This occurs when you enable Instruction Cache and Data Cache, while running the Block Automation for the MicroBlaze V processor.

To use either Memory IP DDR or AXI block RAM, those IP must be in the cacheable area; otherwise, the MicroBlaze V processor cannot read from or write to them. Validating the design will automatically re-configure the MicroBlaze V processor's cacheable address range.

## Step 4: Validate Block Design

To run design rule checks on the design:

1. Click the Validate Design button on the toolbar, or select **Tools > Validate Design**.

The Validate Design dialog box informs you that there are no critical warnings or errors in the design.

2. Click **OK**.
3. Save your design by pressing **Ctrl+S**, or select **File > Save Block Design**.

## Step 5: Generate Output Products

1. In the Sources window, select the block design, then right-click it and select **Generate Output Products**. Alternatively, you can click **Generate Block**

**Design in the Flow Navigator.**

The Generate Output Products dialog box opens.

2. Click **Generate**.



The Generate Output Products dialog box informs you that Out-of-context module runs were launched.

3. Click **OK**.
4. Wait a few minutes for all the Out-of-Context module runs to finish as shown in the Design Runs windows.

| Name                       | Constraints | Status                  | Progress | WNS | TNS | WHS | THS | WBSS | TPWS | Total Power | Failed Routes | Methodology | RQA Score | QoR Suggestions | LUT | FF   | BRAM | URAM | DSP |
|----------------------------|-------------|-------------------------|----------|-----|-----|-----|-----|------|------|-------------|---------------|-------------|-----------|-----------------|-----|------|------|------|-----|
| synth_1 (active)           | constrs_1   | Not started             | 0%       |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| impl_1                     | constrs_1   | Not started             | 0%       |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| Out-of-Context Module Runs |             |                         |          |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| mbv_subsystem              |             | Submodule Runs Complete | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 6597 | 5841 | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 2338 | 180  | 20   | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 108  | 114  | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 48   | 64   | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 283  | 266  | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 0    | 1    | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | Using cached IP results | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 4    | 2    | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 4    | 2    | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 10   | 12   | 16   | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 134  | 193  | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 19   | 40   | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 10   | 12   | 2    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 4895 | 620  | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 134  | 130  | 0    | 0   |
| mbv_subsystem              | mbv_subsys  | Submodule Runs Complete | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     | 1607 | 232  | 3.5  | 0   |
| mbv_subsystem              | mbv_subsys  | synth_design Complete!  | 100%     |     |     |     |     |      |      |             |               |             |           |                 |     |      |      |      |     |

## Step 6: Create a Top-Level Wrapper

- Under Design Sources, right-click the block design `mbv_subsystem` and click **Create HDL Wrapper**.

In the Create HDL Wrapper dialog box, Let Vivado manage wrapper and auto-update is selected by default.

- Click **OK**.



## Step 7: Take the Design through Implementation

- In the Flow Navigator, click **Generate Bitstream**.

The No implementation Results Available dialog box opens.

- Click **Yes**.

The Launch Runs dialog box opens.

- Make the appropriate choices and click **OK**.

Bitstream generation can take several minutes to complete. Once it finishes, the Bitstream Generation Completed dialog box asks you to select what to do next.

4. Keep the default selection of Open Implemented Design and click **OK**.
5. Verify that all timing constraints have been met by looking at the Timing - Design Timing Summary window, as shown in the following figure.



**Note:** The timing summary for your design might be slightly different.

## Step 8: Export the Design to the Vitis software platform



**IMPORTANT!** *For the usb driver to install, you must power on and connect the board to the host PC before launching the Vitis software platform.*

Next, open the design and export to the Vitis software platform.

1. From the Vivado File menu, select **File > Export > Export Hardware**. The Export Hardware Platform dialog box opens.
2. Click **Next**.
3. Select the **Include bitstream** option using the radio button in the Output view and click **Next**.



4. Leave the XSA file name field at its default value and click **Next**. (The following figure shows Windows-specific settings.)

# Programming an Embedded MicroBlaze Processor (XD131)



5. Click **Finish**. This will export the hardware XSA File in the project directory.



6. To launch the Vitis software platform, select **Tools > Launch Vitis IDE**. The Eclipse Launcher dialog box opens.

Step 9: Create a “Platform Component”

The Vitis software platform launches in a separate window.



1. Select **Open Workspace** and select an new folder for the desired Workspace location, such as C:\Projects\Vitis\_Worksspaces\microblaze-system (Windows-specific).
2. Select **File > New Component > Platform** or under **Embedded Development** click **Create Platform Component**.



3. Click **Next**.
4. In the Platform Creation Flow window, select the **Hardware Design** option and Click **Browse** to open the Select Hardware Design (XSA) window. Navigate to

the directory where the XSA file was created in Vivado and click **Open**.



5. Click **Next**.

6. In the Operating System and Processor window, select the **standalone** for the operating system and **microblaze\_riscv\_0** for the processor.



7. Click **Next**.

## 8. Review the Platform Component Creation Summary and click **Finish**.



## Step 10: Create a “Peripheral Test” Application

### 1. Select **File > New Component > From Examples** or under **Get Started** click **Examples**.



## 2. Select Peripheral Tests.



## 3. Click on Create Application Component from Template.



## 4. Click Next.

# Programming an Embedded MicroBlaze Processor (XD131)

5. In the Platform page, select the previously created platform.



6. Click **Next**.

7. In the Domain page leave all the fields at their default values and click **Next**.



8. Review the Summary and click **Finish**.



9. A new **peripheral\_tests** application is created. If the **testperiph.c** file is not already open, select **MICROBLAZE-V-SYSTEM/peripheral\_tests [Application]/Sources/src/testperiph.c**, and double-click to open the source file. Modify the source file by inserting a while statement at approximately line 18. Press **Ctrl + S** to save the file.

- a. In line 18, add **while(1)** above the curly brace as shown in the following figure.

```

C:\testperiph.c
peripheral_tests > src > C:\testperiph.c > main
1  #include <stdio.h>
2  #include "xparameters.h"
3  #include "xil_printf.h"
4
5  #include "xgpio.h"
6  #include "gpio_header.h"
7
8  #include "xuartlite.h"
9  #include "uartlite_header.h"
10
11 int main ()
12 {
13
14     static XGpio axi_gpio_0;
15     static XUartLite axi_uartlite_0;
16
17     print("---Entering main---\n\r");
18     while () {
19         int status;
20         print("\n\rRunning GpioOutputExample for axi_gpio_0 ... \r\n");
21         status = GpioOutputExample(&axi_gpio_0, XPAR_AXI_GPIO_0_BASEADDR);
22         if (status == 0) {
23             print("GpioOutputExample PASSED \r\n");
24         } else {
25             print("GpioOutputExample FAILED \r\n");
26         }
27     }
28 }
```

10. To build the application select **peripheral\_tests** under **Flow > Component** and click on **Build**. Click **OK** on the pop-up window to build the associated platform.



11. Wait for the application to finish compiling.

## Step 11: Execute the Software Application on a SP701 Board



**IMPORTANT!** Make sure that you have connected the target board to the host computer and it is turned on.

1. To start the session, click on the Run icon under **Flow > Component [peripheral\_tests] > Run**.



2. The Debug perspective window opens, if the testperiph.c file is not already open, select .../src/testperiph.c, and double-click to open the source file.

```
#include <stdio.h>
#include <sys/mman.h>
#include <xil_printf.h>
#include <xgpio.h>
#include <gpio_header.h>
#include <xuartlite.h>
#include <xuartlite_header.h>
int main ()
{
    static XGpio axi_gpio_0;
    print(ptr: "----Entering main---\n");
    while(1)
    {
        int status;
        print(ptr: "\r\nRunning GpioOutputExample for axi_gpio_0 ... \r\n");
        status = GpioOutputExample(GpioOutputPtr: &axi_gpio_0, BaseAddress: XPAR_AXI_GPIO_0_BASEADDR);
        if (status == 0)
        {
            print(ptr: "GpioOutputExample PASSED \r\n");
        }
        else
        {
            print(ptr: "GpioOutputExample FAILED \r\n");
        }
    }
    print(ptr: "----Exiting main---");
}
```

1. Configure your terminal program (e.g., TeraTerm, Putty) to connect to the SP701 Serial Port with the settings shown below.

- Baud rate = 9600
- Data bits = 8
- Stop bits = 1
- Flow control = None
- Parity = None

## Step 12: Connect to Vivado Logic Analyzer

Connect to the SP701 board using the Vivado Logic Analyzer.

1. In the Vivado IDE session, from the Program and Debug drop-down list of the Vivado Flow Navigator, select **Open Hardware Manager**.
2. In the Hardware Manager window, click **Open target > Open New Target**.



The Open New Hardware Target dialog box opens, shown in the following figure.



3. Click **Next**.

- On the Hardware Server Settings page, ensure that the Connect to field is set to **Local server (target is on local machine)** as shown in the following figure, and click **Next**.



- On the Select Hardware Target page, click **Next**.
- Ensure that all the settings are correct on the Open Hardware Target Summary dialog box and click **Finish**.

**Note:** You can also use the Auto Connect option to connect to the target hardware.

### Step 13: Set the Logic to Processor Cross- Trigger

When the Vivado Hardware Session successfully connects to the SP701 board, you see the information shown in the following figure:



1. Select the **Settings - hw\_il\_1** tab and keep the Trigger Mode Settings with the default value:

1. Set Trigger mode to **BASIC\_ONLY**.

1. Under Capture Mode Settings, ensure that Trigger position in window is set to **512**.

1. Click on the (+) sign in the Trigger Setup window to add the slot\_0 :

`microblaze_riscv_0_axi_periph_M00_AXI : AWVALID signal from the Add Probes window.`

2. In the Trigger Setup window, for slot\_0 :

`microblaze_riscv_0_axi_periph_M00_AXI : AWVALID signal, ensure that the Operator field is set to ==, the Radix field to [B] (Binary) and the Value field to 1 (logical one).`

This essentially sets up the ILA to trigger when the awvalid transitions to a value of 1.



3. Click the Run Trigger button to ‘arm’ the ILA in the Status - hw\_il\_1 window. The ILA immediately triggers as the application software is continuously performing a write to the GPIO thereby toggling the `net_slot_0\axi_awvalid` signal, which causes the ILA to trigger. The ILA in turn, toggles the `TRIG_OUT` signal, which signals the processor to stop code execution.

## Conclusion

In this tutorial, you:

- Stitched together a design in the Vivado IP integrator
- Took the design through implementation and bitstream generation
- Exported the hardware to Vitis
- Created and modified application code that runs on a Standalone Operating System
- Modified the linker script so that the code executes from the DDR3 memory
- Verified cross-trigger functionality between the MicroBlaze processor executing code and the design logic

## Lab Files

The Tcl script `lab_riscv_mb.tcl` is included with the design files to perform all the tasks in Vivado. The Vitis software platform operations must be done in the Vitis GUI. You will need to modify the Tcl script to match the desired project path and project name on your machine.

---

Copyright © 2019–2024 Advanced Micro Devices, Inc.

[Terms and Conditions](#)