

# Design and Implementation of a Threshold-based 8-bit DAC on ZedBoard

Bathula Venkata Naga Motheendranadh Reddy N.Sundeep Teja

November 3, 2025

## Abstract

This report presents the design, simulation and FPGA implementation of a threshold-based digital-to-analog converter (DAC) model on the ZedBoard (Xilinx Zynq-7000). The design converts an 8-bit digital input into quantized analog steps based on a 3.3 V reference and maps the result to four LEDs on the board. The project includes synthesizable Verilog, a top-level wrapper, testbench, and XDC constraints for the ZedBoard.

## Contents

|          |                                           |          |
|----------|-------------------------------------------|----------|
| <b>1</b> | <b>Introduction</b>                       | <b>2</b> |
| <b>2</b> | <b>Design Requirements</b>                | <b>2</b> |
| <b>3</b> | <b>Theory</b>                             | <b>2</b> |
| <b>4</b> | <b>Verilog Implementation</b>             | <b>2</b> |
| 4.1      | Core module: <code>dac2.v</code>          | 2        |
| 4.2      | Top-level wrapper: <code>dac_top.v</code> | 2        |
| 4.3      | Testbench: <code>tb_dac2.v</code>         | 3        |
| <b>5</b> | <b>XDC Constraints</b>                    | <b>3</b> |
| <b>6</b> | <b>Simulation and Results</b>             | <b>4</b> |
| <b>7</b> | <b>Synthesis and Implementation Notes</b> | <b>5</b> |
| <b>8</b> | <b>Extensions and Future Work</b>         | <b>5</b> |
| <b>9</b> | <b>Conclusion</b>                         | <b>5</b> |

# 1 Introduction

A digital-to-analog converter (DAC) maps discrete digital codes into corresponding analog voltages. The presented design is a *threshold-based* DAC: it computes which discrete analog step an input code corresponds to rather than generating a continuous analog waveform. This approach is useful for FPGA demonstration and control applications where an external DAC or PWM reconstruction may be used for an actual analog output.

## 2 Design Requirements

- Input: 8-bit digital value (0–255) from board switches.
- Reference: 3.3 V (provided as  $V_{ref_m}V = 3300$ ).
- Output: 4-bit quantized step (0–8) shown on 4 LEDs (LD0–LD3).
- Implementation target: ZedBoard (Xilinx Zynq-7000).

## 3 Theory

Given an 8-bit digital input  $D$  and a reference voltage  $V_{ref}$ , the analog voltage equivalent is [  
 $V_{out} = \frac{D}{255} \times V_{ref}$ .] The design divides the analog range into 0.4 V steps (400 mV per step). The step index is therefore  
 $\lfloor \frac{V_{out}}{0.4} \rfloor = \left\lfloor \frac{(V_{ref_m}V \times D)/255}{400} \right\rfloor$ .] All computations in the Verilog implementation use integer arithmetic for synthesis.

## 4 Verilog Implementation

### 4.1 Core module: dac<sub>2.v</sub>

```
1 'timescale 1ns / 1ps
2 // Synthesizable Threshold-based DAC
3 module dac_2 (
4   input wire [7:0] digital_in, // 8-bit digital input
5   input wire [11:0] Vref_mV, // 12-bit reference (e.g., 3300)
6   output reg [3:0] analog_step // 4-bit quantized output
7 );
8   reg [19:0] analog_value_mV;
9
10  /**
11   * @begin
12   *   analog_value_mV = (Vref_mV * digital_in) / 255;
13   *   analog_step = analog_value_mV / 400;
14   */
15
16 endmodule
```

### 4.2 Top-level wrapper: dac<sub>top.v</sub>

```
1 'timescale 1ns / 1ps
2 module dac_top (
3   input wire clk,
4   input wire rstn,
5   input wire [7:0] sw,
```

```

6  output wire [3:0] led
7  );
8  wire [11:0] Vref_mV = 12'd3300;
9
10 /**
11  dac_2 u_dac (
12    .digital_in(sw),
13    .Vref_mV(Vref_mV),
14    .analog_step(led)
15 );
16 /**
17
18 endmodule

```

### 4.3 Testbench: tb<sub>dac2.v</sub>

