## **Toyon Research Corporation**

# Lab 9: QPSK Receiver/Transmitter

Chilipepper Tutorial Projects

## **Table of Contents**

| Introduc  | tion           |                            | 4  |
|-----------|----------------|----------------------------|----|
| Proced    | lure           |                            | 4  |
| Object    | ives           |                            | 4  |
| Generat   | e HDL Code     |                            | 5  |
| 1.1       | MATLAB Fund    | ctions                     | 5  |
| 1.2       | MATLAB Test    | t Bench                    | 5  |
| 1.3 F     | X HDL Coder P  | Project                    | 6  |
| 1.4 d     | _offset HDL C  | Coder Project              | 7  |
| 1.5 T     | K HDL Coder Pr | roject                     | 8  |
| Export S  | mulink mode    | ls                         | 12 |
| 2.1       | Create MCU S   | Simulink Design            | 12 |
| 2.2       | Create ADC d   | driver Simulink Design     | 13 |
| 2.3       | Create DAC d   | driver Simulink Design     | 14 |
| 2.4       | Create Transi  | mitter Simulink Design     | 15 |
| 2.5       | Create Receiv  | ver Simulink Design        | 16 |
| Configur  | e Cores and E  | xport Design               | 19 |
| 3.1       | Needed IP Co   | ores                       | 19 |
| 3.2       | Configuring t  | he Pcore Ports             | 20 |
| 3.3       | Configuring t  | he GPIO Ports              | 20 |
| 3.4       | Configuring t  | he Clock Generator IP Core | 20 |
| 3.5       | Pin Assignme   | ents                       | 22 |
| Create s  | oftware proje  | ct                         | 23 |
| 4.1       | Creating a ne  | ew C Project               | 23 |
| 4.2       | Adding Suppo   | orting files               | 24 |
| 4.3       | Loading Hard   | dware Platform with iMPACT | 25 |
| 4.4       | Debugging w    | rith SDK                   | 26 |
| Testing a | ınd Design Ve  | erification                | 27 |
| 5.1       | Verification w | with Terminal              | 27 |
| Appendi   | A main.c       |                            | 28 |
| Annendi   | R LICE file    |                            | 30 |

| Lab 9: QPSK Receiver/Transmitter       | Toyon Research Corp. embedded@toyon.com        |
|----------------------------------------|------------------------------------------------|
| Creating Wireless Transceivers Using M | ATLAB to HDL translation and Toyon Chilipepper |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |
|                                        |                                                |

# Lab 9: QPSK Receiver/Transmitter

## Introduction

This lab will extend the previous labs and allow you to combine your receive core designed in lab 8 with the transmitter designed in lab 3. The focus of this combination will be on sending a message to one FPGA via terminal, sending this message to the Chilipepper for transmission, receiving the transmitted QPSK waveform using a second Chilipepper board, and outputting this message directly to another terminal as done in the previous lab. Both the transmit and receive core created in the previous labs will be used in this lab to create a single EDK project, and therefore prior knowledge of the workings of HDL Coder as well as the Xilinx EDK environment is assumed. It is recommended that you complete the previous labs before completing this lab.

This lab is created using:

- MATLAB 2013a
- Xilinx ISE Design Suite 14.4 with EDK and System Generator
- Windows 7, 64-bit

## **Procedure**

This lab is organized into a series of steps, each including general instructions and supplementary steps, allowing you to take advantage of the lab according to your experience level.

This lab consists of the following basic steps:

- Generate HDL code from a MATLAB algorithm
- Create and export Simulink models using System Generator
- Configure your created PCores and export the design into SDK
- Create software to run your design
- Test and verify your results

### **Objectives**

After completing this lab, you will be able to:

- Create a Simulink model to interface your PCore to the MicroBlaze Processor
- Send and Receive a QPSK Waveform using the Chilipepper FMC
- Send/Receive output to/from the MicroBlaze Processor Serial Port
- Create a software application to test your design

## **Generate HDL Code**

Step 1

This section will show you how to create your MATLAB function and test bench files as well as the process for generating the HDL code used in the Simulink model.

#### 1.1 MATLAB Functions

Your MATLAB functions will eventually become a core that will be synthesized into hardware. The algorithm describes the operations in each clock cycle, and processes data on a sample-by-sample basis. This lab uses the same MATLAB algorithm used in Lab 8 for the receiver core. Therefore, the MATLAB files are identical and can be simply copied from the previous Lab or downloaded from the GitHub Repo. For your convenience, a list of the required MATLAB files is given below.

- CreateAppend16BitCRC.m
- make\_srrc\_lut.m
- make\_train\_lut.m
- make\_trig\_lut.m
- mybitget.m
- qpsk\_rx.m
- qpsk\_rx\_correlator.m
- qpsk\_rx\_foc.m
- qpsk\_rx\_srrc.m
- qpsk\_rx\_toc

Additionally, the MATLAB files used for the transmitter core are identical to those used in lab 3. They can be copied directly from lab 3 or downloaded from the GitHub repo. The required files are listed below.

