Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

drivers: reset: Add generic reset MMIO driver #87501

Merged
merged 1 commit into from
Apr 2, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions drivers/reset/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -12,3 +12,4 @@ zephyr_library_sources_ifdef(CONFIG_RESET_INTEL_SOCFPGA reset_intel_socfpga.c)
zephyr_library_sources_ifdef(CONFIG_RESET_NPCX reset_npcx.c)
zephyr_library_sources_ifdef(CONFIG_RESET_NXP_SYSCON reset_lpc_syscon.c)
zephyr_library_sources_ifdef(CONFIG_RESET_NXP_RSTCTL reset_nxp_rstctl.c)
zephyr_library_sources_ifdef(CONFIG_RESET_MMIO reset_mmio.c)
1 change: 1 addition & 0 deletions drivers/reset/Kconfig
Original file line number Diff line number Diff line change
Expand Up @@ -36,5 +36,6 @@ rsource "Kconfig.intel_socfpga"
rsource "Kconfig.npcx"
rsource "Kconfig.lpc_syscon"
rsource "Kconfig.nxp_rstctl"
rsource "Kconfig.mmio"

endif # RESET
11 changes: 11 additions & 0 deletions drivers/reset/Kconfig.mmio
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# MMIO reset driver configuration options

# Copyright (c) 2025 Google LLC
# SPDX-License-Identifier: Apache-2.0

config RESET_MMIO
bool "Generic MMIO reset driver"
default y
depends on DT_HAS_RESET_MMIO_ENABLED
help
This option enables the generic MMIO reset driver. This is meant for IPs with a single memory mapped reset bit required to take them out of reset.
111 changes: 111 additions & 0 deletions drivers/reset/reset_mmio.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
/* Copyright (c) 2025 Google LLC.
*
* SPDX-License-Identifier: Apache-2.0
*/

#include <zephyr/device.h>
#include <zephyr/devicetree.h>
#include <zephyr/drivers/reset.h>
#include <zephyr/kernel.h>

struct reset_mmio_dev_config {
uint32_t base;
uint8_t num_resets;
bool active_low;
};

struct reset_mmio_dev_data {
struct k_spinlock lock;
};

static inline int reset_mmio_status(const struct device *dev, uint32_t id, uint8_t *status)
{
const struct reset_mmio_dev_config *config = dev->config;
uint32_t value;

if (id >= config->num_resets) {
return -EINVAL;
}

value = sys_read32(config->base);
*status = FIELD_GET(BIT(id), value);

/* If active low, invert the logic */
if (config->active_low) {
*status = !(*status);
}

return 0;
}

static inline int reset_mmio_update(const struct device *dev, uint32_t id, uint8_t assert)
{
const struct reset_mmio_dev_config *config = dev->config;
struct reset_mmio_dev_data *data = dev->data;
uint32_t value;
bool set;

if (id >= config->num_resets) {
return -EINVAL;
}

/* If active low, invert the logic */
set = config->active_low ? !assert : assert;

K_SPINLOCK(&data->lock) {
value = sys_read32(config->base);

if (set) {
value |= BIT(id);
} else {
value &= ~BIT(id);
}

sys_write32(value, config->base);
}

return 0;
}

static int reset_mmio_line_assert(const struct device *dev, uint32_t id)
{
return reset_mmio_update(dev, id, 1);
}

static int reset_mmio_line_deassert(const struct device *dev, uint32_t id)
{
return reset_mmio_update(dev, id, 0);
}

static int reset_mmio_line_toggle(const struct device *dev, uint32_t id)
{
uint8_t reset_status = 0;
int status;

status = reset_mmio_status(dev, id, &reset_status);
if (status) {
return status;
}

return reset_mmio_update(dev, id, !reset_status);
}

static DEVICE_API(reset, reset_mmio_driver_api) = {
.status = reset_mmio_status,
.line_assert = reset_mmio_line_assert,
.line_deassert = reset_mmio_line_deassert,
.line_toggle = reset_mmio_line_toggle,
};

#define DT_DRV_COMPAT reset_mmio
#define RESET_MMIO_INIT(n) \
BUILD_ASSERT(DT_INST_PROP(n, num_resets) > 0 && DT_INST_PROP(n, num_resets) < 32, \
"num-resets needs to be in [1, 31]."); \
static const struct reset_mmio_dev_config reset_mmio_dev_config_##n = { \
.base = DT_INST_REG_ADDR(n), \
.num_resets = DT_INST_PROP(n, num_resets), \
.active_low = DT_INST_PROP(n, active_low)}; \
static struct reset_mmio_dev_data reset_mmio_dev_data_##n; \
DEVICE_DT_INST_DEFINE(n, NULL, NULL, &reset_mmio_dev_data_##n, &reset_mmio_dev_config_##n, \
POST_KERNEL, CONFIG_RESET_INIT_PRIORITY, &reset_mmio_driver_api);
DT_INST_FOREACH_STATUS_OKAY(RESET_MMIO_INIT)
23 changes: 23 additions & 0 deletions dts/bindings/reset/reset-mmio.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
# Copyright (c) 2025 Google, LLC
# SPDX-License-Identifier: Apache-2.0

description: |
Generic MMIO Reset driver for devices with a single memory mapped reset bit
required to take them out of reset.

compatible: "reset-mmio"

include: [base.yaml]

properties:
reg:
required: true
num-resets:
type: int
required: true
description: |
Number of resets controlled by the register.
Can be in the range [1, 31].
active-low:
description: Reset is active in low state.
type: boolean
9 changes: 9 additions & 0 deletions tests/drivers/reset/mmio/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
# Copyright (c) 2025 Google LLC
# SPDX-License-Identifier: Apache-2.0

cmake_minimum_required(VERSION 3.20.0)
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})

