# **High-Level Synthesis for FPGA developments** TN142 | Posted on June 15, 2021 | Updated on May 7, 2025 Benoît STEINMANN Software Team Leader imperix • in #### **Table of Contents** - Integrating HLS designs in the FPGA - Description of the design - o CPU-side implementation using Simulink - FPGA-side implementation using Vivado **High-level synthesis (HLS)** tools greatly facilitate the implementation of complex power electronics controller algorithms in FPGA. Indeed HLS tools allow the user to work at a higher level of abstraction. For instance, the user can use <u>Xilinx Vitis HLS</u> to develop FPGA modules using **C/C++** or the <u>Model Composer</u> plug-in for **Simulink** to use graphical programming instead. This page shows how IPs generated using high-level synthesis tools can be integrated into the FPGA of an imperix power controller. To this end, the example of a **PI-based current controller for a buck converter** is used to illustrate all the required steps. Power converter control FPGA To find all FPGA-related notes, you can visit FPGA development homepage. ## Integrating HLS designs in the FPGA ## **Description of the design** The image below shows the example that will be implemented on this page. It is a Pl-based current controller for a buck converter, based on the algorithm presented on the <u>Pl controller implementation for current control</u> technical note. This example uses the following resources - the FPGA control starter template from the getting started with FPGA guide - the **PWM modulator IP** from the <u>FPGA PWM modulator</u> example - the high-level synthesis PI-based current control IP from either - the C++ implementation presented in the Xilinx Vitis HLS guide - or the Simulink implemention presented in the Model Composer guide The **axis interface** provides the inputs of the current control algorithm in form of **AXI4-Stream** ports. The following ports are used: - CPU2FPGA\_00 for the current reference I1\_ref (32-bit single-precision) - CPU2FPGA 01 for the parameter Kp (32-bit single-precision) - CPU2FPGA\_02 for the parameter Ki (32-bit single-precision) - ADC 00 for the measured current I1 (16-bit signed integer) - ADC\_01 for the measured output voltage of the converter Vout (16-bit signed integer) - ADC\_02 for the measured input voltage of the converter Vint (16-bit signed integer) - Ts for the sampling period in nanoseconds (32-bit unsigned integer) Aside from AXI4-Stream data, the current control IP also uses the ports: - CLOCK period for the PWM period in ticks (16-bit unsigned) - nReset ctrl to reset the PI when the controller is not in OPERATING state Using these signals, the HLS IP computes a 16-bit unsigned duty\_cycle\_ticks that is forwarded to the PWM IP. And finally, the PWM IP uses the sb\_pwm driver to output the PWM signals to optical fibers of the B-Box RCP controller. The PWM IP and the SB-PWM driver are further documented on the <u>FPGA PWM modulator</u> page. The ADC values provided by the starter template are the raw result from the ADC chips. They are multiplied by a *gain* inside the HLS IP to obtain physical values. An example of *gain* computation is available on the <u>ADC block</u> help page. ### **CPU-side implementation using Simulink** The CPU-side model is quite simple, as the control algorithm runs entirely in the FPGA. The CPU code provides the current reference and Kp/Ki parameters, operates the PI reset signal, and configures the PWM outputs. The *single2sbo* MATLAB Function blocks are used to map the current reference **II\_ref** and the **Kp**, **Ki** parameter to the CPU2FPGA ports. This **nReset\_ctrl** signal is used to keep the PI integrator at reset when the controller is not in OPERATING state. As documented in <u>Getting started with FPGA</u>, this reset signal is controlled using *SBO\_63*. To obtain the desired behavior, we'll simply connect the reset output of a <u>Core state</u> block to *SBO\_63*. And finally, the <u>SB-PWM</u> block is used to activate the output PWM **channel 0 (CH0)** (lane #0 and lane #1). The output is configured as **Dual (PWM\_H + PWM\_L)** with a **deadtime** of 1 µs. This configuration expects a PWM signal coming to **sb\_pwm[0]** input of the *imperix firmware* IP and will automatically generate the complementary signals with the configured deadtime. The ADC blocks are only used to retrieve the analog input signals at the CPU level for real-time monitoring. They do not affect the closed-loop control behavior. # FPGA-side implementation using Vivado The **TN142\_vivado\_design.pdf** file below shows the full Vivado FPGA design. Here are the step-by-step instructions to reproduce it. 1. Create an FPGA control implementation starter template by following the <u>Getting started with FPGA control implementation</u>. Add the PWM IP (from the <u>custom PWM in FPGA</u> page) and <u>current control IP</u> (from the <u>Xilinx Vitis HLS</u> guide or the <u>Model Composer</u> guide) into your Vivado project. In the screenshots of this example, we'll use the IPs generated using System Generator and Vitis HLS, respectively. To read the duty\_cycle\_ticks only when duty\_cycle\_ticks\_ap\_vld is '1', the *RAM-based Shift Register* IP is used. With the configuration shown in the screenshot below, this block adds one register stage that acts as a buffer. It keeps the last computed duty cycle until a new value has been computed. When a new value is available, it replaces the old one. - 3. Add a **Constant** IP to set all the 31 unused sb\_pwm outputs to '0'. Set its *Const Width* to 31 and its *Const Val* to 0. - 4. Add a **Concat** IP. It will serve to concat the pwm output of the PWM IP with the zeros of the Constant IP. 5. Add a **Constant** IP to set to set the update rate. '0' = single rate, '1' = double rate. ### 6. Connect the clock signals as below: #### 7. Connect the AXI4-Streams - M\_AXIS\_CPU2FPGA\_00 to Il\_ref\_V - M\_AXIS\_CPU2FPGA\_01 to Kp\_V - M\_AXIS\_CPU2FPGA\_02 to Ki\_V - M\_AXIS\_ADC\_00 to Il\_raw\_v - M\_AXIS\_ADC\_01 to voltage Vout\_raw\_V - M\_AXIS\_ADC\_02 to Vint\_raw\_V - M\_AXIS\_Ts to Ts\_V 8. The provided *Delay Counter* VHDL module (delay\_counter.vhd) measures the elapsed time between two signals and outputs a time in nanoseconds, encoded as a uint32. In this design, the *delay counter* modules are used purely for debugging purposes. As shown in the image below, one is used to measure the *FPGA processing delay*, which is the delay between the adc\_done\_pulse and the duty\_cycle\_ticks\_ap\_vld. Another module is used to measure the *FPGA cycle delay* by measuring the delay between the sampling\_pulse and the duty\_cycle\_ticks\_ap\_vld. More information on what these delays represent are available on the <u>discrete control delay</u> product node. 9. Connect the nReset\_ctrl signal to ap\_rst\_n. - 10. And finally connect the clk signals to clk\_250\_mhz. - 11. Click **Generate bitstream**. It will launch the synthesis, implementation, and bitstream generation - 12. Once the bitstream generation is completed, click on **File** → **Export** → **Export** Bitstream File... to save the bitstream somewhere on your computer. Back to FPGA development homepage