- qpsk\_srrc.m
- qpsk\_tx.m
- qpsk\_tx\_byte2sym.m

Lastly, the MATLAB file needed for DC Offset Correction (dc\_offset\_correction.m) can also be downloaded from the GitHub Repo or copied from a previous lab.

#### 1.2 MATLAB Test Bench

Now that you have all of the MATLAB files required for the design, you also need the associated test bench scripts to create the HDL coder projects. This lab requires 2 test bench scripts, 1 for tx/rx and the other for the dc offset core. These test bench scripts are identical to the ones used in lab 8 and lab 4 respectively. You can either copy these files from their lab locations, or download them from the GitHub repo.

## 1.3 RX HDL Coder Project

Using the same steps outlined in the previous labs, we will create 3 new HDL coder projects called rx\_qpsk, tx\_qpsk, and dc\_offset\_correction.

The first project will be rx\_qpsk.prj. Add the MATLAB function qpsk\_rx.m and the test bench script qpsk\_tb.m to the MATLAB Function and MATLAB Test Bench categories respectively.

Once you open your Workflow Advisor, you should be greeted with a screen similar to Figure 1-1 which allows you to define input types for your function. You can also allow them to be autodefined by simply selecting run, and letting MATLAB analyze your design. For the inputs listed for the qpsk\_rx function, the auto-defined types are fine.



Figure 1-1: HDL Code Generation Workflow Advisor

- 1. Open Workflow Advisor and select "Run" to define the input types.
- 2. Click on Fixed-Point Conversion and select run this task. This process may take awhile.

Once the process is completed, you should receive the "Validation succeeded" popup as usual. As in the previous labs, not all of the automatic selections are sufficient for our FPGA design; therefore several of the conversions will need to be modified.

3. Using the function dropdown menu at the top of the HDL Code Generation screen, select each of the functions in the design and modify them with the same settings used in Lab 8.

Once all modifications have been made, select "Validate Types" in the top right area of the top toolbar to verify the design for your modified Fixed-Point conversions. Again, once the process is complete, you should get a message saying Validation Succeeded.

4. Click on HDL Code Generation and modify the settings according to the previous labs. There is no pipelining required for this project. Right Click and select run this task to generate your Xilinx Block Box Design.

- 5. Once created, copy the black box and System Generator blocks to a new Model just as in the previous labs.
- 6. Save the new model as rx.slx into the sysgen directory "Lab\_9\sysgen".
- 7. Copy the qpsk\_rx\_FixPt\_xsgbbxcfg.m file into your Sysgen folder just as in the previous labs.
- 8. Create the hdl folder inside the Sysgen folder and copy your vhd files into this directory. Make sure you modify the previously copied m file to point to the new location of the vhd files.

## 1.4 dc\_offset HDL Coder Project

The next project will be dc\_offset.prj. Add the MATLAB function dc\_offset\_correction.m and the test bench script dc\_offset\_correction\_tb.m to the MATLAB Function and MATLAB Test Bench categories respectively.

Once you open your Workflow Advisor, you will again be greeted with a screen similar to Figure 1-1. For the inputs listed for the dc\_offset\_correction function, the auto-defined types are fine.

- 1. Open Workflow Advisor and select "Run" to define the input types.
- 2. Click on Fixed-Point Conversion and select run this task. This process may take awhile. You may need to add the DC.prn file to your MATLAB directory to get the test bench file to fully compile.

As in the previous labs, not all of the automatic selections are sufficient for our FPGA design; therefore several of the conversions will need to be modified.

- 3. Using the function dropdown menu at the top of the HDL Code Generation screen, select each of the functions in the design and modify them with the same settings used in Lab 4.
- 4. Once all modifications have been made, select "Validate Types" in the top right area of the top toolbar to verify the design for your modified Fixed-Point conversions.
- 5. Click on HDL Code Generation and modify the settings according to the previous labs. **This project requires pipelining** to meet timing constraints just as in lab 4. Configure these settings using lab 4 as a reference. Your pipelining settings should look similar to Figure 1-2 below.



1-2: Modified optimization settings to allow for register pipelining

- 6. Once created, **copy the black box into the rx.slx model**. Both of these MATLAB algorithms cores will be exported using the same Simulink model.
- 7. Copy the <code>qpsk\_rx\_FixPt\_xsgbbxcfg.m</code> file into your Sysgen folder just as in the previous labs.
- 8. Copy your vhd files into the hdl directory. Make sure you modify the previously copied m file to point to the new location of the vhd files.

#### 1.5 TX HDL Coder Project

The Last project will be qpsk\_tx.prj. Add the MATLAB function  $qpsk_tm.m$  and the test bench script  $qpsk_tb.m$  to the MATLAB Function and MATLAB Test Bench categories respectively.

For the inputs listed for the <code>qpsk tx</code> function, the auto-defined types are fine.

1. Open Workflow Advisor and select "Run" to define the input types.