project(reset_mmio)

target_sources(app PRIVATE src/main.c)
14 changes: 14 additions & 0 deletions tests/drivers/reset/mmio/boards/qemu_cortex_m3.overlay
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
/ {
reset0: reset@20000004 {
compatible = "reset-mmio";
reg = <0x20000004 0x4>;
num-resets = <16>;
};

reset1: reset@20000008 {
compatible = "reset-mmio";
reg = <0x20000008 0x4>;
num-resets = <16>;
active-low;
};
};
3 changes: 3 additions & 0 deletions tests/drivers/reset/mmio/prj.conf
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
CONFIG_ZTEST=y
CONFIG_RESET=y
CONFIG_RESET_MMIO=y
134 changes: 134 additions & 0 deletions tests/drivers/reset/mmio/src/main.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,134 @@
/*
* Copyright (c) 2025 Google LLC.
*
* SPDX-License-Identifier: Apache-2.0
*/

#include <zephyr/device.h>
#include <zephyr/drivers/reset.h>
#include <zephyr/kernel.h>
#include <zephyr/ztest.h>

ZTEST_SUITE(reset_mmio_tests, NULL, NULL, NULL, NULL, NULL);

#define RESET_MAX_NUM 16

void check_status(const struct device *dev, uint32_t base, uint32_t id, bool expected_state,
bool active_low)
{
uint8_t actual_state;

zassert_ok(reset_status(dev, id, &actual_state), "Failed getting reset state");
zassert_equal(actual_state, expected_state,
"reset state %u doesn't match expected state %u", actual_state,
expected_state);
zassert_equal(FIELD_GET(BIT(id), sys_read32(base)), active_low ^ expected_state);
}

/* Tests that the reset driver assert functionality is correct for active
* low devices.
*/
ZTEST(reset_mmio_tests, test_reset_mmio_assert_active_low)
{
const struct device *dev = DEVICE_DT_GET(DT_NODELABEL(reset1));
uint32_t base = DT_REG_ADDR(DT_NODELABEL(reset1));
bool active_low = true;
uint8_t i;

for (i = 0; i < RESET_MAX_NUM; i++) {
reset_line_deassert(dev, i);
check_status(dev, base, i, false, active_low);
/* Check idempotency */
reset_line_deassert(dev, i);
check_status(dev, base, i, false, active_low);

/* Check ressserting resets */
reset_line_assert(dev, i);
check_status(dev, base, i, true, active_low);
/* Check idempotency */
reset_line_assert(dev, i);
check_status(dev, base, i, true, active_low);
}
}

/* Tests that the reset driver assert functionality is correct for active
* high devices.
*/
ZTEST(reset_mmio_tests, test_reset_mmio_assert_active_high)
{
const struct device *dev = DEVICE_DT_GET(DT_NODELABEL(reset0));
uint32_t base = DT_REG_ADDR(DT_NODELABEL(reset0));
bool active_low = false;
uint8_t i;

for (i = 0; i < RESET_MAX_NUM; i++) {
reset_line_deassert(dev, i);
check_status(dev, base, i, false, active_low);
/* Check idempotency */
reset_line_deassert(dev, i);
check_status(dev, base, i, false, active_low);

/* Check ressserting resets */
reset_line_assert(dev, i);
check_status(dev, base, i, true, active_low);
/* Check idempotency */
reset_line_assert(dev, i);
check_status(dev, base, i, true, active_low);
}
}

/* Tests that the reset driver toggle functionality is correct for active
* low devices
*/
ZTEST(reset_mmio_tests, test_reset_mmio_toggle_active_low)
{
const struct device *dev = DEVICE_DT_GET(DT_NODELABEL(reset1));
uint32_t base = DT_REG_ADDR(DT_NODELABEL(reset1));
bool active_low = true;
uint8_t i;

for (i = 0; i < RESET_MAX_NUM; i++) {
/* Begin by making sure the reset is asserted */
reset_line_assert(dev, i);
check_status(dev, base, i, true, active_low);
reset_line_toggle(dev, i);
check_status(dev, base, i, false, active_low);
reset_line_toggle(dev, i);
check_status(dev, base, i, true, active_low);
}
}

/* Tests that the reset driver toggle functionality is correct for active
* high devices
*/
ZTEST(reset_mmio_tests, test_reset_mmio_toggle_active_high)
{
const struct device *dev = DEVICE_DT_GET(DT_NODELABEL(reset0));
uint32_t base = DT_REG_ADDR(DT_NODELABEL(reset0));
bool active_low = false;
uint8_t i;

for (i = 0; i < RESET_MAX_NUM; i++) {
/* Begin by making sure the reset is asserted */
reset_line_assert(dev, i);
check_status(dev, base, i, true, active_low);
reset_line_toggle(dev, i);
check_status(dev, base, i, false, active_low);
reset_line_toggle(dev, i);
check_status(dev, base, i, true, active_low);
}
}

/* Tests that the reset driver rejects out of bounds bits */
ZTEST(reset_mmio_tests, test_reset_mmio_oob)
{
const struct device *dev = DEVICE_DT_GET(DT_NODELABEL(reset0));
uint8_t i, status;

for (i = RESET_MAX_NUM; i < 32; i++) {
zassert_not_ok(reset_line_assert(dev, i));
zassert_not_ok(reset_line_deassert(dev, i));
zassert_not_ok(reset_status(dev, i, &status));
zassert_not_ok(reset_line_toggle(dev, i));
}
}
7 changes: 7 additions & 0 deletions tests/drivers/reset/mmio/testcase.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
# Copyright (c) 2025 Google LLC
# SPDX-License-Identifier: Apache-2.0

tests:
drivers.reset.mmio:
platform_allow:
- qemu_cortex_m3