Hi Cédric

> Subject: Re: [PATCH v1 3/4] hw/arm/aspeed: Introduce AST1040 A0 SoC model
> 
> On 5/21/26 08:52, Jamin Lin wrote:
> > The AST1040 is based on an ARM Cortex-M4F CPU core. Since QEMU
> > currently does not provide Cortex-M4F support, use the existing
> > Cortex-M4 CPU model as a temporary replacement.
> >
> > This initial implementation provides the basic infrastructure required
> > to boot firmware and run a minimal firmware shell,
> > including:
> >
> > - ARM Cortex-M4 CPU integration
> > - NVIC interrupt controller support
> > - Internal SDRAM and SRAM memory regions
> > - SCU integration
> > - UART devices and interrupt wiring
> >
> > AST1040 SCU behavior is compatible with the AST2700 SCUIO model, so
> > reuse the existing AST2700 SCUIO implementation directly instead of
> > introducing another identical SCU model. This reduces duplicate code
> > and helps minimize long-term codebase maintenance.
> >
> > Several peripherals are currently modeled as unimplemented devices and
> > can be added incrementally in future updates.
> >
> > Signed-off-by: Jamin Lin <[email protected]>
> > ---
> >   hw/arm/aspeed_ast1040.c | 246
> ++++++++++++++++++++++++++++++++++++++++
> >   hw/arm/meson.build      |   3 +-
> >   2 files changed, 248 insertions(+), 1 deletion(-)
> >   create mode 100644 hw/arm/aspeed_ast1040.c
> >
> > diff --git a/hw/arm/aspeed_ast1040.c b/hw/arm/aspeed_ast1040.c new
> > file mode 100644 index 0000000000..615f9c4e15
> > --- /dev/null
> > +++ b/hw/arm/aspeed_ast1040.c
> > @@ -0,0 +1,246 @@
> > +/*
> > + * ASPEED AST1040 SoC
> > + *
> > + * Copyright (C) 2026 ASPEED Technology Inc.
> > + *
> > + * SPDX-License-Identifier: GPL-2.0-or-later
> > + *
> > + */
> > +
> > +#include "qemu/osdep.h"
> > +#include "qapi/error.h"
> > +#include "system/address-spaces.h"
> > +#include "system/system.h"
> > +#include "hw/core/qdev-clock.h"
> > +#include "hw/misc/unimp.h"
> > +#include "hw/arm/aspeed_soc.h"
> > +
> > +static const hwaddr aspeed_soc_ast1040_memmap[] = {
> > +    [ASPEED_DEV_SDRAM]     = 0x00000000,
> > +    [ASPEED_DEV_FMC]       = 0x74000000,
> > +    [ASPEED_DEV_SPI0]      = 0x74010000,
> > +    [ASPEED_DEV_SPI1]      = 0x74020000,
> > +    [ASPEED_DEV_PWM]       = 0x740C0000,
> > +    [ASPEED_DEV_UDC]       = 0x74120000,
> > +    [ASPEED_DEV_SRAM]      = 0x74B80000,
> > +    [ASPEED_DEV_ADC]       = 0x74C00000,
> > +    [ASPEED_DEV_JTAG0]     = 0x74C01000,
> > +    [ASPEED_DEV_SCU]       = 0x74C02000,
> > +    [ASPEED_DEV_ESPI]      = 0x74C05000,
> > +    [ASPEED_DEV_JTAG1]     = 0x74C09000,
> > +    [ASPEED_DEV_GPIO]      = 0x74C0B000,
> > +    [ASPEED_DEV_SGPIOM0]   = 0x74C0C000,
> > +    [ASPEED_DEV_SGPIOM1]   = 0x74C0D000,> +    [ASPEED_DEV_I2C]
> = 0x74C0F000,
> > +    [ASPEED_DEV_I3C]       = 0x74C20000,
> > +    [ASPEED_DEV_UART0]     = 0x74C33000,
> > +    [ASPEED_DEV_UART1]     = 0x74C33100,
> > +    [ASPEED_DEV_UART2]     = 0x74C33200,
> > +    [ASPEED_DEV_UART3]     = 0x74C33300,
> > +    [ASPEED_DEV_UART4]     = 0x74C33400,
> > +    [ASPEED_DEV_UART5]     = 0x74C33500,
> > +    [ASPEED_DEV_UART6]     = 0x74C33600,
> > +    [ASPEED_DEV_UART7]     = 0x74C33700,
> > +    [ASPEED_DEV_UART8]     = 0x74C33800,
> > +    [ASPEED_DEV_UART9]     = 0x74C33900,
> > +    [ASPEED_DEV_UART10]    = 0x74C33A00,
> > +    [ASPEED_DEV_UART11]    = 0x74C33B00,
> > +    [ASPEED_DEV_UART12]    = 0x74C33C00,
> > +    [ASPEED_DEV_WDT]       = 0x74C37000,
> > +    [ASPEED_DEV_TIMER1]    = 0x74C3A000,
> > +};
> > +
> > +static const int aspeed_soc_ast1040_irqmap[] = {
> > +    [ASPEED_DEV_ESPI]      = 10,
> > +    [ASPEED_DEV_I2C]       = 64, /* 64 ~ 77 */
> > +    [ASPEED_DEV_ADC]       = 80,
> > +    [ASPEED_DEV_GPIO]      = 82,
> > +    [ASPEED_DEV_SGPIOM0]   = 85,
> > +    [ASPEED_DEV_TIMER1]    = 92,
> > +    [ASPEED_DEV_I3C]       = 96, /* 96 ~ 103 */
> > +    [ASPEED_DEV_WDT]       = 112,
> > +    [ASPEED_DEV_FMC]       = 121,
> > +    [ASPEED_DEV_SPI0]      = 122,
> > +    [ASPEED_DEV_SPI1]      = 123,
> > +    [ASPEED_DEV_PWM]       = 125,
> > +    [ASPEED_DEV_UART0]     = 135,
> > +    [ASPEED_DEV_UART1]     = 136,
> > +    [ASPEED_DEV_UART2]     = 137,
> > +    [ASPEED_DEV_UART3]     = 138,
> > +    [ASPEED_DEV_UART4]     = 139,
> > +    [ASPEED_DEV_UART5]     = 140,
> > +    [ASPEED_DEV_UART6]     = 141,
> > +    [ASPEED_DEV_UART7]     = 142,
> > +    [ASPEED_DEV_UART8]     = 143,
> > +    [ASPEED_DEV_UART9]     = 144,
> > +    [ASPEED_DEV_UART10]    = 145,
> > +    [ASPEED_DEV_UART11]    = 146,
> > +    [ASPEED_DEV_UART12]    = 147,
> > +    [ASPEED_DEV_JTAG0]     = 162,
> > +};
> > +
> > +static qemu_irq aspeed_soc_ast1040_get_irq(AspeedSoCState *s, int
> > +dev) {
> > +    Aspeed10x0SoCState *a = ASPEED10X0_SOC(s);
> > +    AspeedSoCClass *sc = ASPEED_SOC_GET_CLASS(s);
> > +
> > +    return qdev_get_gpio_in(DEVICE(&a->armv7m), sc->irqmap[dev]); }
> > +
> > +static void aspeed_soc_ast1040_init(Object *obj) {
> > +    Aspeed10x0SoCState *a = ASPEED10X0_SOC(obj);
> > +    AspeedSoCState *s = ASPEED_SOC(obj);
> > +    AspeedSoCClass *sc = ASPEED_SOC_GET_CLASS(s);
> > +    int i;
> > +    object_initialize_child(obj, "armv7m", &a->armv7m, TYPE_ARMV7M);
> > +
> > +    s->sysclk = qdev_init_clock_in(DEVICE(s), "sysclk", NULL, NULL,
> > + 0);
> > +
> > +    /* AST1040 uses the AST2700 IO SCU model */
> > +    object_initialize_child(obj, "scu", &s->scu,
> TYPE_ASPEED_2700_SCUIO);
> > +    qdev_prop_set_uint32(DEVICE(&s->scu), "silicon-rev",
> > + sc->silicon_rev);
> > +
> > +    object_property_add_alias(obj, "hw-strap1", OBJECT(&s->scu),
> "hw-strap1");
> > +    object_property_add_alias(obj, "hw-strap2", OBJECT(&s->scu),
> > + "hw-strap2");
> > +
> > +    for (i = 0; i < sc->uarts_num; i++) {
> > +        object_initialize_child(obj, "uart[*]", &s->uart[i],
> TYPE_SERIAL_MM);
> > +    }
> > +
> > +    object_initialize_child(obj, "pwm", &s->pwm,
> TYPE_UNIMPLEMENTED_DEVICE);
> > +    object_initialize_child(obj, "espi", &s->espi,
> TYPE_UNIMPLEMENTED_DEVICE);
> > +    object_initialize_child(obj, "udc", &s->udc,
> TYPE_UNIMPLEMENTED_DEVICE);
> > +    object_initialize_child(obj, "sgpiom", &s->sgpiom[0],
> > +                            TYPE_UNIMPLEMENTED_DEVICE);
> > +    object_initialize_child(obj, "jtag[0]", &s->jtag[0],
> > +                            TYPE_UNIMPLEMENTED_DEVICE);
> > +    object_initialize_child(obj, "jtag[1]", &s->jtag[1],
> > +                            TYPE_UNIMPLEMENTED_DEVICE); }
> > +
> > +static void aspeed_soc_ast1040_realize(DeviceState *dev_soc, Error
> > +**errp) {
> > +    Aspeed10x0SoCState *a = ASPEED10X0_SOC(dev_soc);
> > +    AspeedSoCState *s = ASPEED_SOC(dev_soc);
> > +    AspeedSoCClass *sc = ASPEED_SOC_GET_CLASS(s);
> > +    DeviceState *armv7m;
> > +    Error *err = NULL;
> > +    char name[64];
> > +    int uart;
> > +    int i;
> > +
> > +    if (!clock_has_source(s->sysclk)) {
> > +        error_setg(errp, "sysclk clock must be wired up by the board
> code");
> > +        return;
> > +    }
> > +
> > +    /* AST1040 CPU Core */
> > +    armv7m = DEVICE(&a->armv7m);
> > +    qdev_prop_set_uint32(armv7m, "num-irq", 256);
> > +    qdev_prop_set_string(armv7m, "cpu-type",
> > +                         aspeed_soc_cpu_type(sc->valid_cpu_types));
> > +    qdev_connect_clock_in(armv7m, "cpuclk", s->sysclk);
> > +    object_property_set_link(OBJECT(&a->armv7m), "memory",
> > +                             OBJECT(s->memory), &error_abort);
> > +    sysbus_realize(SYS_BUS_DEVICE(&a->armv7m), &error_abort);
> > +
> > +    /* Internal SDRAM */
> > +    snprintf(name, sizeof(name), "aspeed.sdram.%d",
> > +             CPU(a->armv7m.cpu)->cpu_index);
> 
> please use a g_autofree variable.

Will do

> 
> > +    memory_region_init_ram(&s->sdram, OBJECT(s), name,
> sc->sdram_size, &err);
> > +    if (err) {
> > +        error_propagate(errp, err);
> > +        return;
> > +    }
> > +    memory_region_add_subregion(s->memory,
> sc->memmap[ASPEED_DEV_SDRAM],
> > +                                &s->sdram);
> > +
> > +    /* Internal SRAM */
> > +    snprintf(name, sizeof(name), "aspeed.sram.%d",
> > +             CPU(a->armv7m.cpu)->cpu_index);
> > +    memory_region_init_ram(&s->sram, OBJECT(s), name, sc->sram_size,
> &err);
> > +    if (err) {
> > +        error_propagate(errp, err);
> > +        return;
> > +    }
> > +    memory_region_add_subregion(s->memory,
> sc->memmap[ASPEED_DEV_SRAM],
> > +                                &s->sram);
> > +
> > +    /* SCU */
> > +    if (!sysbus_realize(SYS_BUS_DEVICE(&s->scu), errp)) {
> > +        return;
> > +    }
> > +    aspeed_mmio_map(s->memory, SYS_BUS_DEVICE(&s->scu), 0,
> > +                    sc->memmap[ASPEED_DEV_SCU]);
> > +
> > +    /* UART */
> > +    for (i = 0, uart = sc->uarts_base; i < sc->uarts_num; i++, uart++) {
> > +        if (!aspeed_soc_uart_realize(s->memory, &s->uart[i],
> > +                                     sc->memmap[uart], errp)) {
> > +            return;
> > +        }
> > +        sysbus_connect_irq(SYS_BUS_DEVICE(&s->uart[i]), 0,
> > +                           aspeed_soc_ast1040_get_irq(s, uart));
> > +    }
> > +
> > +    /* Unimplemented peripherals */
> > +    aspeed_mmio_map_unimplemented(s->memory,
> SYS_BUS_DEVICE(&s->pwm),
> > +                                  "aspeed.pwm",
> > +
> sc->memmap[ASPEED_DEV_PWM],
> > + 0x10000);
> > +
> > +    aspeed_mmio_map_unimplemented(s->memory,
> SYS_BUS_DEVICE(&s->espi),
> > +                                  "aspeed.espi",
> > +
> sc->memmap[ASPEED_DEV_ESPI],
> > + 0x1000);
> > +
> > +    aspeed_mmio_map_unimplemented(s->memory,
> SYS_BUS_DEVICE(&s->udc),
> > +                                  "aspeed.udc",
> > +
> sc->memmap[ASPEED_DEV_UDC],
> > + 0x4000);
> > +
> > +    aspeed_mmio_map_unimplemented(s->memory,
> SYS_BUS_DEVICE(&s->sgpiom[0]),
> > +                                  "aspeed.sgpiom",
> > +
> sc->memmap[ASPEED_DEV_SGPIOM0],
> > + 0x1000);
> 
> ASPEED_DEV_SGPIOM1 too may be. Or we don't care ?
> 

Will add

Thanks for the review and suggestion.
Jamin
> > +
> > +    aspeed_mmio_map_unimplemented(s->memory,
> SYS_BUS_DEVICE(&s->jtag[0]),
> > +                                  "aspeed.jtag0",
> > +
> sc->memmap[ASPEED_DEV_JTAG0],
> > + 0x100);
> > +
> > +    aspeed_mmio_map_unimplemented(s->memory,
> SYS_BUS_DEVICE(&s->jtag[1]),
> > +                                  "aspeed.jtag1",
> > +
> sc->memmap[ASPEED_DEV_JTAG1],
> > +0x100); }
> > +
> > +static void aspeed_soc_ast1040_class_init(ObjectClass *klass, const
> > +void *data) {
> > +    static const char * const valid_cpu_types[] = {
> > +        ARM_CPU_TYPE_NAME("cortex-m4"), /* TODO cortex-m4f */
> > +        NULL
> > +    };
> > +    DeviceClass *dc = DEVICE_CLASS(klass);
> > +    AspeedSoCClass *sc = ASPEED_SOC_CLASS(dc);
> > +
> > +    /* Reason: The Aspeed SoC can only be instantiated from a board */
> > +    dc->user_creatable = false;
> > +    dc->realize = aspeed_soc_ast1040_realize;
> > +
> > +    sc->valid_cpu_types = valid_cpu_types;
> > +    sc->silicon_rev     = AST1040_A0_SILICON_REV;
> > +    sc->sdram_size      = 16 * MiB;
> > +    sc->sram_size       = 512 * KiB;
> > +    sc->uarts_num       = 13;
> > +    sc->uarts_base      = ASPEED_DEV_UART0;
> > +    sc->irqmap          = aspeed_soc_ast1040_irqmap;
> > +    sc->memmap          = aspeed_soc_ast1040_memmap;
> > +    sc->num_cpus        = 1;
> > +}
> > +
> > +static const TypeInfo aspeed_soc_ast1040_types[] = {
> > +    {
> > +        .name           = "ast1040-a0",
> > +        .parent         = TYPE_ASPEED10X0_SOC,
> > +        .instance_init  = aspeed_soc_ast1040_init,
> > +        .class_init     = aspeed_soc_ast1040_class_init,
> > +    }
> > +};
> > +
> > +DEFINE_TYPES(aspeed_soc_ast1040_types)
> > diff --git a/hw/arm/meson.build b/hw/arm/meson.build index
> > 80068f70bb..fa3a848492 100644
> > --- a/hw/arm/meson.build
> > +++ b/hw/arm/meson.build
> > @@ -62,7 +62,8 @@ arm_ss.add(when: 'CONFIG_ASPEED_SOC', if_true:
> files(
> >     'aspeed_ast2600_gb200nvl.c',
> >     'aspeed_ast2600_rainier.c',
> >     'aspeed_ast10x0.c',
> > -  'aspeed_ast10x0_evb.c'))
> > +  'aspeed_ast10x0_evb.c',
> > +  'aspeed_ast1040.c'))
> >   arm_common_ss.add(when: ['CONFIG_ASPEED_SOC',
> 'TARGET_AARCH64'], if_true: files(
> >     'aspeed_ast1700.c',
> >     'aspeed_ast27x0.c',

Reply via email to