## FPGA-based hysteresis current controller for three-phase inverter

TN120 | Posted on April 2, 2021 | Updated on May 7, 2025



Benoît STEINMANN Software Team Leader imperix • in



Julien ORSINGER
Power Applications Specialist
imperix • in

#### **Table of Contents**

- Software resources
- Implementation of the hysteresis current control
- FPGA logic of the hysteresis controller
- CPU implementation (using the imperix blockset for Simulink)
- Experimental results and validation of the hysteresis current control

This technical note provides an example of how a fast hysteresis current controller can be implemented, leveraging the possibility of <u>editing the FPGA firmware</u> for rapid control prototyping applications.

This example implements the direct current control of a three-phase passive load. It relies on manually-generated VHDL code. The automated generation of VHDL code is presented in <u>TN121</u>, which addresses the same application from Simulink and using <u>Matlab HDL Coder</u>.

Instructions on how to set up the FPGA development toolchain for <u>imperix controllers</u> are given in the dedicated note <u>PN168</u>. Quick-start recommendations regarding FPGA firmware customization can also be found in the <u>getting started with FPGA control</u>.

To find all FPGA-related notes, you can visit <u>FPGA development homepage</u>.

#### **Software resources**

<u>hysteresis\_current\_control\_simulinkDownload</u> <u>hysteresis\_current\_control\_FPGA\_vivado\_projectDownload</u>

The first file is the Simulink model used to generate the CPU code, and the second file contains a Vivado project that implements the FPGA logic described below.

### Implementation of the hysteresis current control

This implementation stems from the following choices:

- The current references are generated inside the CPU at a 40 kHz interval.
- The measured 3-phase currents are sampled at 400 kHz.

- The hysteresis control is implemented inside the programmable logic area (FPGA) by comparing the "fast" measured currents (400 kHz) with the "slow" generated references (40 kHz).
- The hysteresis comparators are 2-level comparators.
- The PWM output states are directly derived from the comparator state (see logic below).
- A counter prevents the PWM output from switching excessively rapidly between states (state change limiter). This ensures that the switching frequency of each phase is kept below a defined threshold.

Both CPU and FPGA tasks are illustrated below. The current references and the tolerance values are computed in the CPU and transferred to the FPGA with <u>SBO blocks</u>.



#### FPGA logic of the hysteresis controller

The following control logic is implemented in VHDL. Details on how to edit the firmware of the B-Box RCP are given in PN116.