```

1  `timescale 1ns / 1ps
2  module tb_dac_2;
3  reg [7:0] digital_in;
4  reg [11:0] Vref_mV;
5  wire [3:0] analog_step;
6  real analog_value_mV;
7
8 /**
9  dac_2 uut (
10   .digital_in(digital_in),
11   .Vref_mV(Vref_mV),
12   .analog_step(analog_step)
13 );
14
15 initial begin
16   Vref_mV = 12'd3300;
17   $display("Time(ns)\tDigital_Input\tAnalog(V)\tStep_Output");
18   $monitor("%0t\t%0d\t%0.2f\t%0d", $time, digital_in, analog_value_mV,
19           analog_step);
20
21   digital_in = 0; analog_value_mV = (digital_in / 255.0) * (Vref_mV / 1000.0); #10;
22   digital_in = 64; analog_value_mV = (digital_in / 255.0) * (Vref_mV / 1000.0); #10;
23   digital_in = 128; analog_value_mV = (digital_in / 255.0) * (Vref_mV / 1000.0);
24     #10;
25   digital_in = 192; analog_value_mV = (digital_in / 255.0) * (Vref_mV / 1000.0);
26     #10;
27   digital_in = 255; analog_value_mV = (digital_in / 255.0) * (Vref_mV / 1000.0);
28     #10;
29
30   #10 $finish;
end
/**
endmodule

```

## 5 XDC Constraints

Use the following XDC mapping (adjust pins by your ZedBoard revision if needed):

```

1
2 ## Clock Input
3
4 set_property PACKAGE_PIN E3 [get_ports clk]
5 set_property IOSTANDARD LVCMS33 [get_ports clk]
6
7 ## Reset Button (BTNO)
8
9 set_property PACKAGE_PIN T18 [get_ports rstn]
10 set_property IOSTANDARD LVCMS33 [get_ports rstn]
11
12 ## Switches (SW0-SW7)
13
14 set_property PACKAGE_PIN F22 [get_ports {sw[0]}]
15 set_property PACKAGE_PIN G22 [get_ports {sw[1]}]
16 set_property PACKAGE_PIN H22 [get_ports {sw[2]}]
17 set_property PACKAGE_PIN F21 [get_ports {sw[3]}]
18 set_property PACKAGE_PIN H19 [get_ports {sw[4]}]
19 set_property PACKAGE_PIN H18 [get_ports {sw[5]}]
20 set_property PACKAGE_PIN H17 [get_ports {sw[6]}]
21 set_property PACKAGE_PIN M15 [get_ports {sw[7]}]
22 set_property IOSTANDARD LVCMS33 [get_ports {sw[*]}]
23
24 ## LEDs (LD0-LD3)
25
26 set_property PACKAGE_PIN T22 [get_ports {led[0]}]
27 set_property PACKAGE_PIN T21 [get_ports {led[1]}]
28 set_property PACKAGE_PIN U22 [get_ports {led[2]}]
29 set_property PACKAGE_PIN U21 [get_ports {led[3]}]
30 set_property IOSTANDARD LVCMS33 [get_ports {led[*]}]

```

## 6 Simulation and Results

Run the provided testbench in Vivado or ModelSim to observe printed monitor values and waveforms. Example output (expected):

| Digital Input | Analog Voltage (V) | Step Output |
|---------------|--------------------|-------------|
| 0             | 0.00               | 0 64        |
| 0.83          | 2 128              | 1.65        |
| 4 192         | 2.48               | 6 255       |
| 3.30          | 8                  |             |

Include waveform screenshots from Vivado's simulator here (replace the file path below with your actual images):



Figure 1: Simulation waveform (replace with your capture).

## 7 Synthesis and Implementation Notes

- Ensure `dac_top.v` is set as the top module in Vivado.
- Use explicit-width constants (e.g., `12'd3300`) to avoid port-width mismatch warnings.
- Avoid declaring variables inside combinational always blocks; declare regs outside the block for synthesis.

## 8 Extensions and Future Work

- Replace the step output with a PWM generator and low-pass filter to produce a true analog voltage on an FPGA pin.
- Implement higher resolution (e.g., 10-bit) and increase the number of output LEDs or use an external resistor ladder.
- Add calibration to compensate for reference voltage variations.

## 9 Conclusion

This project demonstrates a compact and synthesizable threshold-based DAC design using integer arithmetic in Verilog. It is suitable for FPGA demonstrations, classroom labs, and as a base for building a PWM or resistor-ladder DAC.

## References

- Xilinx Zynq-7000 SoC Technical Reference Manual
- FPGA4Fun, PWM DAC examples and sigma-delta implementations