Power Measure

[中文]

Power Measure IC (Integrated Circuit) is an integrated circuit used for monitoring and managing power supply. It can monitor various parameters of the power supply in real time, such as voltage, current, and power, and provide this information for system use. Power Measure ICs are crucial in various applications, including computers, power management systems, consumer electronics, industrial control systems, and communication devices.

This example demonstrates how to use the BL0937, INA236, and BL0942 power measurement chips to detect electrical parameters such as voltage, current, active power, and energy consumption (BL0937 and BL0942 only).

Features

  • Measures voltage, current, active power, power factor, energy, and frequency.

  • Supports BL0937, INA236, and BL0942 power measurement chips.

  • Configurable chip selection via BL0937, INA236, and BL0942 macros.

  • Supports overcurrent, overvoltage, and undervoltage protection.

  • Energy detection is enabled for accurate readings (BL0937 and BL0942 only).

  • Real-time measurements for INA236, cached measurements for BL0937 and BL0942.

  • The BL0937 chip employs a hardware pulse counting method with a calibration factor, converting the pulse count into actual electrical parameters via this calibration factor.

BL0937 Calibration Process and Method

The following process can be used to calibrate the voltage (KU), current (KI), and power (KP) factors of the BL0937, improving measurement accuracy:

  • Preparation: A traceable multimeter/power meter, a stable AC source/adjustable AC source, and a known resistive load (such as a resistive load or 12W incandescent lamp).

  • Terms: - KU: Voltage calibration factor (to make the measured voltage close to the true voltage). - KI: Current calibration factor (to make the measured current close to the true current). - KP: Power calibration factor (to make the measured active power close to the true power).

Steps: 1. Fix the hardware parameters (shunt resistor, voltage divider resistor) and burn the example project, open the serial port log. 2. Voltage calibration (KU): At no load or light load, read voltage_meas, use the multimeter to get voltage_true, update:

KU_new = KU_old * (voltage_true / voltage_meas)
  1. Current calibration (KI): Connect the known resistive load, read current_meas, use the multimeter/power meter to calculate/read current_true, update:

    KI_new = KI_old * (current_true / current_meas)
    
  2. Power calibration (KP): At the same load, read power_meas, use the multimeter/power meter’s power_true as the reference, update:

    KP_new = KP_old * (power_true / power_meas)
    
  3. Repeat steps 2-4 several times until the relative error of voltage/current/power meets the requirements (e.g., ≤1%~2%).

In the code, configure the calibrated KU/KI/KP factors into power_measure_bl0937_config_t (the ki/ku/kp fields in the example).

More background and overall information can be found in: README.md.

How It Works

The app_main() function in main/power_measure_example.c demonstrates the handle-based factory API:

  1. (Optional) Initialize I2C bus (INA236 only):

    • Use i2c_bus_create() with I2C_MASTER_SCL_IO/I2C_MASTER_SDA_IO to create the I2C bus.

  2. Create a power measurement device:

    • Prepare common config power_measure_config_t (overcurrent/overvoltage/undervoltage thresholds, enable energy detection).

    • For BL0937, prepare power_measure_bl0937_config_t (CF/CF1/SEL pins, divider/sampling resistors, KI/KU/KP factors).

    • For INA236, prepare power_measure_ina236_config_t (I2C bus, address, alert config).

    • For BL0942, prepare power_measure_bl0942_config_t (UART/SPI config, shunt resistor, voltage divider ratio, device address).

    • Call power_measure_new_bl0937_device(), power_measure_new_ina236_device(), or power_measure_new_bl0942_device() to get a power_measure_handle_t.

  3. Fetch measurements periodically:

    • Use power_measure_get_voltage/current/active_power/power_factor() as needed;

    • power_measure_get_energy() returns valid data only on chips that support energy (e.g., BL0937 and BL0942);

    • power_measure_get_line_freq() returns valid data only on BL0942 chip;

    • The example logs values every 1 second.

  4. Error handling:

    • If any API returns a non-ESP_OK value, log the error with ESP_LOGE.

  5. Resource cleanup:

    • Release the device with power_measure_delete(handle);

    • For INA236, also release the I2C bus with i2c_bus_delete(&i2c_bus).

Code Snippets

BL0937 (GPIO)

#include "power_measure.h"
#include "power_measure_bl0937.h"

power_measure_config_t common = {
    .overcurrent = 2000,
    .overvoltage = 250,
    .undervoltage = 180,
    .enable_energy_detection = true,
};

power_measure_bl0937_config_t bl = {
    .sel_gpio = GPIO_NUM_4,
    .cf1_gpio = GPIO_NUM_7,
    .cf_gpio = GPIO_NUM_3,
    .pin_mode = 0,
    .sampling_resistor = 0.001f,
    .divider_resistor = 1981.0f,
    .ki = 2.75f, .ku = 0.65f, .kp = 15.0f,
};

