

## Implementation in Vivado

After we have finished simulating, it is now time to move on and run the implementation. In Vivado, **implementation** is the process that maps your synthesized HDL design onto the physical resources of a Xilinx FPGA. This step is similar to what Quartus refers to as **fitting**.

After synthesis is completed, Vivado proceeds with:

1. **Placement** – Vivado places the logic elements (LUTs, flip-flops, etc.) onto the FPGA fabric.
2. **Routing** – Vivado connects the placed elements using the FPGA's routing architecture.
3. **Timing Analysis** – Vivado checks whether all signal paths meet the timing constraints (setup/hold).

Once implementation is successful, you can proceed to **bitstream generation** and **programming the FPGA**.

---

### ❖ Constraints and I/O Assignments

Vivado uses a **.xdc file** (Xilinx Design Constraints) to assign pins and define I/O standards. This is equivalent to the .qsf file in Quartus.

We will also go through:

- How to add a .xdc file to your project
- How to assign ports to FPGA pins using the .xdc file or via the graphical interface

This will ensure that your design is properly connected to the hardware on your FPGA board.

Let's start!

To avoid complicating things at the beginning, I will continue using a simple design, just like last time(and\_gate).

We will start by adding the xdc file to the design sources, but first you need to download the xdc file. Click on the link below to download it.

<https://github.com/Digilent/digilent-xdc/blob/master/Arty-S7-50-Master.xdc>



The screenshot shows a GitHub code editor displaying the contents of the Arty-S7-50-Master.xdc file. The file is a Xilinx Design Constraints (xdc) file, which defines pin assignments and other design constraints for an Arty S7-50 FPGA. The code is as follows:

```
## This file is a general .xdc for the Arty S7-50 Rev. E
## To use it in a project:
## - uncomment the lines corresponding to used pins
## - rename the used ports (in each line, after get_ports) according to the top level signal names in the project
```

After you download and save the **xdc** file:

In the **Sources** window, click on **Add Sources**.



Choose **Add or Create Constraints**, then click **Next**



Select Create File, choose **xdc** as the file type, and click on finish.



The file has now been added to the constraint's files.

