After completing this lab, you will be able to:
- Add a VIO core in the design
- Use a VIO core to inject stimulus to the design and monitor the response
- Mark nets as debug so AXI transactions can be monitored
- Add an ILA core in Vivado
- Perform hardware debugging using the hardware analyzer
- Perform software debugging using the SDK
Open the Vivado program. Open the lab1 project you created in the previous lab or use the lab1 project from the labsolution directory, and save the project as lab2. Set Project Settings to point to the IP repository provided in the sources directory.
- Start Vivado if necessary and open either the lab1 project (lab1.xpr) you created in the previous lab or the lab1 project in the labsolutions directory using the Open Project link in the Getting Started page.
- Select File > Project > Save As … to open the Save Project As dialog box. Enter lab2 as the project name. Make sure that the Create Project Subdirectory option is checked, the project directory path is {labs} and click OK. This will create the lab2 directory and save the project and associated directory with lab2 name
- Click Settings in the Flow Navigator pane.
- Expand IP in the left pane of the Project Settings form and select Repository.
- Click on the plus button of the IP Repositories panel, browse to {sources}\lab2\math_ip and click Select.
The directory will be scanned and one IP will be detected and reported.
Specify IP Repository
- Click OK twice to close the window.
-
Click Open Block Design in the Flow Navigator pane to open the block diagram.
-
Double-click the math_ip_v1_0 to add an instance of the core to the design.
-
Click on Run Connection Automation, ensure math_ip_0 and S_AXI are selected, and click OK.
The Math IP consists of a hierarchical design with the lower-level module performing the addition. The higher-level module includes the two slave registers.
Custom Core's Main Functional Block
We want to connect the ILA core to the LED interface. Vivado prohibits connecting ILA cores to interfaces. In order to monitor the LED output signals, we need to convert the LED interface to simple output port.
- Double-click the leds instance to open its configuration form.
- Click Clear Board Parameters and click OK to close the configuration form.
- Select leds_4bit port and delete it.
- Expand the gpio interface of the leds instance to see the associate ports.
-
Move the mouse close to the end of the gpio_io_o port, left-click to select (do not select the main GPIO port), and then right click and select Make External.
Select the gpio_io_o port
The port connector named gpio_io_o will be created and connected to the port. -
Select the port gpio_io_o and change its name to leds by typing it in the properties form.
-
Enable cross triggering between the PL and PS
-
Double click on the Zynq block to open the configuration properties.
-
Click on PS-PL Configuration, and enable the PS-PL Cross Trigger interface.
-
Expand PS-PL Cross Trigger interface > Input Cross Trigger, and select CPU0 DBG REQ for Cross Trigger Input 0.
-
Similarly, expand Output Cross Trigger, and select CPU0 DBG ACK for Cross Trigger Output 0 and click OK.
Enabling cross triggering in the Zynq processing system
- Click the button and search for ila in the catalog.
- Double-click on the ILA (Integrated Logic Analyzer) to add an instance of it. The ila_0 instance will be added.
- Double-click on the ila_0 instance.
- Select Native as the Monitor type.
- Enable Trigger Out Port, and Trigger In port.
- Select the Probe Ports tab, and set the Probe Width of PROBE0 to 4 and click OK.
- Using the drawing tool, connect the PROBE0 port of the ila_0 instance to the gpio_io_o port of the leds instance.
- Connect the clk port of the ila_0 instance to the FCLK_CLK0 port of the Zynq subsystem.
- Connect TRIGG_IN of the ILA to TRIGGER_OUT_0 of the Zynq processing system, and TRIG_OUT of the ILA to the TRIGGER_IN_0.
- Click the button and search for vio in the catalog.
- Double-click on the VIO (Virtual Input/Output) to add an instance of it.
- Double-click on the _vio_instance to open the configuration form.
- In the General Options tab, leave the Input Probe Count set to 1 and set the Output Probe Count to 3
- Select the PROBE_IN Ports tab and set the PROBE_IN0 width to 9.
- Select the PROBE_OUT Ports tab and set PROBE_OUT0 width to 1 , PROBE_OUT1 width to 8 , and PROBE_OUT2 width to 8.
- Click OK.
- Connect the VIO ports to the math instance ports as follows:
PROBE_IN -> result PROBE_OUT0 -> sel PROBE_OUT1 -> ain_vio PROBE_OUT2 -> bin_vio
- Connect the CLK port of the vio_0 to FCLK_CKL0 net.
- The block diagram should look similar to shown below.
VIO added and connections made
Mark Debug the S_AXI connection between the AXI Interconnect and math_0 instance. Validate the design.
-
Select the S_AXI connection between the AXI Interconnect and the math_ip_0 instance .
-
Right-click and select Debug to monitor the AXI4Lite transactions.
Notice that a system_ila IP instance got added and the M03_AXI <-> S_AXI connection is connected to its SLOT_0_AXI interface.
The block diagram should look as shown below.
Block diagram of the design after marking AXI connection to the math_ip_0 instance for debugging
-
Click the Run Connection Automation link to see the form where you can select the desired channels to monitor.
-
Change AXI Read Address and AXI Read Data channels to Data since we will not trigger any signals of those channels.
This saves resources being used by the design.
Selecting channels for debugging
-
Select the Diagram tab, and click on the (Validate Design) button to make sure that there are no errors.
-
Verify that there are no unmapped addresses shown in the Address Editor tab.
- Right click in the Sources panel, and select Add Sources.
- Select Add or Create Constraints and click Next.
- Click the Plus button then Add Files, browse to *{sources}\lab2* and select lab2_pynz1.xdc, or lab2_pynqz2.xdc depending on the board you are using.
- Click OK and then click Finish.
- Click on the Generate Bitstream to run the implementation and bit generation processes.
- Click Save to save the project (if prompted), OK to ignore the warning (if prompted), and Yes to launch Implementation (if prompted). Click OK to launch the runs.
- When the bitstream generation process has completed successfully, click Cancel.
- Export the hardware configuration by clicking File > Export > Export Hardware… , click the box to Include Bitstream
- Click OK to export and Yes to overwrite the previous project created by lab1.
- Launch SDK by clicking File > Launch SDK and click OK.
- Right-click on the lab1 and standalone_bsp_0 and system_wrapper_hw_platfrom_0 projects in the Project Explorer view and select close project.
-
Select File > New > Application Project.
-
In the Project Name field, enter lab2 as the project name, leave all other settings to their default's and click Next (a new BSP will be created) .
-
Select the Empty Application template and click Finish.
The lab2 project will be created in the Project Explorer window of the SDK.
-
Select lab2 > src in the project view, right-click, and select Import.
-
Expand the General category and double-click on File System.
-
Browse to the {sources}\lab2 folder.
-
Select lab2.c and click Finish.
A snippet of the part of the source code is shown in the following figure. It shows that two operands are written to the custom core, the result is read, and printed out. The write transaction will be used as a trigger condition in the Vivado Logic Analyzer.
Source Code snippet
-
Right click on lab2, and select Debug As > Debug Configurations
-
Double click on Xilinx C/C++ application (System Debugger) to create a new configuration (lab2 Debug will be created), and in the Target__Setup tab, check the Enable Cross-Triggering option , and click the Browse button.
Enable cross triggering in the software environment
-
When the Cross Trigger Breakpoints dialog box opens, click Create
-
Select the options as shown below and click OK to set up the cross-trigger condition for Processor to Fabric.
Enabling CPU0 for request from PL
-
In the Cross Trigger Breakpoints dialog box click Create again.
-
Select the options as shown below and click OK to set up the cross trigger condition for Fabric to Processor.
Enabling CPU0 for request to PL
-
Click OK , then click Apply, then Close
Connect and power up the board. Download the bitstream into the target device. Start the debug session on lab2 project. Switch to the Debug perspective and establish serial communication.
- Connect and power up the board.
- Select Xilinx > Program FPGA and click Program
- Select the lab2 project in Project Explorer, right-click and select Debug As > Launch on Hardware (System Debugger) to download the application, execute ps7_init. (If prompted, click Yes to switch to the Debug perspective.) The program execution starts and suspends at the entry point.
- Select the tab. If it is not visible then select Window > Show view > Terminal tab.
- Click on and select the appropriate COM port (depending on your computer), and configure it as you did it in Lab 1.
-
Switch to Vivado.
-
Click on Open Hardware Manager from the Program and Debug group of the Flow Navigator pane to invoke the analyzer.
-
Click on the Open Target > Auto connect to establish the connection with the board.
-
Select Window > Debug Probes
The hardware session will open showing the Debug Probes tab in the Console view.
Debug probes
The hardware session status window also opens showing that the FPGA is programmed (we did it in SDK), there are three cores out of which the two ila cores are in the idle state.Hardware session status
-
Select the XC7Z020, and click on the Run Trigger Immediate button to see the signals in the waveform window.
Opening the waveform window
Setup trigger conditions to trigger on a write transaction (WSTRB) when the desired data (WDATA) of XXXX_XX12 is written. The transaction takes place when WVALID and WREADY are equal to 1.
-
Click on the hw_ila_2 tab to select it. In the Debug Probes window, under hw_ila_2, drag and drop the WDATA signal to the ILA Basic Trigger setup window .
-
Set the value to XXXX_XX12 (HEX) (the value written to the math_0 instance at line 24 of the program).
-
Similarly, add WREADY,** WSTRB, and WVALID** signals to the ILA Basic Trigger setup window.
-
Change the radix to binary for WSTRB, and change the value from xxxx to xxx1
-
Change the value of WVALID and WREADY to 1.
-
Set the trigger position of the hw_ila_2 to 512 in the Settings – hw_ila_2
Setting up the ILA
-
Similarly, set the trigger position in the Settings – hw_ila_1 tab to 512.
-
Select hw_ila_2 in the Hardware window and click on the Run Trigger button and observe that the hw_ila_2 core is armed and showing the status as Waiting For Trigger.
Hardware analyzer running and in capture mode
-
Switch to SDK.
-
Near line 27 (right click in the margin and select Show Line Numbers if necessary_),_ double click on the left border on the line where xil_printf statement is (before the while (1) statement) is defined in the lab2.c window to set a breakpoint.
Setting a breakpoint
-
Click on the Resume () button to execute the program and stop at the breakpoint.
-
In the Vivado program, notice that the hw_ila_2 status changed from Waiting for Trigger to Idle, and the waveform window shows the triggered output (select the hw_ila_data_2.wcfg tab if necessary).
-
Move the cursor to closer to the trigger point and then click on the button to zoom at the cursor. Click on the Zoom In button couple of times to see the activity near the trigger point. Similarly, you can see other activities by scrolling to right as needed.
Zoomed waveform view of the three AXI transactions
Observe the following:Around the 512th sample WDATA being written is 0x012 at offset 0 (AWADDR=0x0). At the 536th sample, offset is 0x4 (AWADDR), and the data being written is 0x034. At the 559th sample, data is being read from the IP at the offset 0x0 (ARADDR), and at 561# st mark the result (0x46) is on the RDATA bus.
-
You also should see the following output in the SDK Terminal console.
Terminal Output
In Vivado, select the VIO Cores related from the Dashboard Options windows, set the vio_1_probe_out0 so math_ip's input can be controlled manually through the VIO core. Try entering various values for the two operands and observe the output on the math_ip_1_result port in the Console pane.
- Select the hw_vio_1 core in the Dashboard Options panel.
- Click on the button and select all signals to stimulate and monitoring. Change the vio_0_probe_out0 value to 1 so the math_ip core input can be controlled via the VIO core.
VIO probes
- Change vio_0_probe_out1 value to 55 (in Hex), and similarly, vio_0_probe_out2 value to 44 (in Hex). Notice that for a brief moment a blue-colored up-arrow will appear in the Activity column and the result value changes to 099 (in Hex).
Input stimuli through the VIO core's probes
- Try a few other inputs and observe the outputs.
- Once done, set the vio_0_probe_out0 to 0 to isolate the vio interactions with the math_ip core.
Setup the ILA core (hw_ila_1) trigger condition to 0x2 for the PYNQ-Z1/PYNQ-Z2. Make sure that the switches on the board are not set at x3 (for PYNQ-Z1/PYNQ-Z2). Set the trigger equation to be ==, and arm the trigger. Click on the Resume button in the SDK to continue executing the program. Change the switches and observe that the hardware core triggers when the preset condition is met.
-
Select the hw_ila_1 in the Dashboard Options panel.
-
Add the LEDs to the Basic Trigger Setup, and set the trigger condition of the hw_ila_1 to trigger at LED output value equal to 0x5 for the PYNQ-Z1/PYNQ-Z2.
Setting up Trigger for hw_ila_1
-
Ensure that the trigger position for the hw_ila_1 is set to 512.
Make sure that the switches are not set to 11 (PYNQ-Z1/PYNQ-Z2) as this is the exit pattern.
-
Right-click on the hw_ila_1 in the hardware window, and arm the trigger by selecting Run Trigger.
The hardware analyzer should be waiting for the trigger condition to occur.
-
In the SDK window, click on the Resume button.
-
Press the push-buttons and see the corresponding LED turning ON and OFF.
-
When the condition is met, the waveform will be displayed.
ILA waveform window after Trigger
-
In Vivado, select hw_ila_1
-
In the ILA properties, set the Trigger mode to BASIC_OR_TRIGG_IN , and the TRIG_OUT mode to TRIGGER_OR_TRIG_IN
-
In SDK, in the C/C++ view, relaunch the software by right clicking on the lab2 project, and selecting Debug As > Launch on Hardware (System Debugger).
Click OK if prompted to reset the processor. The program will be loaded and the excution will suspend at the entry point
-
Arm the hw_ila_1 trigger.
-
In SDK continue execution of the software to the next breakpoint (line 27).
When the next breakpoint in SDK is reached, return to Vivado and notice the ILA has triggered.
-
Click Step Over (F6) button twice to pass the current breakpoint
-
Arm the hw_ila_1 trigger
-
Resume the software (F8) until it enters the while loop
-
Verify it is executing by toggling the dip switches
-
In Vivado, arm the hw_ila_1 trigger
-
Press the push-buttons to 0x5, and notice that the application in SDK will break at some point (This point will be somewhere within the while loop)
-
Click on the Resume button
The program will continue execution. Flip switches until it is 0x03.
-
Click the Disconnect button () in the SDK to terminate the execution.
-
Close the SDK by selecting File > Exit.
-
Close the hardware session by selecting File > Close Hardware Manager. Click OK.
-
Close Vivado program by selecting File > Exit.
-
Turn OFF the power on the board.
In this lab, you added a custom core with extra ports so you can debug the design using the VIO core. You instantiated the ILA and the VIO cores into the design. You used Mark Debug feature of Vivado to debug the AXI transactions on the custom peripheral. You then opened the hardware session from Vivado, setup various cores, and verified the design and core functionality using SDK and the hardware analyzer.
.