power_measure_handle_t h;
ESP_ERROR_CHECK(power_measure_new_bl0937_device(&common, &bl, &h));

float u,i,p,e;
power_measure_get_voltage(h, &u);
power_measure_get_current(h, &i);
power_measure_get_active_power(h, &p);
power_measure_get_energy(h, &e);

power_measure_delete(h);

INA236 (I2C)

#include "power_measure.h"
#include "power_measure_ina236.h"
#include "i2c_bus.h"

i2c_config_t conf = {
    .mode = I2C_MODE_MASTER,
    .sda_io_num = I2C_MASTER_SDA_IO, // GPIO_NUM_20
    .sda_pullup_en = GPIO_PULLUP_ENABLE,
    .scl_io_num = I2C_MASTER_SCL_IO, // GPIO_NUM_13
    .scl_pullup_en = GPIO_PULLUP_ENABLE,
    .master.clk_speed = 100000,
};
i2c_bus_handle_t bus = i2c_bus_create(I2C_NUM_0, &conf);

power_measure_config_t common = {
    .overcurrent = 15,
    .overvoltage = 260,
    .undervoltage = 180,
    .enable_energy_detection = false,
};

power_measure_ina236_config_t ina = {
    .i2c_bus = bus,
    .i2c_addr = 0x41,
    .alert_en = false,
    .alert_pin = -1,
    .alert_cb = NULL,
};

power_measure_handle_t h;
ESP_ERROR_CHECK(power_measure_new_ina236_device(&common, &ina, &h));

float u,i,p;
power_measure_get_voltage(h, &u);
power_measure_get_current(h, &i);
power_measure_get_active_power(h, &p);

power_measure_delete(h);
i2c_bus_delete(&bus);

BL0942 (UART/SPI)

#include "power_measure.h"
#include "power_measure_bl0942.h"

power_measure_config_t common = {
    .overcurrent = 15,
    .overvoltage = 260,
    .undervoltage = 180,
    .enable_energy_detection = true,
};

power_measure_bl0942_config_t bl0942 = {
    .addr = 0,
    .shunt_resistor = 0.001f,
    .divider_ratio = 3760.0f,
    .use_spi = false,
    .uart = {
        .uart_num = UART_NUM_1,
        .tx_io = GPIO_NUM_6,
        .rx_io = GPIO_NUM_7,
        .sel_io = -1,
        .baud = 4800,
    },
    .spi = {
        .spi_host = SPI2_HOST,
        .mosi_io = -1,
        .miso_io = -1,
        .sclk_io = -1,
        .cs_io = -1,
    },
};

// power_measure_bl0942_config_t bl0942 = {
//     .addr = 0,
//     .shunt_resistor = 0.001f,
//     .divider_ratio = 3760.0f,
//     .use_spi = true,
//     .uart = {
//         .uart_num = UART_NUM_1,
//         .tx_io = -1,
//         .rx_io = -1,
//         .sel_io = -1,
//         .baud = 4800,
//     },
//     .spi = {
//         .spi_host = SPI2_HOST,
//         .mosi_io = GPIO_NUM_11,
//         .miso_io = GPIO_NUM_12,
//         .sclk_io = GPIO_NUM_10,
//         .cs_io = GPIO_NUM_13,
//     },
// };

power_measure_handle_t h;
ESP_ERROR_CHECK(power_measure_new_bl0942_device(&common, &bl0942, &h));

float u, i, p, pf, e, f;
power_measure_get_voltage(h, &u);
power_measure_get_current(h, &i);
power_measure_get_active_power(h, &p);
power_measure_get_power_factor(h, &pf);
power_measure_get_energy(h, &e);
power_measure_get_line_freq(h, &f);

power_measure_delete(h);

Troubleshooting

BL0937 Issues:

  1. Failed Initialization : If the initialization fails, ensure that all GPIO pins are correctly defined and connected to the BL0937 chip.

  2. Measurement Failures : If the measurements fail (e.g., voltage, current), verify that the BL0937 chip is properly powered and communicating with your ESP32 series chips.

INA236 Issues:

  1. I2C Bus Initialization Failed : Check that the I2C pins (SDA/SCL) are correctly connected and not conflicting with other peripherals.

  2. INA236 Not Detected : Verify the I2C address (default 0x41) and ensure the chip is properly powered.

  3. Measurement Failures : Check I2C communication and ensure the INA236 chip is functioning correctly.

  4. No Real-time Data : INA236 provides real-time measurements, so if you see cached values, check the power_measure component implementation.