Double click on it to open the file. Now you will see a list of available signals (ports), each one preceded by a comment symbol (#).

```

Sources Project Summary and grind.vhd Arty-S7-50-Master.xdc
Design Sources (1)
  Constraints (1)
    constrs_1 (1)
      Arty-S7-50-Master.xdc
Simulation Sources (1)
  sim_1 (1)
Utility Sources

Project Summary
Arty-S7-50-Master.xdc
C:/Users/menyu/Downloads/Arty-S7-50-Master.xdc

1: ## This file is a general .xdc for the Arty S7-50 Rev. E
2: ## To use it in a project:
3: ## - uncomment the lines corresponding to used pins
4: ## - rename the used ports (in each line, after get_ports) according to the top level signal names in the project
5:
6: ## Clock Signals
7: #set_property -dict { PACKAGE_PIN F14 IOSTANDARD LVCMS33 } [get_ports { CLK12MHz }]; #IO_L13P_T2_MRCC_15 Sch=clk
8: #create_clock -add -name sys_clk_pin -period 83.333 -waveform { 0 41.667 } [get_ports { CLK12MHz }];
9: #set_property -dict { PACKAGE_PIN R2 IOSTANDARD SSTL13S } [get_ports { CLK100MHz }]; #IO_L12P_T1_MRCC_34 Sch=ddr3.clk
10: #create_clock -add -name sys_clk_pin -period 10.000 -waveform { 0 5.000 } [get_ports { CLK100MHz }];
11:
12: ## Switches
13: #set_property -dict { PACKAGE_PIN H14 IOSTANDARD LVCMS33 } [get_ports { sw[0] }]; #IO_L20N_T3_A19_15 Sch=sw[0]
14: #set_property -dict { PACKAGE_PIN H18 IOSTANDARD LVCMS33 } [get_ports { sw[1] }]; #IO_L21P_T3_DQS_15 Sch=sw[1]
15: #set_property -dict { PACKAGE_PIN G18 IOSTANDARD LVCMS33 } [get_ports { sw[2] }]; #IO_L21N_T3_DQS_A18_15 Sch=sw[2]
16: #set_property -dict { PACKAGE_PIN M5 IOSTANDARD SSTL13S } [get_ports { sw[3] }]; #IO_L6N_TO_VREF_34 Sch=sw[3]
17:
18: ## RGB LEDs
19: #set_property -dict { PACKAGE_PIN J15 IOSTANDARD LVCMS33 } [get_ports { led0_r }]; #IO_L23N_T3_FWE_B_15 Sch=led0_r
20: #set_property -dict { PACKAGE_PIN G17 IOSTANDARD LVCMS33 } [get_ports { led0_g }]; #IO_L14N_T2_SRCC_15 Sch=led0_g
21: #set_property -dict { PACKAGE_PIN P15 IOSTANDARD LVCMS33 } [get_ports { led0_b }]; #IO_L13N_T2_MRCC_15 Sch=led0_b
22: #set_property -dict { PACKAGE_PIN F15 IOSTANDARD LVCMS33 } [get_ports { led1_r }]; #IO_L15N_T2_DQS_ADV_B_15 Sch=led1_i
23: #set_property -dict { PACKAGE_PIN F18 IOSTANDARD LVCMS33 } [get_ports { led1_g }]; #IO_L16P_T2_A28_15 Sch=led1_g
24: #set_property -dict { PACKAGE_PIN E14 IOSTANDARD LVCMS33 } [get_ports { led1_b }]; #IO_L15P_T2_DQS_15 Sch=led1_b
25:
26: ## LEDs
27: #set_property -dict { PACKAGE_PIN E18 IOSTANDARD LVCMS33 } [get_ports { led0_l }]; #IO_L16N_T2_A27_15 Sch=led[2]
28: #set_property -dict { PACKAGE_PIN F13 IOSTANDARD LVCMS33 } [get_ports { led1_l }]; #IO_L17P_T2_A26_15 Sch=led[3]
29: #set_property -dict { PACKAGE_PIN K13 IOSTANDARD LVCMS33 } [get_ports { led2_l }]; #IO_L15N_T2_DQS_ADV_B_15 Sch=led[4]
30: #set_property -dict { PACKAGE_PIN H15 IOSTANDARD LVCMS33 } [get_ports { led3_l }]; #IO_L15P_T2_A24_15 Sch=led[5]

```

These are placeholders for assignments. To activate a constraint, simply **remove the comment symbol** and **modify the line** to match your desired pin and standard.

This is how you assign a physical pin to a specific signal in your design.

Mark the lines you want to comment out and then press this symbol

```

## This file is a general .xdc for Toggle Line Comments (Ctrl+Slash)
## To use it in a project:
## - uncomment the lines corresponding to used pins
## - rename the used ports (in each line, after get_ports) accordir

```

In this case we need to assign 2 inputs and one output for the and\_gate and I chose to have 2 Switches for **a** and **b** inputs and one Led for the output **y**.

```

## Switches
set_property -dict { PACKAGE_PIN H14 IOSTANDARD LVCMS33 } [get_ports { a }]; #IO_L20N_T3_A19_15 Sch=sw[0] port a
set_property -dict { PACKAGE_PIN H18 IOSTANDARD LVCMS33 } [get_ports { b }]; #IO_L21P_T3_DQS_15 Sch=sw[1] port b
set_property -dict { PACKAGE_PIN G18 IOSTANDARD LVCMS33 } [get_ports { sw[2] }]; #IO_L21N_T3_DQS_A18_15 Sch=sw[2]
set_property -dict { PACKAGE_PIN M5 IOSTANDARD SSTL13S } [get_ports { sw[3] }]; #IO_L6N_TO_VREF_34 Sch=sw[3]

## RGB LEDs
#set_property -dict { PACKAGE_PIN J15 IOSTANDARD LVCMS33 } [get_ports { led0_r }]; #IO_L23N_T3_FWE_B_15 Sch=led0_r
#set_property -dict { PACKAGE_PIN G17 IOSTANDARD LVCMS33 } [get_ports { led0_g }]; #IO_L14N_T2_SRCC_15 Sch=led0_g
#set_property -dict { PACKAGE_PIN P15 IOSTANDARD LVCMS33 } [get_ports { led0_b }]; #IO_L13N_T2_MRCC_15 Sch=led0_b
#set_property -dict { PACKAGE_PIN F15 IOSTANDARD LVCMS33 } [get_ports { led1_r }]; #IO_L15N_T2_DQS_ADV_B_15 Sch=led1_i
#set_property -dict { PACKAGE_PIN F18 IOSTANDARD LVCMS33 } [get_ports { led1_g }]; #IO_L16P_T2_A28_15 Sch=led1_g
#set_property -dict { PACKAGE_PIN E14 IOSTANDARD LVCMS33 } [get_ports { led1_b }]; #IO_L15P_T2_DQS_15 Sch=led1_b

## LEDs
#set_property -dict { PACKAGE_PIN E18 IOSTANDARD LVCMS33 } [get_ports { y }]; #IO_L16N_T2_A27_15 Sch=led[2] port y

```

## Manuell pin-assignment i Vivado

There is another way to do this in vivado. Click on **Layout** and then choose **I/O Planing**.



Here you can see the entire packaging and your inputs and outputs in your design and under the pin package row you can select and change the pin.

## We have already run synthesis so now we can click on implementation.

When you click on Run Implementation you will get a optional to change the number of jobs on the launch runs on local host. and that means the number of available cores you have and want to use for the process. It is recommended to use all of them for faster processing. In my case I have 8.



Click on OK.

Under the Design Runs you can see the running processes.

| Design Runs        |             |                                  |   |
|--------------------|-------------|----------------------------------|---|
| Name               | Constraints | Status                           | W |
| ✓ synth_1 (active) | constrs_1   | <b>synth_design Complete!</b>    |   |
| impl_1             | constrs_1   | Running Design Initialization... |   |
| synth_1_copy_1     | constrs_1   | Not started                      |   |

Once the implementation process is complete, you will see this window. Here, you choose 'Generate Bitstream'. This is the file we will use to program the FPGA board. And click OK.



When the generation of the Bitstream is complete, this window will appear.



- **Open Implemented Design:** To view your design after implementation.
- **View Reports:** To check timing, resource, or other reports.
- **Open Hardware Manager:** to program the FPGA.
- **Generate Memory Configuration File:** To create files for external memory.

You can also close it without choosing any option for now. To view the design or edit timing and constraints, you can always click on '**Open Implemented Design**'. We will go through this later on the Demo.



To program the FPGA board, connect it to the computer click on **Open Hardware Manager**.

Click on **Open Target** and choose **Auto Connect**.



Vivado will automatically connect directly with your FPGA



To program the FPGA right click on fpga-an serial number and choose **Program Device**.



Choose th **bit.file** and click **OK** then **Program**.



Now you can validate if your design works on the FPGA board !

## Timing Analysis in Vivado

To check if the design meets timing, I clicked on "Report Timing Summary" in Vivado. This window appeared.

- I kept the delay type as min\_max, which checks both setup and hold timing.
- I checked "Report unconstrained paths" to include paths that are not covered by timing constraints.
- You can also limit the number of paths shown or filter by slack value under Path Display.
- The command at the bottom shows the exact TCL command Vivado will run.

Then, click **OK** to generate the report and view timing results.



This will generate a time report for you

The screenshot shows the Xilinx ISE Design Suite interface with the 'Timing' tab selected. The main window displays a 'Design Timing Summary' report. On the left, a sidebar lists navigation options: General Information, Timer Settings, Design Timing Summary (which is selected and highlighted in blue), Methodology Summary, Check Timing (0), Intra-Clock Paths, Inter-Clock Paths, and Other Path Groups. The main content area is divided into three columns: Setup, Hold, and Pulse Width. Under 'Setup', it shows Worst Negative Slack (WNS) as inf, Total Negative Slack (TNS) as 0.000 ns, Number of Failing Endpoints as 0, and Total Number of Endpoints as 1. Under 'Hold', it shows Worst Hold Slack (WHS) as inf, Total Hold Slack (THS) as 0.000 ns, Number of Failing Endpoints as 0, and Total Number of Endpoints as 1. Under 'Pulse Width', it shows Worst Pulse Width Slack (WPWS) as NA, Total Pulse Width Negative Slack (TPWS) as NA, Number of Failing Endpoints as NA, and Total Number of Endpoints as NA. A note at the bottom states: 'There are no user specified timing constraints.'

## Demo for synthesis implementation, time analyzing and PIN assignment.

<https://www.youtube.com/watch?v=4oTwGYvK8Uq>

## Demo for implementation, generate bit file and programming the FPGA board.

<https://www.youtube.com/watch?v=gM2B3UIOZm0>

## Demo for implementation and Comparison of Different Strategies ( Using project example, synthesized CPU )

<https://www.youtube.com/watch?v=aSIQoeDdAzA>

---