- 2. Click on Fixed-Point Conversion and select run this task. This process may take awhile.
- 3. Again, several of the conversions will need to be modified. Using the function dropdown menu at the top of the HDL Code Generation screen, select each of the functions in the design and modify them with the same settings used in Lab 3, shown in Figure 1-3 and 1-4 below.

| <b>▲</b> Input  |                |    |     |     |                      |
|-----------------|----------------|----|-----|-----|----------------------|
| clear_fifo_in   | double         | 0  | 1   | Yes | numerictype(0, 1, 0) |
| data_in         | double         | 0  | 211 | Yes | numerictype(0, 8, 0) |
| empty_in        | double         | 0  | 1   | Yes | numerictype(0, 1, 0) |
| tx_en_in        | double         | 0  | 1   | Yes | numerictype(0, 1, 0) |
| <b>▲</b> Output |                |    |     |     |                      |
| d1              | double         | 0  | 1   | Yes | numerictype(0, 1, 0) |
| d2              | double         | 0  | 211 | Yes | numerictype(0, 8, 0) |
| d3              | double         | 0  | 17  | Yes | numerictype(0, 5, 0) |
| d_out           | complex double | -1 | 1   | Yes | numerictype(1, 3, 0) |
| re_byte_out     | double         | 0  | 1   | Yes | numerictype(0, 1, 0) |
| tx_done_out     | double         | 0  | 1   | Yes | numerictype(0, 1, 0) |
| ▲ Persistent    |                |    |     |     |                      |
| count           | double         | 0  | 32  | Yes | numerictype(0, 10, 0 |
| diLatch         | double         | -1 | 1   | Yes | numerictype(1, 3, 0) |
| dqLatch         | double         | -1 | 1   | Yes | numerictype(1, 3, 0) |
| rdCount         | double         | 0  | 17  | Yes | numerictype(0, 10, 0 |
| reBuf           | double         | 0  | 8   | Yes | numerictype(0, 4, 0) |
| sentTrain       | double         | 0  | 90  | Yes | numerictype(0, 7, 0) |
| symIndex        | double         | 0  | 4   | Yes | numerictype(0, 3, 0) |
| txDone          | double         | 0  | 1   | Yes | numerictype(0, 1, 0) |
| tx_fifo         | 1×1024 double  | 0  | 211 | Yes | numerictype(0, 8, 0) |
| wrCount         | double         | 0  | 17  | Yes | numerictype(0, 10, ( |

| ⊿ Local      |               |    |      |     |                       |
|--------------|---------------|----|------|-----|-----------------------|
| CORE_LATENCY | double        | 8  | 8    | Yes | numerictype(0, 4, 0)  |
| data         | double        | 0  | 211  | Yes | numerictype(0, 8, 0)  |
| OS_RATE      | double        | 8  | 8    | Yes | numerictype(0, 4, 0)  |
| PAD_BITS     | double        | 24 | 24   | Yes | numerictype(0, 5, 0)  |
| rdIndex      | double        | 1  | 18   | Yes | numerictype(0, 10, 0) |
| sym2         | double        | 0  | 6    | Yes | numerictype(0, 3, 0)  |
| SYM_PER_BYTE | double        | 4  | 4    | Yes | numerictype(0, 3, 0)  |
| tbi          | 65 × 1 double | -1 | 1    | Yes | numerictype(1, 3, 0)  |
| tbq          | 65 × 1 double | -1 | 1    | Yes | numerictype(1, 3, 0)  |
| wrIndex      | double        | 1  | 1024 | Yes | numerictype(0, 11, 0) |

- 4. Once all modifications have been made, select "Validate Types" in the top right area of the top toolbar to verify the design for your modified Fixed-Point conversions.
- 5. Click on HDL Code Generation and modify the settings according to the previous labs. **Pipelining is required for this project**, and the settings are shown in Figure 1-3 below. Right Click and select run this task to generate your Xilinx Block Box Design.



1-3: Modified optimization settings to allow for register pipelining

- 6. Once created, copy the black box and System Generator blocks to a new Model just as in the previous labs.
- 7. Save the new model as tx.slx into the sysgen directory "Lab\_9\sysgen".
- 8. Copy the qpsk rx FixPt xsgbbxcfg.m file into your Sysgen folder.
- 9. Copy your vhd files into the hdl directory. Make sure you modify the previously copied m file to point to the new location of the vhd files.

## **Export Simulink models**

Step 2

This section will show you which of the previously created Simulink Models will be used for this lab. It is assumed you have completed the previous labs and have already created the needed Simulink models.

## 2.1 Create MCU Simulink Design

The **Simulink model**<sup>1</sup> in Figure 2-1 will be used for the control signals to and from the **MCU**.



Figure 2-1: Simulink model for MCU control

1. **Configure** this model and the system generator the same as in Lab 1, and **save** the design into the Sysgen folder. Name the file **mcu.slx** or something similar.

\_

<sup>&</sup>lt;sup>1</sup> This model can be downloaded from https://github.com/Toyon/Chilipepper/tree/master/Labs/Lab\_9/sysgen

## 2.2 Create ADC driver Simulink Design

The **Simulink model**<sup>2</sup> In Figure 2-2 will be used for creating the signals which drive the **ADC** on Chilipepper.





By default inphase is IQ\_sel high and quadrature is IQ\_sel low

This core runs at 40 MHz and demultiplexes the rxd data into two 20 MHz streams for I and Q

Figure 2-2: Simulink model for ADC control

1. **Configure** this model and the system generator the same as in Lab 1, and **save** the design into the Sysgen folder. Name the file **mcu.slx** or something similar.

-

<sup>&</sup>lt;sup>2</sup> This model can be downloaded from https://github.com/Toyon/Chilipepper/tree/master/Labs/Lab\_8/sysgen

## 2.3 Create DAC driver Simulink Design

The <u>Simulink model</u><sup>3</sup> In Figure 2-3 will be used for creating the signals which interface to the DAC. This is the same model used in Lab 2 and 3.



By default inphase is IQ\_sel high and quadrature is IQ\_sel low

Figure 2-3: Simulink model for DAC control

Configure this model and the system generator the same as in previous labs, and save the
design. Name the file dac\_driver.slx or something similar using the appropriate directory
structures.

<sup>&</sup>lt;sup>3</sup> This model can be downloaded from https://github.com/rcagley/Chilipepper/tree/master/Labs/Lab\_3/sysgen

## 2.4 Create Transmitter Simulink Design

The **Simulink model**<sup>4</sup> In Figure 2-4 will be used for transmitting the data from the DAC to the receiver core on the other board.



Figure 2-4: Simulink model for receiving DC Offset output

1. This model is the same one used in Lab 3 without the ChipScope model. **Configure** this model and the system generator the same as in the previous lab, and **save** the design. Name the file **dac\_driver.slx** or something similar using the appropriate directory structures.

\_

<sup>&</sup>lt;sup>4</sup> This model can be downloaded from https://github.com/Toyon/Chilipepper/tree/master/Labs/Lab\_9/sysgen

## 2.5 Create Receiver Simulink Design

The last **Simulink model** <sup>5</sup>needed for this FPGA design is the receiver design which includes both the rx.slx and dc\_offset.slx files from previous labs. The Model and its sub parts are shown in Figures 2.5 2.6 and 2.7.



Figure 2-5: Simulink model for receiving ADC output and processing QPSK signal

\_

<sup>&</sup>lt;sup>5</sup> This model can be downloaded from https://github.com/Toyon/Chilipepper/tree/master/Labs/Lab\_8/sysgen



2-6: Simulink model for performing dc offset correction.



2-7: Simulink design for processing the received QPSK waveform.

Configure this model and the system generator the same as in the previous labs, and save
the design. Name the file rx.slx or something similar using the appropriate directory
structures.

Don't forget to sync your memory using EDK before exporting your designs!

## **Configure Cores and Export Design**

Step 3

This section will show you how to integrate your PCores into your FPGA design using EDK. There are several components that must be configured for the design of this project. A quick list of the cores needed is given below. Refer to lab 0 sections 4.3 and 5.1 for information on how to add cores to the design.

#### 3.1 Needed IP Cores

- ADC Driver PCore created in Simulink
- MCU PCore created in Simulink
- rx PCore created in Simulink (includes DC offset)
- tx Pcore created in Simulink
- DAC Driver PCore created in Simulink
- Clock Generator IP Core
- Processing System IP Core
- AXI Interconnect IP Core
- GPIO Cores for LEDs
- AXI\_UART (Lite) Core

In addition, several of these cores will require external ports. Be sure that you have access to modifying the external port settings. Refer to Figure 3-1 Below.



Figure 3-1: EDK project ports list

## 3.2 Configuring the Pcore Ports

Each of the ports used in this lab can be configured exactly as the previous labs design. Refer to Lab 5 (RX) and lab 3 (TX) for more information on individual port configuration.

## 3.3 Configuring the GPIO Ports

There are several GPIO cores that are used in this lab to help with program verification. The required cores and their configurations are given below.

- 1) Add a GPIO core for initiating packet transmission via button press. Configure it as Channel 1 only, with a bit width of just 1 bit. Call it axi\_gpio\_button or something similar. Expand the port, right click the GPIO\_IO\_I input pin, and select "Make External". Configure the others as "no connection".
- 2) Add a GPIO core for setting the LEDs with the gain information. This is useful to determine what the approximate value of your current gain is set to. Configure it as Channel 1 only with a bit width of 6 bits. Call it axi\_gpio\_gain or something similar. Expand the port, right click the GPIO\_IO\_O input pin, and select "Make External". Configure the others as "no connection".
- 3) In addition to the LEDs for setting the gain value, this lab will have 2 general purpose LEDs that can be used for program debugging if necessary. Configure these LEDs the same as in the previous labs, 2 channels each with a bit width of 1 bit.
- 4) The next GPIO used will be an input for the user to manually set the transmit gain using the FPGA switches. Configure this input as single channel with a 5 bit width (input only), and name it axi\_gpio\_pa or something similar. Expand the port, right click the GPIO\_IO\_I input pin, and select "Make External". Configure the others as "no connection".
- 5) The last GPIO used for this lab is for changing the mode of the demo from continuous transmit to button transmit to normal operation etc. Configure this core as a single channel with a width of 2 bits. Call it axi\_gpio\_switch\_test\_modes or something similar.

