From: Bernhard Beschow <[email protected]>

Add emulation of the i.MX8MP System Reset Controller (SRC).
The SRC manages reset control for various subsystems on the i.MX8MP SoC.
SRC registers are used to start/stop/poll the M7 core, hence needed for
AMP boot.

Signed-off-by: Bernhard Beschow <[email protected]>
---
 docs/system/arm/imx8m.rst    |   1 +
 hw/arm/Kconfig               |   1 +
 hw/arm/fsl-imx8mp.c          |  10 +
 hw/misc/Kconfig              |   3 +
 hw/misc/imx8mp_src.c         | 431 +++++++++++++++++++++++++++++++++++
 hw/misc/meson.build          |   1 +
 hw/misc/trace-events         |   5 +
 include/hw/arm/fsl-imx8mp.h  |   2 +
 include/hw/misc/imx8mp_src.h |  32 +++
 9 files changed, 486 insertions(+)
 create mode 100644 hw/misc/imx8mp_src.c
 create mode 100644 include/hw/misc/imx8mp_src.h

diff --git a/docs/system/arm/imx8m.rst b/docs/system/arm/imx8m.rst
index 1e110bc810..49c100c4b1 100644
--- a/docs/system/arm/imx8m.rst
+++ b/docs/system/arm/imx8m.rst
@@ -27,6 +27,7 @@ following devices:
  * Clock Tree
  * General Power Controller (GPC)
  * General Purpose Register (GPR)
+ * System Reset Controller (SRC)
 
 Boot options
 ------------
diff --git a/hw/arm/Kconfig b/hw/arm/Kconfig
index 9e4b71264a..8e89e30c45 100644
--- a/hw/arm/Kconfig
+++ b/hw/arm/Kconfig
@@ -602,6 +602,7 @@ config FSL_IMX8MP
     select FSL_IMX8MP_CCM
     select FSL_IMX8MP_GPC
     select FSL_IMX8MP_GPR
+    select FSL_IMX8MP_SRC
     select IMX
     select IMX_FEC
     select IMX_I2C
diff --git a/hw/arm/fsl-imx8mp.c b/hw/arm/fsl-imx8mp.c
index c4aafbdc1c..168fcb0334 100644
--- a/hw/arm/fsl-imx8mp.c
+++ b/hw/arm/fsl-imx8mp.c
@@ -208,6 +208,8 @@ static void fsl_imx8mp_init(Object *obj)
 
     object_initialize_child(obj, "gpr", &s->gpr, TYPE_IMX8MP_GPR);
 
+    object_initialize_child(obj, "src", &s->src, TYPE_IMX8MP_SRC);
+
     for (i = 0; i < FSL_IMX8MP_NUM_UARTS; i++) {
         g_autofree char *name = g_strdup_printf("uart%d", i + 1);
         object_initialize_child(obj, name, &s->uart[i], TYPE_IMX_SERIAL);
@@ -426,6 +428,13 @@ static void fsl_imx8mp_realize(DeviceState *dev, Error 
**errp)
                            qdev_get_gpio_in(gicdev, serial_table[i].irq));
     }
 
