Skip to content

Modify the firmware

Manuel Bleichenbacher edited this page Jan 8, 2023 · 4 revisions

Since this is an open-source project, it can be modified. One of the use cases is to change what the ZY12PD does at startup, how it reacts to button presses and how it uses the LED. So the USB PD communincation is reused, but the "user interface" is changed.

To do that, copy or clone the entire project and modify main.cpp.

Minimal example

Below is a minimal example. It selects a fixed voltage (9V) once the power source announces its capabilities. When the desired voltage is active, the LED is green. Otherwise, it's red.

This illustrates the minimal setup that needs to be in place:

  • The hardware abstraction layer (HAL) and the power sink need to be initialized.
  • An event callback function must be registered.
  • When the source announces its capabilities (supported voltages and currents), a voltage must be requested almost immediately.
  • The poll() functions of the hardware abstraction layer and the power sink must be call frequently so they can handle the events they are responsible for.

main.cpp

#include "pd_sink.h"

using namespace usb_pd;

// voltage in mV
const int desired_voltage = 9000;

mcu_hal usb_pd::hal;
pd_sink power_sink;


// called when the USB PD controller triggers an event
void sink_callback(callback_event event) {

    if (event == callback_event::source_caps_changed) {
        // source has announced its capabilities; sink must request voltage
        int res = power_sink.request_power(desired_voltage);

        if (res == -1) {
            // desired voltage is not available, request 5V instead
            power_sink.request_power(5000);
        }
    }
}

int main() {
    // initialize the board
    hal.init();
    hal.set_led(color::red);

    // initialize PD power sink
    power_sink.set_event_callback(sink_callback);
    power_sink.init();

    // loop and poll
    while (true) {
        hal.poll();
        power_sink.poll();

        // set LED: green for desired voltage, red otherwise
        hal.set_led(power_sink.active_voltage == desired_voltage
            ? color::green : color::red);
    }
}

Working with programmable power supply

This more advanced example is for power supply with PPS capability: They allow to select any voltage is a given range. The protocol supports a resolution of 10mV.

When the source announces its capabilities, this example will locate the PPS capability. The capability describes the minimum voltage (min_voltage) and the maximum voltage (voltage).

Initially, the lowest voltage will be requested. Every press of the button increases the voltage by 100mV. Once the maximum voltage has been exceed, it starts over at the lowest voltage.

The power supply might be able to deliver down to 3.3V. However, this is not sufficient for the LDO on the ZY12PDN board. It will cut out between 3.5V and 3.9V. So the minimum voltage is at least 4V.

#include "pd_sink.h"
#include <algorithm>

using namespace usb_pd;

// PPS might be able to deliver less than 5V, but
// below 4V, the LDO on the board might cut out.
constexpr int cutout_voltage = 4000;

mcu_hal usb_pd::hal;
pd_sink power_sink;

int voltage = 5000;
const source_capability* pps_cap = nullptr;

// Find the PPS capability
// (take the last one if multiple ones are available)
const source_capability* find_pps_cap() {
    int index = 0;
    for (int i = 0; i < power_sink.num_source_caps; i++) {
        if (power_sink.source_caps[i].supply_type == pd_supply_type::pps)
            index = i;
    }
    return &power_sink.source_caps[index];
}

// called when the USB PD controller triggers an event
void sink_callback(callback_event event) {

    if (event == callback_event::source_caps_changed) {
        // source has announced its capabilities; sink must request voltage
        pps_cap = find_pps_cap();
        voltage = std::max((int)pps_cap->min_voltage, cutout_voltage);
        power_sink.request_power(voltage);
    }
}

int main() {
    // initialize the board
    hal.init();
    hal.set_led(color::red);

    // initialize PD power sink
    power_sink.set_event_callback(sink_callback);
    power_sink.init();

    // loop and poll
    while (true) {
        hal.poll();
        power_sink.poll();

        // If button has been pressed, increase voltage by 100mV
        if (hal.has_button_been_pressed() && pps_cap != nullptr) {
            voltage += 100;
            if (voltage > pps_cap->voltage)
                voltage = std::max((int)pps_cap->min_voltage, cutout_voltage);
            power_sink.request_power(voltage);
        }

        // set LED: green for desired voltage, red otherwise
        hal.set_led(power_sink.active_voltage == voltage ? color::green : color::red);
    }
}
Clone this wiki locally