Once all of your cores have been added, you design should look similar to Figure 3.1 shown above.

## 3.4 Configuring the Clock Generator IP Core

The Clock Generator is used in this project to distribute the appropriate clock signals to each of the PCores, as well as any external hardware which may require a clock signal. For this project, the Clock Generator is sourced from the 40 MHz pll\_clk\_out on the Chilipepper radio board (as described in the **Chilipepper user's guide**).

- 1. **Double click** the Clock Generator PCore and **configure** the settings as follows
  - Input Clock Frequency of 40Mhz
  - CLKFBIN Required Frequency of **40Mhz** with **no Clock Deskew**
  - CLKFBOUT Required Frequency of **40Mhz**, Required Group **PLLEO**, and **Buffered True**
  - CLKOUTO Required frequency of **20Mhz**, OPhase, **PLLEO** group and **Buffered true**
  - CLKOUT1 Required Frequency of **40MHz**, OPhase, **PLLE0** group and **Buffered true**
  - CLKOUT2 Required Frequency of 40MHz, 90 Phase, PLLE0 group and Buffered true

Now that the settings are configured you should have several clocks in your clock generator list.

2. **Connect** the pins according to the following.

| • | CLKIN -   | External Ports                               |
|---|-----------|----------------------------------------------|
| • | CLKOUT0 → | mcu::sysgen                                  |
|   |           | rx_axiw::Sysgen                              |
|   |           | tx_axiw::sysgen                              |
| • | CLKOUT1 → | Ports::clock_generator_0_tx_clk              |
|   |           | dac_driver::Sysgen_clk                       |
|   |           | adc_driver::sysgen_clk                       |
| • | CLKOUT2 → | External Ports::clock_generator_0_rx_clk_pin |
| • | CLKFBIN → | CLKFBOUT                                     |
| • | RST →     | net_gnd                                      |
| • | LOCKED -  |                                              |

Your Clock Generator port should look similar to Figure 3-2 below.

| i clock_generator_0 |                                                                                                                |
|---------------------|----------------------------------------------------------------------------------------------------------------|
| CLKIN               | External Ports::clock_generator_0_pll_pin                                                                      |
| CLKOUT0             | mcu_axiw_0::sysgen_clk<br>rx_axiw_0::sysgen_clk<br>tx_axiw_0::sysgen_clk                                       |
| CLKOUT1             | External Ports::clock_generator_0_tx_clk_pin<br>adc_driver_axiw_0::sysgen_clk<br>dac_driver_axiw_0::sysgen_clk |
| CLKOUT2             | External Ports::clock_generator_0_rx_clk_pin                                                                   |
| CLKFBIN             | clock_generator_0::CLKFBOUT                                                                                    |
| CLKFBOUT            | clock_generator_0::CLKFBIN                                                                                     |
| RST                 | net_gnd                                                                                                        |
| LOCKED              |                                                                                                                |

3-2: Clock Settings for QPSK radio Lab EDK project

### 3.5 Pin Assignments

Once the ports are configured correctly, the sysgen clock for the cores should be set as well. The last step is to setup the **pin assignments** for the external ports.

- 1. Rename the pins of the external ports so they are easily identifiable. Appendix B shows the names used in this demo, however you don't have to use the same naming convention.
- 2. Open the **Project** tab.
- 3. Double-click on the **UCF File: data\system.ucf** from this panel, to open the constraints file.
- 4. Fill in the pin out information for your design using Appendix B as a reference.
- 5. Prior to EDK version 14.4, Xilinx had a <u>documented issue</u><sup>6</sup> with AXI-bus generation for Simulink PCores targeting the Zynq FPGA. Refer to this issue for more information. As in Lab 0 section 5.2, this bug must be corrected if your **EDK version** is **14.3 or lower**. The steps to perform are identical to those in the previous labs; however they must be performed for **all** of the PCores used in this lab.
- 6. Select the **Export Design** button from the navigator window on the left. Click the **Export** and Launch SDK button. This process may take awhile.

-

<sup>&</sup>lt;sup>6</sup> Issue can be found at http://www.xilinx.com/support/answers/51739.htm

## **Create software project**

Step 4

Once the design is compiled and exported, you'll be greeted with a screen asking you where you would like to store your software project. It is very helpful to create the workspace folder in the same directory as your Sysgen and EDK folders. Doing this will keep all relevant files in the same location.

## 4.1 Creating a new C Project

This section will show you how to create a C program to test your FPGA design. The primary objectives of this program are to add processor integration on both the transmitter and receiver cores. Additionally, this code will add some Layer two features to allow for packet id and acknowledgment.

Note

It would be helpful if you have completed the Embedded System Design tutorial in the *ZedBoard AP SoC Concepts Tools and Techniques Guide*. Refer to Lab 1 for more information on the MCU signal control using C code within SDK.

- 1. Select File  $\rightarrow$  New  $\rightarrow$  Project.
- 2. Select **Xilinx** → **Application Project**, and hit next.
- 3. Name the project qpsk\_example or something similar and leave the other settings at their defaults. Hit next.
- 4. Select **Hello World** from the **Select Project Template** section.
- 5. Click **Finish**. You should now see your hello\_world.c file in your project src folder, as well as a **board support package** (bsp) folder. Feel free to rename this file to main.c or something more appropriate.
- 6. **Double click** the file to open it and **replace** all of its contents with the code in Appendix A.



If your SDK project cannot find your include files such as "xuartlite.h" and "xparameters.h" you can add these include files manually by selecting Xilinx Tools from the top menu and selecting Repositories. Under Global Repositories, click New and select your EDK project directory. See <a href="here">here</a> <sup>7</sup>for more information

\_

<sup>&</sup>lt;sup>7</sup> See http://www.xilinx.com/support/answers/35443.htm

## 4.2 Adding Supporting files

In addition to the main c file, you need the library files for the Chilipepper board. The 2 required files for this Lab are Chilipepper.c and Chilipepper.h and can be found on the GitHub<sup>8</sup> repo. Place these files in the src directory of your project workspace.

- 1. <u>Chilipepper.c</u> This file is the primary library file for the Chilipepper board. It contains functions for modifying the MCU registers as well as basic helper functions for tasks such as initialization, transmitting, and receiving.
- 2. <u>Chilipepper.h</u> This file holds the function prototypes for the Chilipepper.c functions.

Note

In addition to the Library files, you also need to include a Math library which contains the pow function that is used when creating the CRC. See Lab 3 section 4.1 for more information on how to add the Math Library to your project.

The Chilipepper.c library file is configured for both TX and RX cores as well as a UART to talk to the on board MCU and configure its settings. To use the library file properly, you must specify which of these features you will use. To do this, modify lines 8-13 of the Chilipepper.c file to define a variable for the cores you will be using. Your code should resemble the following, as we will be using all cores for this Lab.

```
#define MCU_UART
#define MCU_DRIVER
#define DC_OFFSET
#define TX_DRIVER
#define DAC_DRIVER
#define RX_DRIVER
```



If you are still not able to compile your C design due to include errors, you may need to tell SDK where your PCore drivers are stored. If you click on Xilinx Tools → Repositories, you can specify (in Global Repositories) where the EDK directory of your project is. (This will need to be changed for each new project)

<sup>&</sup>lt;sup>8</sup> https://github.com/Toyon/Chilipepper

## 4.3 Loading Hardware Platform with iMPACT

Once your program is written and compiled you are ready to test the design! This is done by programming the FPGA with your hardware descriptions defined in the bit file generated in EDK, and running your software on top of this design. For this lab, you can verify your design by connecting two Chilipepper boards together using an attenuator.

- 1. Connect the Chilipepper to the FPGA board and verify all cables are connected properly and the jumper settings are correct. Verify this by using the *Chilipepper user guide* and the *ZED Board Hardware users guide* as a reference. Also See Lab 0 for details on Jumper Configuration.
- 2. Once the FPGA and radio board are connected correctly, turn on the board.
- 3. Open iMPACT in the ISE Design tools.
- 4. Select no if Impact asks you to load the last saved project.
- 5. Select yes to allow iMPACT to automatically create a new project for you. If you receive any connection errors, verify your USB or JTAG programmer cables are connected properly.
- 6. Select the Automatic option for the JTAG boundary scan setting and click ok.
- 7. Hit yes to assign configuration files. Bypass the first file selection, but for the second selection, browse to the location of your system.bit file. It should be inside the "Implementation" folder of your EDK project folder.
- 8. Select ok on the next screen verifying that the board displayed is your Zynq xc7z020 board. It should look similar to Figure 4-2 below.
- 9. Right click on the xc7z020 board icon (should be on the right), select program and hit ok.
- 10. Repeat this process for each board you would like to test.



4-2: iMPACT configuration screen

#### 4.4 Debugging with SDK

If the hardware design is correct, you should see the LEDs on the board light up, as well as a blue light indicating the program was successful. You can now return to the SDK project screen to test your software.

- 1. Test it by **right clicking** the project name folder and selecting **Debug As** → **Launch on Hardware**.
- 2. You should now be taken to a screen which shows the first pointer initialization as highlighted. You can now start the software program by clicking the (play) button in the top menu.

## **Testing and Design Verification**

Step 5

#### **5.1** Verification with Terminal

Once you have both labs running successfully, the next step is to verify functionality by connecting the FPGAs which are running your Lab 8 design to terminals to view the TX and RX QPSK character packet.

1. Connect your FPGA to the PC using a micro USB cable. The cable should be plugged into the UART port on the FPGA, shown in Figure 5-1 below.



Figure 5-0-1: Circled in this figure is the UART port of the Xilinx Zed Board FPGA.

2. With the board powered on, open up a hyper terminal window such as Tera Term, and configure it with the following settings

## Baud: 115200; Data: 8 bit; No Parity; 1 Stop bit; No Flow Conrol.

Notice that the baud rate used is configured in the SetupPeripherals function in our main.c function created earlier. The other settings are all defaults for the XUartPs port.

3. Once the terminal is configured, you should be able to view your packets by clicking the button on the Lab 3 Demo. In addition, if you flip the switch for continuous packet transmission, you should see several hello world packets on your terminal output.

## Appendix A main.c

#### SDK function main.c

```
#include <stdio.h>
#include "platform.h"
#include "xbasic types.h"
#include "xgpio.h"
#include "xparameters.h"
#include "xstatus.h"
#include "chilipepper.h"
#include "xuartps.h"
#include "xil printf.h"
#include "xscugic.h"
#include "xil exception.h"
XGpio gpio blinky;
XUartPs uartPs;
XUartPs Config *pUartPsConfig;
int DebouncButton( void );
int SetupPeripherals( void );
void WriteLedGain( int gain );
int main()
     int gain;
    int aliveLed = 0;
    int sentCount;
     int numBytes;
    static int BlinkCount = 0;
   unsigned char id;
   unsigned char curValue;
    unsigned char rxBuf[256];
    init platform();
    if(SetupPeripherals() != XST SUCCESS)
     return -1;
    if ( Chilipepper Initialize() != 0 )
     return -1;
    Chilipepper SetPA(1);
    Chilipepper SetTxRxSw(1); // 0- transmit, 1-receive
    while (1)
      gain = Chilipepper ControlAgc(); //update the Chilipepper AGC
            // main priority is to parse OTA packets
            numBytes = Chilipepper ReadPacket( rxBuf, &id );
            if (numBytes > 0)
                  XGpio DiscreteWrite(&gpio_blinky, 1, 1);
```

```
// This is a normal receive situation.
            // We get a packet, write it to UART.
           if (numBytes > 0)
                  sentCount = 0;
                  while (sentCount < numBytes)</pre>
                        curValue = rxBuf[sentCount+4];
                        sentCount += XUartPs Send(&uartPs, &curValue, 1);
                  }
            }
            // flip the LED1 so the user knows the processor is alive
         if (BlinkCount > 500000)
             if (aliveLed == 0)
                  aliveLed = 1;
             else
                  aliveLed = 0;
            BlinkCount = 1;
            XGpio DiscreteWrite(&gpio blinky, 2, aliveLed); //blink LED
         }
   cleanup platform();
   return 0;
int SetupPeripherals( void )
     int status;
   XGpio Initialize (&gpio blinky, XPAR AXI GPIO LED DEVICE ID);
   XGpio SetDataDirection(&gpio blinky, 2, 0);
   XGpio SetDataDirection(&gpio blinky, 1, 0);
     XGpio DiscreteWrite(&gpio blinky, 1, 0);
     XGpio DiscreteWrite (&gpio blinky, 2, 0);
     //Setup UART for serial port communication
   pUartPsConfig = XUartPs LookupConfig(XPAR PS7 UART 1 DEVICE ID);
     if (NULL == pUartPsConfig) {
           return XST FAILURE;
     }
     status = XUartPs CfgInitialize(&uartPs, pUartPsConfig, pUartPsConfig-
>BaseAddress);
     if (status != XST SUCCESS) {
           return XST FAILURE;
     XUartPs SetBaudRate (&uartPs, 115200);
     return XST SUCCESS;
```

## Appendix B UCF file

EDK UCF file system.ucf

| ######################################                                       |                                    |                                                    |  |  |  |  |
|------------------------------------------------------------------------------|------------------------------------|----------------------------------------------------|--|--|--|--|
| NET clock_generator_0_pll_pin                                                | LOC = D18   IOSTANDARD = LVCMOS25; |                                                    |  |  |  |  |
| NET clock_generator_0_pll_pin                                                | TNM_NET = clock_generator_0_pll;   |                                                    |  |  |  |  |
| TIMESPEC TS_clock_generator_0_pll = PERIOD clock_generator_0_pll 40.000 MHz; |                                    |                                                    |  |  |  |  |
| ######################################                                       |                                    |                                                    |  |  |  |  |
| NET clock_generator_0_tx_clk_pin                                             | LOC = C17                          | IOSTANDARD = LVCMOS25   DRIVE = 4   SLEW = FAST;   |  |  |  |  |
| NET dac_driver_axiw_0_tx_iq_sel_pin                                          | LOC = B16                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[0]                                             | LOC =A18                           | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[1]                                             | LOC = A19                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[2]                                             | LOC = E20                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[3]                                             | LOC = G21                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[4]                                             | LOC = F19                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[5]                                             | LOC = G15                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[6]                                             | LOC = E19                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[7]                                             | LOC = G16                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[8]                                             | LOC = G19                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[9]                                             | LOC = A16                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[10]                                            | LOC = A17                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET dac_driver_axiw_0_txd_pin[11]                                            | LOC = C18                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| #######################################                                      | #####Rx - FMC inte                 | rface at 2.5V #################################### |  |  |  |  |
| NET clock_generator_0_rx_clk_pin                                             | LOC = J18                          | IOSTANDARD = LVCMOS25   DRIVE = 4   SLEW = FAST;   |  |  |  |  |
| NET adc_driver_axiw_0_rx_iq_sel_pin                                          | LOC = N19                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[0]                                             | LOC =M21                           | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[1]                                             | LOC = J21                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[2]                                             | LOC = M22                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[3]                                             | LOC = J22                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[4]                                             | LOC = T16                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[5]                                             | LOC = P20                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[6]                                             | LOC = T17                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[7]                                             | LOC = N17                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[8]                                             | LOC = J20                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[9]                                             | LOC = P21                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[10]                                            | LOC = N18                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |
| NET adc_driver_axiw_0_rxd_pin[11]                                            | LOC = J16                          | IOSTANDARD = LVCMOS25;                             |  |  |  |  |

```
NET axi_uartlite_0_RX_pin
                                  LOC = R19
                                                | IOSTANDARD = LVCMOS25 | DRIVE = 4 | SLEW = FAST;
NET axi_uartlite_0_TX_pin
                                  LOC = L21
                                                | IOSTANDARD = LVCMOS25 | DRIVE = 4 | SLEW = FAST;
NET mcu_axiw_0_mcu_reset_pin
                                  LOC = K20
                                                | IOSTANDARD = LVCMOS25 | DRIVE = 4 | SLEW = FAST;
NET mcu_axiw_0_tx_en_pin
                                  LOC = D22
                                                | IOSTANDARD = LVCMOS25 | DRIVE = 4 | SLEW = FAST;
                                                | IOSTANDARD = LVCMOS25 | DRIVE = 4 | SLEW = FAST;
NET mcu_axiw_0_tr_sw_pin
                                  LOC = D20
NET mcu_axiw_0_rx_en_pin
                                  LOC = C22
                                                | IOSTANDARD = LVCMOS25 | DRIVE = 4 | SLEW = FAST;
NET mcu_axiw_0_pa_en_pin
                                  LOC = E21
                                                | IOSTANDARD = LVCMOS25 | DRIVE = 4 | SLEW = FAST;
NET mcu_axiw_0_init_done_pin
                                  LOC = K19
                                                | IOSTANDARD = LVCMOS25;
NET axi_gpio_gain_GPIO_IO_O_pin[0]
                                         LOC = U14 | IOSTANDARD = LVCMOS33; # "LD7"
NET axi_gpio_gain_GPIO_IO_O_pin[1]
                                         LOC = U19 | IOSTANDARD = LVCMOS33; # "LD6"
NET axi_gpio_gain_GPIO_IO_O_pin[2]
                                         LOC = W22 | IOSTANDARD = LVCMOS33; # "LD5"
NET axi_gpio_gain_GPIO_IO_O_pin[3]
                                         LOC = V22 | IOSTANDARD =LVCMOS33; # "LD4"
NET axi_gpio_gain_GPIO_IO_O_pin[4]
                                         LOC = U21 | IOSTANDARD = LVCMOS33; # "LD3"
                                         LOC = U22 | IOSTANDARD =LVCMOS33; # "LD2"
NET axi_gpio_gain_GPIO_IO_O_pin[5]
NET axi_gpio_led_GPIO_IO_pin
                                         LOC = T22 | IOSTANDARD =LVCMOS33; # "LD0"
                                         LOC = T21 | IOSTANDARD = LVCMOS33; # "LD1"
NET axi_gpio_led_GPIO2_IO_pin
NET axi_qpio_switch_test_modes_GPIO_IO_pin[0]
                                         LOC = F22 | IOSTANDARD =LVCMOS33; # "LD0"
NET axi_gpio_switch_test_modes_GPIO_IO_pin[1]
                                         LOC = G22 | IOSTANDARD =LVCMOS33; # "LD1"
                                         LOC = M15 | IOSTANDARD = LVCMOS33; # "LD2"
NET axi_gpio_pa_GPIO_IO_I_pin[0]
NET axi_gpio_pa_GPIO_IO_I_pin[1]
                                         LOC = H17 | IOSTANDARD = LVCMOS33; # "LD3"
NET axi_gpio_pa_GPIO_IO_I_pin[2]
                                         LOC = H18 | IOSTANDARD = LVCMOS33; # "LD4"
NET axi_gpio_pa_GPIO_IO_I_pin[3]
                                         LOC = H19 | IOSTANDARD = LVCMOS33; # "LD5"
                                         LOC = F21 | IOSTANDARD =LVCMOS33; # "LD5"
NET axi_qpio_pa_GPIO_IO_I_pin[4]
NET axi_gpio_button_GPIO_IO_I_pin
                                         LOC = P16 | IOSTANDARD =LVCMOS33; # "BTCenter"
```