+    /* SRC */
+    if (!sysbus_realize(SYS_BUS_DEVICE(&s->src), errp)) {
+        return;
+    }
+    sysbus_mmio_map(SYS_BUS_DEVICE(&s->src), 0,
+                    fsl_imx8mp_memmap[FSL_IMX8MP_SRC].addr);
+
     /* GPC */
     if (!sysbus_realize(SYS_BUS_DEVICE(&s->gpc), errp)) {
         return;
@@ -716,6 +725,7 @@ static void fsl_imx8mp_realize(DeviceState *dev, Error 
**errp)
         case FSL_IMX8MP_PCIE_PHY1:
         case FSL_IMX8MP_RAM:
         case FSL_IMX8MP_SNVS_HP:
+        case FSL_IMX8MP_SRC:
         case FSL_IMX8MP_UART1 ... FSL_IMX8MP_UART4:
         case FSL_IMX8MP_USB1 ... FSL_IMX8MP_USB2:
         case FSL_IMX8MP_USDHC1 ... FSL_IMX8MP_USDHC3:
diff --git a/hw/misc/Kconfig b/hw/misc/Kconfig
index 7464788d82..185bd6f62a 100644
--- a/hw/misc/Kconfig
+++ b/hw/misc/Kconfig
@@ -107,6 +107,9 @@ config FSL_IMX8MP_GPC
 config FSL_IMX8MP_GPR
     bool
 
+config FSL_IMX8MP_SRC
+    bool
+
 config STM32_RCC
     bool
 
diff --git a/hw/misc/imx8mp_src.c b/hw/misc/imx8mp_src.c
new file mode 100644
index 0000000000..f339a7a088
--- /dev/null
+++ b/hw/misc/imx8mp_src.c
@@ -0,0 +1,431 @@
+/*
+ * i.MX 8M Plus System Reset Controller
+ *
+ * Copyright (c) 2025 Bernhard Beschow <[email protected]>
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "qemu/osdep.h"
+#include "hw/misc/imx8mp_src.h"
+#include "hw/core/cpu.h"
+#include "hw/core/qdev-properties.h"
+#include "hw/core/registerfields.h"
+#include "target/arm/arm-powerctl.h"
+#include "migration/vmstate.h"
+#include "qemu/bitops.h"
+#include "qemu/log.h"
+#include "qemu/main-loop.h"
+#include "trace.h"
+
+REG32(SRC_SCR, 0x0000)
+
+REG32(SRC_A53RCR0, 0x0004)
+    FIELD(SRC_A53RCR0, CORE_RESET3, 7, 1)
+    FIELD(SRC_A53RCR0, CORE_RESET2, 6, 1)
+    FIELD(SRC_A53RCR0, CORE_RESET1, 5, 1)
+    FIELD(SRC_A53RCR0, CORE_RESET0, 4, 1)
+
+REG32(SRC_A53RCR1, 0x0008)
+    FIELD(SRC_A53RCR1, CORE3_ENABLE, 3, 1)
+    FIELD(SRC_A53RCR1, CORE2_ENABLE, 2, 1)
+    FIELD(SRC_A53RCR1, CORE1_ENABLE, 1, 1)
+    FIELD(SRC_A53RCR1, CORE0_ENABLE, 0, 1)
+
+REG32(SRC_M7RCR, 0x000c)
+REG32(SRC_SUPERMIX_RCR, 0x0018)
+REG32(SRC_AUDIOMIX_RCR, 0x001c)
+REG32(SRC_USBPHY1_RCR, 0x0020)
+REG32(SRC_USBPHY2_RCR, 0x0024)
+REG32(SRC_MLMIX_RCR, 0x0028)
+REG32(SRC_PCIEPHY_RCR, 0x002c)
+REG32(SRC_HDMI_RCR, 0x0030)
+REG32(SRC_MEDIA_RCR, 0x0034)
+REG32(SRC_GPU2D_RCR, 0x0038)
+REG32(SRC_GPU3D_RCR, 0x003c)
+REG32(SRC_GPU_RCR, 0x0040)
+REG32(SRC_VPU_RCR, 0x0044)
+REG32(SRC_VPU_G1_RCR, 0x0048)
+REG32(SRC_VPU_G2_RCR, 0x004c)
+REG32(SRC_VPUVC8KE_RCR, 0x0050)
+REG32(SRC_NOC_RCR, 0x0054)
+REG32(SRC_SBMR1, 0x0058)
+REG32(SRC_SRSR, 0x005c)
+REG32(SRC_SISR, 0x0068)
+REG32(SRC_SIMR, 0x006c)
+
+REG32(SRC_SBMR2, 0x0070)
+    FIELD(SRC_SBMR2, IPP_BOOT_MODE, 24, 4)
+
+REG32(SRC_GPR1, 0x0074)
+REG32(SRC_GPR2, 0x0078)
+REG32(SRC_GPR3, 0x007c)
+REG32(SRC_GPR4, 0x0080)
+REG32(SRC_GPR5, 0x0084)
+REG32(SRC_GPR6, 0x0088)
+REG32(SRC_GPR7, 0x008c)
+REG32(SRC_GPR8, 0x0090)
+REG32(SRC_GPR9, 0x0094)
+REG32(SRC_GPR10, 0x0098)
+REG32(SRC_DDRC_RCR, 0x1000)
+REG32(SRC_HDMIPHY_RCR, 0x1008)
+REG32(SRC_MIPIPHY1_RCR, 0x100c)
+REG32(SRC_MIPIPHY2_RCR, 0x1010)
+REG32(SRC_HSIO_RCR, 0x1014)
+REG32(SRC_MEDIAISPDWP_RCR, 0x1018)
+
+static const char *fsl_imx8mp_src_reg_name(uint32_t reg)
+{
+    switch (reg) {
+    case R_SRC_SCR:
+        return " (SRC_SCR)";
+    case R_SRC_A53RCR0:
+        return " (SRC_A53RCR0)";
+    case R_SRC_A53RCR1:
+        return " (SRC_A53RCR1)";
+    case R_SRC_M7RCR:
+        return " (SRC_M7RCR)";
+    case R_SRC_SUPERMIX_RCR:
+        return " (SRC_SUPERMIX_RCR)";
+    case R_SRC_AUDIOMIX_RCR:
+        return " (SRC_AUDIOMIX_RCR)";
+    case R_SRC_USBPHY1_RCR:
+        return " (SRC_USBPHY1_RCR)";
+    case R_SRC_USBPHY2_RCR:
+        return " (SRC_USBPHY2_RCR)";
+    case R_SRC_MLMIX_RCR:
+        return " (SRC_MLMIX_RCR)";
+    case R_SRC_PCIEPHY_RCR:
+        return " (SRC_PCIEPHY_RCR)";
+    case R_SRC_HDMI_RCR:
+        return " (SRC_HDMI_RCR)";
+    case R_SRC_MEDIA_RCR:
+        return " (SRC_MEDIA_RCR)";
+    case R_SRC_GPU2D_RCR:
+        return " (SRC_GPU2D_RCR)";
+    case R_SRC_GPU3D_RCR:
+        return " (SRC_GPU3D_RCR)";
+    case R_SRC_GPU_RCR:
+        return " (SRC_GPU_RCR)";
+    case R_SRC_VPU_RCR:
+        return " (SRC_VPU_RCR)";
+    case R_SRC_VPU_G1_RCR:
+        return " (SRC_VPU_G1_RCR)";
+    case R_SRC_VPU_G2_RCR:
+        return " (SRC_VPU_G2_RCR)";
+    case R_SRC_VPUVC8KE_RCR:
+        return " (SRC_VPUVC8KE_RCR)";
+    case R_SRC_NOC_RCR:
+        return " (SRC_NOC_RCR)";
+    case R_SRC_SBMR1:
+        return " (SRC_SBMR1)";
+    case R_SRC_SRSR:
+        return " (SRC_SRSR)";
+    case R_SRC_SISR:
+        return " (SRC_SISR)";
+    case R_SRC_SIMR:
+        return " (SRC_SIMR)";
+    case R_SRC_SBMR2:
+        return " (SRC_SBMR2)";
+    case R_SRC_GPR1:
+        return " (SRC_GPR1)";
+    case R_SRC_GPR2:
+        return " (SRC_GPR2)";
+    case R_SRC_GPR3:
+        return " (SRC_GPR3)";
+    case R_SRC_GPR4:
+        return " (SRC_GPR4)";
+    case R_SRC_GPR5:
+        return " (SRC_GPR5)";
+    case R_SRC_GPR6:
+        return " (SRC_GPR6)";
+    case R_SRC_GPR7:
+        return " (SRC_GPR7)";
+    case R_SRC_GPR8:
+        return " (SRC_GPR8)";
+    case R_SRC_GPR9:
+        return " (SRC_GPR9)";
+    case R_SRC_GPR10:
+        return " (SRC_GPR10)";
+    case R_SRC_DDRC_RCR:
+        return " (SRC_DDRC_RCR)";
+    case R_SRC_HDMIPHY_RCR:
+        return " (SRC_HDMIPHY_RCR)";
+    case R_SRC_MIPIPHY1_RCR:
+        return " (SRC_MIPIPHY1_RCR)";
+    case R_SRC_MIPIPHY2_RCR:
+        return " (SRC_MIPIPHY2_RCR)";
+    case R_SRC_HSIO_RCR:
+        return " (SRC_HSIO_RCR)";
+    case R_SRC_MEDIAISPDWP_RCR:
+        return " (SRC_MEDIAISPDWP_RCR)";
+    default:
+        return " (reserved)";
+    }
+}
+
+static uint64_t fsl_imx8mp_src_read(void *opaque, hwaddr offset,
+                                    unsigned size)
+{
+    FslImx8mpSrcState *s = opaque;
+    const uint32_t reg = offset / 4;
+    uint32_t value = 0;
+
+    switch (reg) {
+    default:
+        if (reg < FSL_IMX8MP_SRC_NUM_REGS) {
+            value = s->regs[reg];
+        }
+        qemu_log_mask(LOG_UNIMP, "[%s]%s: Unimplemented register at offset 0x%"
+                      HWADDR_PRIx "\n", TYPE_IMX8MP_SRC, __func__,
+                      offset);
+        break;
+    }
+
+    trace_fsl_imx8mp_src_read(offset, fsl_imx8mp_src_reg_name(reg), value);
+
+    return value;
+}
+
+/*
+ * The reset is asynchronous so we need to defer clearing the reset bit until
+ * the work is completed.
+ */
+
+struct FslImx8mpSrcResetInfo {
+    FslImx8mpSrcState *s;
+    int reset_bit;
+};
+
+
+static void imx8mp_src_reset(DeviceState *dev)
+{
+    FslImx8mpSrcState *s = IMX8MP_SRC(dev);
+
+    memset(s->regs, 0, sizeof(s->regs));
+
+    /* Primary A53 core enabled by default */
+    s->regs[R_SRC_A53RCR1] = BIT(0);
+
+    /*
+     * Default CM7 STOP state for Linux imx-rproc MMIO mode detection.
+     * Historically your minimal SRC-mmio stub seeded offset 0x0C with 0xA.
+     * In this model, offset 0x0C is R_SRC_M7RCR.
+     */
+    s->regs[R_SRC_M7RCR] = 0x0000000A;
+
+    /* Boot mode field (kept from previous realize-time init) */
+    s->regs[R_SRC_SBMR2] = FIELD_DP32(s->regs[R_SRC_SBMR2], SRC_SBMR2,
+                                      IPP_BOOT_MODE, s->boot_mode);
+}
+
+
+static void fsl_imx8mp_src_clear_reset_bit(CPUState *cpu, run_on_cpu_data data)
+{
+    struct FslImx8mpSrcResetInfo *ri = data.host_ptr;
+    FslImx8mpSrcState *s = ri->s;
+
+    assert(bql_locked());
+
+    s->regs[R_SRC_A53RCR0] = deposit32(s->regs[R_SRC_A53RCR0],
+                                       ri->reset_bit, 1, 0);
+    
trace_fsl_imx8mp_src_clear_reset_bit(fsl_imx8mp_src_reg_name(R_SRC_A53RCR0),
+                                         s->regs[R_SRC_A53RCR0]);
+
+    g_free(ri);
+}
+
+static void fsl_imx8mp_src_defer_clear_reset_bit(FslImx8mpSrcState *s,
+                                                  int cpuid,
+                                                  unsigned long reset_shift)
+{
+    struct FslImx8mpSrcResetInfo *ri;
+    CPUState *cpu = arm_get_cpu_by_id(cpuid);
+
+    if (!cpu) {
+        return;
+    }
+
+    ri = g_new(struct FslImx8mpSrcResetInfo, 1);
+    ri->s = s;
+    ri->reset_bit = reset_shift;
+
+    async_run_on_cpu(cpu, fsl_imx8mp_src_clear_reset_bit,
+                     RUN_ON_CPU_HOST_PTR(ri));
+}
+
+static void fsl_imx8mp_src_write(void *opaque, hwaddr offset, uint64_t value,
+                                 unsigned size)
+{
+    FslImx8mpSrcState *s = opaque;
+    const uint32_t reg = offset / 4;
+    unsigned long change_mask;
+
+    change_mask = s->regs[reg] ^ (uint32_t)value;
+
+    switch (reg) {
+    case R_SRC_A53RCR0:
+        if (FIELD_EX32(change_mask, SRC_A53RCR0, CORE_RESET0)) {
+            arm_reset_cpu(0);
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 0,
+                 R_SRC_A53RCR0_CORE_RESET0_SHIFT);
+        }
+        if (FIELD_EX32(change_mask, SRC_A53RCR0, CORE_RESET1)) {
+            arm_reset_cpu(1);
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 1,
+                 R_SRC_A53RCR0_CORE_RESET1_SHIFT);
+        }
+        if (FIELD_EX32(change_mask, SRC_A53RCR0, CORE_RESET2)) {
+            arm_reset_cpu(2);
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 2,
+                 R_SRC_A53RCR0_CORE_RESET2_SHIFT);
+        }
+        if (FIELD_EX32(change_mask, SRC_A53RCR0, CORE_RESET3)) {
+            arm_reset_cpu(3);
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 3,
+                 R_SRC_A53RCR0_CORE_RESET3_SHIFT);
+        }
+        s->regs[reg] = value;
+        break;
+    case R_SRC_A53RCR1:
+        if (FIELD_EX32(change_mask, SRC_A53RCR1, CORE3_ENABLE)) {
+            if (FIELD_EX32(value, SRC_A53RCR1, CORE3_ENABLE)) {
+                /* CORE 3 is brought up */
+                arm_set_cpu_on(3, s->regs[R_SRC_GPR8] << 2, 0, 3, true);
+            } else {
+                /* CORE 3 is shut down */
+                arm_set_cpu_off(3);
+            }
+            /* We clear the reset bit as the processor changed state */
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 3, 
R_SRC_A53RCR0_CORE_RESET3_SHIFT);
+        }
+        if (FIELD_EX32(change_mask, SRC_A53RCR1, CORE2_ENABLE)) {
+            if (FIELD_EX32(value, SRC_A53RCR1, CORE2_ENABLE)) {
+                /* CORE 2 is brought up */
+                arm_set_cpu_on(2, s->regs[R_SRC_GPR6] << 2, 0, 3, true);
+            } else {
+                /* CORE 2 is shut down */
+                arm_set_cpu_off(2);
+            }
+            /* We clear the reset bit as the processor changed state */
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 2, 
R_SRC_A53RCR0_CORE_RESET2_SHIFT);
+        }
+        if (FIELD_EX32(change_mask, SRC_A53RCR1, CORE1_ENABLE)) {
+            if (FIELD_EX32(value, SRC_A53RCR1, CORE1_ENABLE)) {
+                /* CORE 1 is brought up */
+                arm_set_cpu_on(1, s->regs[R_SRC_GPR4] << 2, 0, 3, true);
+            } else {
+                /* CORE 1 is shut down */
+                arm_set_cpu_off(1);
+            }
+            /* We clear the reset bit as the processor changed state */
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 1, 
R_SRC_A53RCR0_CORE_RESET1_SHIFT);
+        }
+        if (FIELD_EX32(change_mask, SRC_A53RCR1, CORE0_ENABLE)) {
+            if (FIELD_EX32(value, SRC_A53RCR1, CORE0_ENABLE)) {
+                /* CORE 1 is brought up */
+                arm_set_cpu_on(0, s->regs[R_SRC_GPR2] << 2, 0, 3, true);
+            } else {
+                /* CORE 1 is shut down */
+                arm_set_cpu_off(0);
+            }
+            /* We clear the reset bit as the processor changed state */
+            fsl_imx8mp_src_defer_clear_reset_bit(s, 0, 
R_SRC_A53RCR0_CORE_RESET0_SHIFT);
+        }
+        s->regs[reg] = value;
+        break;
+    default:
+        if (reg < FSL_IMX8MP_SRC_NUM_REGS) {
+            s->regs[reg] = value;
+        }
+        qemu_log_mask(LOG_UNIMP, "[%s]%s: Unimplemented register at offset 0x%"
+                      HWADDR_PRIx "\n", TYPE_IMX8MP_SRC, __func__,
+                      offset);
+        break;
+    }
+
+    trace_fsl_imx8mp_src_write(offset, fsl_imx8mp_src_reg_name(reg), value);
+}
+
+static const struct MemoryRegionOps imx8mp_src_ops = {
+    .read = fsl_imx8mp_src_read,
+    .write = fsl_imx8mp_src_write,
+    .endianness = DEVICE_LITTLE_ENDIAN,
+    .valid = {
+        /*
+         * Our device would not work correctly if the guest was doing
+         * unaligned access. This might not be a limitation on the real
+         * device but in practice there is no reason for a guest to access
+         * this device unaligned.
+         */
+        .min_access_size = 4,
+        .max_access_size = 4,
+        .unaligned = false,
+    },
+};
+
+void imx8mp_src_start_cpu(FslImx8mpSrcState *s, int cpuid)
+{
+    switch (cpuid) {
+    case 0:
+        arm_set_cpu_on(0, s->regs[R_SRC_GPR2] << 2, 0, 3, true);
+        break;
+    case 1:
+        arm_set_cpu_on(1, s->regs[R_SRC_GPR4] << 2, 0, 3, true);
+        break;
+    case 2:
+        arm_set_cpu_on(2, s->regs[R_SRC_GPR6] << 2, 0, 3, true);
+        break;
+    case 3:
+        arm_set_cpu_on(3, s->regs[R_SRC_GPR8] << 2, 0, 3, true);
+        break;
+    default:
+        g_assert_not_reached();
+        break;
+    }
+}
+
+static void imx8mp_src_realize(DeviceState *dev, Error **errp)
+{
+    FslImx8mpSrcState *s = IMX8MP_SRC(dev);
+
+    memory_region_init_io(&s->iomem, OBJECT(dev), &imx8mp_src_ops, s,
+                          TYPE_IMX8MP_SRC, 0x1000);
+    sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->iomem);
+}
+
+static const VMStateDescription imx8mp_src_vmstate = {
+    .name = TYPE_IMX8MP_SRC,
+    .version_id = 1,
+    .minimum_version_id = 1,
+    .fields = (const VMStateField[]) {
+        VMSTATE_UINT32_ARRAY(regs, FslImx8mpSrcState, FSL_IMX8MP_SRC_NUM_REGS),
+        VMSTATE_END_OF_LIST()
+    },
+};
+
+static const Property imx8mp_src_properties[] = {
+    DEFINE_PROP_UINT8("boot-mode", FslImx8mpSrcState, boot_mode, 0),
+};
+
+static void imx8mp_src_class_init(ObjectClass *klass, const void *data)
+{
+    DeviceClass *dc = DEVICE_CLASS(klass);
+
+    device_class_set_legacy_reset(dc, imx8mp_src_reset);
+    dc->realize = imx8mp_src_realize;
+    dc->vmsd = &imx8mp_src_vmstate;
+    device_class_set_props(dc, imx8mp_src_properties);
+    dc->desc = "i.MX 8M Plus System Reset Controller";
+}
+
+static const TypeInfo imx8mp_src_types[] = {
+    {
+        .name          = TYPE_IMX8MP_SRC,
+        .parent        = TYPE_SYS_BUS_DEVICE,
+        .instance_size = sizeof(FslImx8mpSrcState),
+        .class_init    = imx8mp_src_class_init,
+    },
+};
+
+DEFINE_TYPES(imx8mp_src_types)
diff --git a/hw/misc/meson.build b/hw/misc/meson.build
index 67dd1a98e5..174aed40b6 100644
--- a/hw/misc/meson.build
+++ b/hw/misc/meson.build
@@ -61,6 +61,7 @@ system_ss.add(when: 'CONFIG_FSL_IMX8MP_ANALOG', if_true: 
files('imx8mp_analog.c'
 system_ss.add(when: 'CONFIG_FSL_IMX8MP_CCM', if_true: files('imx8mp_ccm.c'))
 system_ss.add(when: 'CONFIG_FSL_IMX8MP_GPC', if_true: files('imx8mp_gpc.c'))
 system_ss.add(when: 'CONFIG_FSL_IMX8MP_GPR', if_true: files('imx8mp_gpr.c'))
+system_ss.add(when: 'CONFIG_FSL_IMX8MP_SRC', if_true: files('imx8mp_src.c'))
 system_ss.add(when: 'CONFIG_IMX', if_true: files(
   'imx25_ccm.c',
   'imx31_ccm.c',
diff --git a/hw/misc/trace-events b/hw/misc/trace-events
index b88accc437..080e79787c 100644
--- a/hw/misc/trace-events
+++ b/hw/misc/trace-events
@@ -129,6 +129,11 @@ imx7_gpr_write(uint64_t offset, uint64_t value) "addr 
0x%08" PRIx64 "value 0x%08
 imx7_snvs_read(uint64_t offset, uint64_t value, unsigned size) "i.MX SNVS 
read: offset 0x%08" PRIx64 " value 0x%08" PRIx64 " size %u"
 imx7_snvs_write(uint64_t offset, uint64_t value, unsigned size) "i.MX SNVS 
write: offset 0x%08" PRIx64 " value 0x%08" PRIx64 " size %u"
 
+# imx8mp_src.c
+fsl_imx8mp_src_read(uint64_t offset, const char *reg_name, uint32_t value) 
"[0x%" PRIx64 "%s] <- 0x%" PRIx32
+fsl_imx8mp_src_write(uint64_t offset, const char *reg_name, uint64_t value) 
"[0x%" PRIx64 "%s] <- 0x%" PRIx64
+fsl_imx8mp_src_clear_reset_bit(const char *reg_name, uint32_t value) "reg[%s] 
<= 0x%" PRIx32
+
 # mos6522.c
 mos6522_set_counter(int index, unsigned int val) "T%d.counter=%d"
 mos6522_get_next_irq_time(uint16_t latch, int64_t d, int64_t delta) "latch=%d 
counter=0x%"PRIx64 " delta_next=0x%"PRIx64
diff --git a/include/hw/arm/fsl-imx8mp.h b/include/hw/arm/fsl-imx8mp.h
index b3aca1f692..c6c133cc05 100644
--- a/include/hw/arm/fsl-imx8mp.h
+++ b/include/hw/arm/fsl-imx8mp.h
@@ -19,6 +19,7 @@
 #include "hw/misc/imx8mp_ccm.h"
 #include "hw/misc/imx8mp_gpc.h"
 #include "hw/misc/imx8mp_gpr.h"
+#include "hw/misc/imx8mp_src.h"
 #include "hw/net/imx_fec.h"
 #include "hw/core/or-irq.h"
 #include "hw/pci-host/designware.h"
@@ -64,6 +65,7 @@ struct FslImx8mpState {
     IMX8MPAnalogState  analog;
     IMX7SNVSState      snvs;
     IMXSPIState        spi[FSL_IMX8MP_NUM_ECSPIS];
+    FslImx8mpSrcState  src;
     IMXI2CState        i2c[FSL_IMX8MP_NUM_I2CS];
     IMXSerialState     uart[FSL_IMX8MP_NUM_UARTS];
     IMXFECState        enet;
diff --git a/include/hw/misc/imx8mp_src.h b/include/hw/misc/imx8mp_src.h
new file mode 100644
index 0000000000..9905d217b8
--- /dev/null
+++ b/include/hw/misc/imx8mp_src.h
@@ -0,0 +1,32 @@
+/*
+ * i.MX 8M Plus System Reset Controller
+ *
+ * Copyright (c) 2025 Bernhard Beschow <[email protected]>
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef FSL_IMX8MP_SRC_H
+#define FSL_IMX8MP_SRC_H
+
+#include "hw/core/sysbus.h"
+#include "system/memory.h"
+#include "qom/object.h"
+
+#define TYPE_IMX8MP_SRC "fsl-imx8mp-src"
+OBJECT_DECLARE_SIMPLE_TYPE(FslImx8mpSrcState, IMX8MP_SRC)
+
+#define FSL_IMX8MP_SRC_NUM_REGS (0x100 / 4)
+
+struct FslImx8mpSrcState {
+    SysBusDevice parent_obj;
+
+    MemoryRegion iomem;
+
+    uint32_t regs[FSL_IMX8MP_SRC_NUM_REGS];
+    uint8_t boot_mode;
+};
+
+void imx8mp_src_start_cpu(FslImx8mpSrcState *s, int cpuid);
+
+#endif /* FSL_IMX8MP_SRC_H */
-- 
2.34.1


Reply via email to