#### Source code of dcc.vhd

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC STD.ALL;
entity DirectCurrentControl is
    Port (
        -- Measure
        meas_in : in std_logic_vector(15 downto 0) := (others => '0');
        -- Reference
        ref in : in std logic vector(15 downto 0) := (others => '0');
        -- Tolerence
        tol_in : in std_logic_vector(15 downto 0) := (others => '0');
        -- Delay
        delay_in : in std_logic_vector(15 downto 0) := (others => '0');
        -- Error = Measure - Reference
        err_out : out std_logic_vector(15 downto 0);
        -- Timing pulses
        adc_done_pulse_in : in std_logic;
        data valid pulse in : in std logic;
        -- PWM output
        pwm_out : out std_logic;
        -- Main clock running at 250 MHz
        clk_in : in std_logic
end DirectCurrentControl;
```

```
architecture impl of DirectCurrentControl is
    ATTRIBUTE X_INTERFACE_INFO : STRING;
    ATTRIBUTE X_INTERFACE_INFO of clk_in: SIGNAL is "xilinx.com:signal:clock:1.0 clk CLK";
    signal meas_ff1 : std_logic_vector(15 downto 0) := (others => '0');
    signal ref_ff1 : std_logic_vector(15 downto 0) := (others => '0');
    signal ref_ff2 : std_logic_vector(15 downto 0) := (others => '0');
    signal tol_ff1 : std_logic_vector(15 downto 0) := (others => '0');
    signal tol_ff2 : std_logic_vector(15 downto 0) := (others => '0');
    signal counter reg : unsigned(15 downto 0) := (others => '0');
    signal err : signed(15 downto 0);
    type type_fsm_state is (STATE_HIGH, STATE_LOW, STATE_H2L, STATE_L2H);
    signal fsm_state : type_fsm_state := STATE_LOW;
begin
    INPUT : process(clk_in)
    begin
        if rising edge(clk in) then
            -- new SBO data are available
            if data_valid_pulse_in = '1' then
                ref ff1 <= ref in;
                tol ff1 <= tol in;
            end if;
            -- new measure is available
            if adc_done_pulse_in = '1' then
                meas ff1 <= meas in;</pre>
                ref_ff2 <= ref_ff1;</pre>
                tol ff2 <= tol ff1;
            end if;
        end if;
    end process INPUT;
    err <= signed(meas_ff1) - signed(ref_ff2);</pre>
    err_out <= std_logic_vector(err);</pre>
    HYSTERESIS : process(clk_in)
    begin
        if rising edge(clk in) then
            case fsm_state is
                when STATE_HIGH =>
                     pwm_out <= '0';</pre>
                     counter_reg <= (others => '0');
                     if err < -signed(tol ff2) then
                         fsm_state <= STATE_H2L;</pre>
                     end if;
                when STATE H2L =>
                     pwm_out <= '1';</pre>
                     counter_reg <= counter_reg + 1;</pre>
                     if counter_reg >= unsigned(delay_in) then
                         fsm_state <= STATE_LOW;</pre>
                     end if;
```

when STATE\_LOW =>

```
pwm out <= '1';
                 counter_reg <= (others => '0');
                 if err > signed(tol_ff2) then
                      fsm_state <= STATE_L2H;</pre>
                 end if;
             when STATE_L2H =>
                 pwm_out <= '0';</pre>
                 counter_reg <= counter_reg + 1;</pre>
                 if counter reg >= unsigned(delay in) then
                      fsm state <= STATE HIGH;</pre>
                 end if;
             when others => null;
    end case;
  end if;
end process HYSTERESIS;
```

end impl;Code language: VHDL (vhdl)

The resulting logic is illustrated in the figure below. The current error is simply obtained by comparing the current reference with the actual measurement. Subsequently, as basic state machine controls the state transitions of the pwm\_out signal, enforcing a minimum delay to limit the converter switching frequency.



This subsystem is instantiated three times, one for each phase. Connections are made as shown in the next figure:

- meas in are connected to ADC reg 00, ADC reg 01 and ADC reg 02
- ref\_in are connected to SBO\_reg\_00, SBO\_reg\_01and SB0\_reg\_02
- tol\_in are connected to SBO\_reg\_03
- delay in are connected to SBO reg 04
- err\_out are connected to a SBI\_reg\_00, SBI\_reg\_01 and SBI\_02
- pwm\_out are connected to sb\_pwm[0], sb\_pwm[2] and sb\_pwm[4]



## **CPU implementation (using the imperix blockset for Simulink)**

The following configuration is used:

- CLOCK\_0 (main clock, controlling the sampling frequency): 400 kHz
- Interrupt postscaler: 5 (i.e. control interrupt executed at 40 kHz)
- SBO registers 00 to 02: current references (real-time registers)
- SBO register 03: hysteresis tolerance (real-time register)
- SBO register 04: state counter period (configuration register)
- SBI registers (read from FPGA) 00 to 02: current errors (for monitoring purpose only)
- SB\_PWM: PWM channels 0 to 2 configured with dual outputs with a 1 us dead time
- State change limiter: maximum switching frequency: 40 kHz (i.e. H2L and L2H states have a counter period of 3125. Counter period is 4 ns)

The CPU code is generated automatically from Simulink, using the following model:



The PWM output configuration is the following. Further information on sandbox PWM configuration is available on the <u>custom FPGA PWM</u> page.





This model generates 3-phase current references with a user-defined amplitude and a frequency of 50Hz. These references are then converted into unsigned int16 format to match the format of the FPGA registers. The conversion from Amperes to bits or inversely is done with the following operations. It considers the sensitivity of the current sensors and an input voltage range of ±10V with 16 bits ADCs. The intermediate conversion to int16 ensures that the signed float number is coded or interpreted correctly in 2's complement.



Data type conversion for CPU to FPGA transfer



Data type conversion for FPGA to CPU transfer

All probe variables are used for monitoring and debug purposes only.

# **Experimental results and validation of the hysteresis current control**

The subsequent results consider the following parameters:

- V<sub>DC</sub> = 70 V
- L<sub>load</sub> = 5 mH
- R<sub>load</sub> = 8 Ohm

The resulting current waveforms are shown below, for a current reference of 4 A peak and a hysteresis tolerance of  $\pm 0.3$  A and  $\pm 0.1$  A:



It can be clearly seen that reducing the hysteresis tolerance reduces the current ripples but increases the switching frequency. It also appears that the current error is not kept fully within the hysteresis band. This is unavoidable in digital implementations, for the following 2 reasons:

- A finite sampling frequency implies a maximum delay of Ts. In the present implementation, this
  corresponds to an additional current error of at most 1.5\*V<sub>DC</sub>\*T<sub>s</sub>/L<sub>load</sub> = 50 mA
- The ADCs have a non-zero conversion delay. In the present implementation, the ADC have a 2  $\mu$ s delay, which leads to an additional current error of at most 1.5\*V<sub>DC</sub>\*T<sub>delay</sub>/L<sub>load</sub> = 42 mA

All in all, this explains a current error of approx. 100 mA beyond the hysteresis bounds.

Back to FPGA development homepage