GPIO 初始化

某些驱动程序依赖于静态 GPIO 引脚或引脚多路复用配置, 因主板而异在这些情况下,通常需要将 GPIO 保持在 配置详细信息,以确保从属驱动程序 可移植性这可以通过让板驱动程序通过 元数据和初始化步骤列表传递给 GPIO 驱动程序。请参阅下文,了解 示例。

在板驱动程序中配置 GPIO 元数据

假设 SDMMC 驱动程序要求引脚配置 某些替代功能并推动力量。为此,板驱动程序需要 创建一个 fuchsia.hardware.gpioimpl.InitStep 对象列表,用于指定 要进行的 GPIO 协议调用,以及进行这些调用的 GPIO 索引:

#include <fidl/fuchsia.hardware.gpioimpl/cpp/wire.h>

// Helper lambda to simplify the following code.
auto sdmmc_gpio = [&](uint64_t alt_function, uint64_t drive_strength_ua) {
  return fuchsia_hardware_gpioimpl::wire::InitOptions::Builder(arena)
      .alt_function(alt_function)
      .drive_strength_ua(drive_strength_ua)
      .Build();
};

const fuchsia_hardware_gpioimpl::wire::InitStep init_steps[] = {
    {SDMMC_GPIO_0, sdmmc_gpio(SDMMC_GPIO_0_ALT_FUNCTION, 4000)},
    {SDMMC_GPIO_1, sdmmc_gpio(SDMMC_GPIO_1_ALT_FUNCTION, 4000)},
    {SDMMC_GPIO_2, sdmmc_gpio(SDMMC_GPIO_2_ALT_FUNCTION, 4000)},
    {SDMMC_GPIO_3, sdmmc_gpio(SDMMC_GPIO_3_ALT_FUNCTION, 4000)},
    {SDMMC_GPIO_4, sdmmc_gpio(SDMMC_GPIO_4_ALT_FUNCTION, 4000)},
    {SDMMC_GPIO_5, sdmmc_gpio(SDMMC_GPIO_5_ALT_FUNCTION, 4000)},
};

然后,系统会将此列表作为元数据提供给 GPIO 驱动程序:


fuchsia_hardware_gpioimpl::wire::InitMetadata metadata;
metadata.steps = fidl::VectorView<fuchsia_hardware_gpioimpl::wire::InitStep>::FromExternal(
    init_steps, std::size(init_steps));

fit::result encoded = fidl::Persist(metadata);
if (!encoded.is_ok()) {
  return encoded.error_value().status();
}

const std::vector<fpbus::Metadata> gpio_metadata{
    {{
        .type = DEVICE_METADATA_GPIO_INIT_STEPS,
        .data = std::move(encoded.value()),
    }},
    // Other metadata goes here.
};

const fpbus::Node gpio_dev = []() {
  fpbus::Node dev = {};
  dev.name() = "gpio";
  dev.vid() = PDEV_VID_SOME_VENDOR;
  dev.pid() = PDEV_PID_SOME_PRODUCT;
  dev.did() = PDEV_DID_SOME_GPIO_DEVICE;
  dev.mmio() = gpio_mmios;
  dev.irq() = gpio_irqs;
  dev.metadata() = gpio_metadata;
  return dev;
}();

// Add the node here.

使用绑定规则创建对 GPIO 配置的依赖项

现在,GPIO 驱动程序已正确配置这些引脚,SDMMC 驱动程序 必须使该配置依赖于此配置这是 通过在主板中向 SDMMC 设备添加额外的 Fragment 来实现 驱动程序绑定文件:

using fuchsia.gpio;

// Other nodes go here.

node "gpio-init" {
  fuchsia.BIND_INIT_STEP == fuchsia.gpio.BIND_INIT_STEP.GPIO;
  fuchsia.BIND_GPIO_CONTROLLER = 1;
}

之后,GPIO 驱动程序会添加 fuchsia.gpio.BIND_INIT_STEP.GPIO 设备, 它就已完成初始化步骤的处理。任意数量的子元素 绑定到此设备,以确保之前已配置其 GPIO 启动。

GPIO 驱动程序如何处理配置错误

如果 GPIO 驱动程序在处理初始化步骤时遇到错误, 它会继续设置其余的 GPIO,但不会添加 init 设备。 这是为了确保将尽可能多的引脚置于已知状态,并且 驱动程序不会尝试使用可能 配置有误。同样,如果 GPIO 驱动程序不添加 init 设备, 具有或无法解析初始化元数据。

在以下两种情况下,建议使用 GPIO 初始化步骤:

  • 驱动程序需要特定于板或平台的静态 GPIO 配置。

上拉/下拉、Alt 函数和驱动强度设置通常 特定于主板或平台,因此可使用 GPIO init 在主板中对其进行配置 驱动程序。如果遇到以下情况,也建议设置 GPIO 输出值 所有开发板上的值都不同(或不要求)。

  • 多个驱动程序依赖于一个 GPIO 的静态配置。

如果多个驱动程序依赖于一个 GPIO 的静态配置,那么 配置应由 GPIO init 完成。这样可以防止多个驾驶员 必须对单个 GPIO 进行潜在冲突的调用。

不得将 GPIO 初始化步骤用于更改 在运行时使用驱动程序

GPIO 初始化选项

如需查看初始化选项的完整列表及其说明,请参阅 fuchsia.hardware.pinimpl FIDL 规范