BL0942 Issues:

  1. UART Initialization Failed : Check that UART pins (TX/RX) are correctly connected and verify baud rate settings (default 4800).

  2. SPI Initialization Failed : Check that SPI pins (MOSI/MISO/SCLK/CS) are correctly connected and not conflicting with other peripherals.

  3. BL0942 Not Detected : Verify device address (default 0) and ensure the chip is properly powered.

  4. Communication Failures : Check UART/SPI communication parameters and confirm interface mode configuration.

  5. Abnormal Measurements : Check shunt resistor and voltage divider ratio configuration to match actual hardware.

  6. Watchdog Reset : If watchdog reset occurs, check hardware connections and communication configuration.

Adapted Products

Power Measurement Chips

Name

Function

Vendor

Datasheet

HAL

BL0937

detect electrical parameters such as voltage, current, active power, and energy consumption

BELLING

BL0937 Datasheet

INA236

precision power monitor with I2C interface for voltage, current, and power measurement

TI

INA236 Datasheet

BL0942

Single-phase power measurement IC with UART/SPI interface for voltage, current, power, power factor, energy, and frequency measurement

BELLING

BL0942 Datasheet

API Reference

The following API implements hardware abstraction for power measure. Users can directly call this layer of code to write sensor applications.

Header File

Functions

esp_err_t power_measure_delete(power_measure_handle_t handle)

Delete a power measurement device.

Parameters

handle – Power measure device handle

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid handle

esp_err_t power_measure_get_voltage(power_measure_handle_t handle, float *voltage)

Get current voltage (V)

Parameters
  • handle – Power measure device handle

  • voltage – Pointer to store the voltage value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

esp_err_t power_measure_get_current(power_measure_handle_t handle, float *current)

Get current current (A)

Parameters
  • handle – Power measure device handle

  • current – Pointer to store the current value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

esp_err_t power_measure_get_active_power(power_measure_handle_t handle, float *power)

Get current active power (W)

Parameters
  • handle – Power measure device handle

  • power – Pointer to store the power value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

esp_err_t power_measure_get_power_factor(power_measure_handle_t handle, float *power_factor)

Get power factor.

Parameters
  • handle – Power measure device handle

  • power_factor – Pointer to store the power factor value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_get_energy(power_measure_handle_t handle, float *energy)

Get energy consumption (kWh)

Parameters
  • handle – Power measure device handle

  • energy – Pointer to store the energy value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_reset_energy_calculation(power_measure_handle_t handle)

Reset energy calculation.

Parameters

handle – Power measure device handle

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid handle

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_get_apparent_power(power_measure_handle_t handle, float *apparent_power)

Get apparent power measurement.

Parameters
  • handle – Power measure device handle

  • apparent_power – Pointer to store the apparent power value in volt-amperes (VA)

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_calibrate_voltage(power_measure_handle_t handle, float expected_voltage)

Dynamic runtime calibration for voltage (BL0937 only)

Parameters
  • handle – Power measure device handle

  • expected_voltage – The known reference voltage value in volts

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_calibrate_current(power_measure_handle_t handle, float expected_current)

Dynamic runtime calibration for current (BL0937 only)

Parameters
  • handle – Power measure device handle

  • expected_current – The known reference current value in amperes

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_calibrate_power(power_measure_handle_t handle, float expected_power)

Dynamic runtime calibration for power (BL0937 only)

Parameters
  • handle – Power measure device handle

  • expected_power – The known reference power value in watts

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_get_voltage_multiplier(power_measure_handle_t handle, float *multiplier)

Get voltage multiplier (BL0937 only, advanced users)

Parameters
  • handle – Power measure device handle

  • multiplier – Pointer to store the voltage multiplier value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_get_current_multiplier(power_measure_handle_t handle, float *multiplier)

Get current multiplier (BL0937 only, advanced users)

Parameters
  • handle – Power measure device handle

  • multiplier – Pointer to store the current multiplier value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature

esp_err_t power_measure_get_power_multiplier(power_measure_handle_t handle, float *multiplier)

Get power multiplier (BL0937 only, advanced users)

Parameters
  • handle – Power measure device handle

  • multiplier – Pointer to store the power multiplier value

Returns

  • ESP_OK: Success

  • ESP_ERR_INVALID_ARG: Invalid arguments

  • ESP_ERR_INVALID_STATE: Device not initialized

  • ESP_ERR_NOT_SUPPORTED: Chip doesn’t support this feature POWER_MEASURE

Structures

struct power_measure_config_t

Common power measure configuration.

Public Members

uint16_t overcurrent

Overcurrent threshold

uint16_t overvoltage

Overvoltage threshold

uint16_t undervoltage

Undervoltage threshold

bool enable_energy_detection

Flag to enable energy detection

Type Definitions

typedef struct power_measure_dev_t *power_measure_handle_t

Power measure device handle.