Adds a test machine for the IBM PPE42 processor, including a
DEC, FIT, WDT and 1MB of ram.
The purpose of this machine is only to provide a generic platform
for testing instructions of the recently added PPE42 processor
model which is used extensively in the IBM Power9, Power10 and
future Power server processors.
Signed-off-by: Glenn Miles <milesg@linux.ibm.com>
---
Changes from previous version
- Added ppe42_machine.c to MAINTAINERS file with self as maintainer
MAINTAINERS | 6 ++++
hw/ppc/Kconfig | 9 ++++++
hw/ppc/meson.build | 2 ++
hw/ppc/ppc_booke.c | 7 ++++-
hw/ppc/ppe42_machine.c | 69 ++++++++++++++++++++++++++++++++++++++++++
include/hw/ppc/ppc.h | 1 +
6 files changed, 93 insertions(+), 1 deletion(-)
create mode 100644 hw/ppc/ppe42_machine.c
diff --git a/MAINTAINERS b/MAINTAINERS
index a07086ed76..52fa303e0a 100644
--- a/MAINTAINERS
+++ b/MAINTAINERS
@@ -1531,6 +1531,12 @@ F: include/hw/pci-host/grackle.h
F: pc-bios/qemu_vga.ndrv
F: tests/functional/test_ppc_mac.py
+PPE42
+M: Glenn Miles <milesg@linux.ibm.com>
+L: qemu-ppc@nongnu.org
+S: Odd Fixes
+F: hw/ppc/ppe42_machine.c
+
PReP
M: Hervé Poussineau <hpoussin@reactos.org>
L: qemu-ppc@nongnu.org
diff --git a/hw/ppc/Kconfig b/hw/ppc/Kconfig
index ced6bbc740..3fdea5919c 100644
--- a/hw/ppc/Kconfig
+++ b/hw/ppc/Kconfig
@@ -44,6 +44,15 @@ config POWERNV
select SSI_M25P80
select PNV_SPI
+config PPC405
+ bool
+ default y
+ depends on PPC
+ select M48T59
+ select PFLASH_CFI02
+ select PPC4XX
+ select SERIAL
+
config PPC440
bool
default y
diff --git a/hw/ppc/meson.build b/hw/ppc/meson.build
index 9893f8adeb..170b90ae7d 100644
--- a/hw/ppc/meson.build
+++ b/hw/ppc/meson.build
@@ -57,6 +57,8 @@ ppc_ss.add(when: 'CONFIG_POWERNV', if_true: files(
'pnv_n1_chiplet.c',
))
# PowerPC 4xx boards
+ppc_ss.add(when: 'CONFIG_PPC405', if_true: files(
+ 'ppe42_machine.c'))
ppc_ss.add(when: 'CONFIG_PPC440', if_true: files(
'ppc440_bamboo.c',
'ppc440_uc.c'))
diff --git a/hw/ppc/ppc_booke.c b/hw/ppc/ppc_booke.c
index 3872ae2822..13403a56b1 100644
--- a/hw/ppc/ppc_booke.c
+++ b/hw/ppc/ppc_booke.c
@@ -352,7 +352,12 @@ void ppc_booke_timers_init(PowerPCCPU *cpu, uint32_t freq, uint32_t flags)
booke_timer = g_new0(booke_timer_t, 1);
cpu->env.tb_env = tb_env;
- tb_env->flags = flags | PPC_TIMER_BOOKE | PPC_DECR_ZERO_TRIGGERED;
+ if (flags & PPC_TIMER_PPE) {
+ /* PPE's use a modified version of the booke behavior */
+ tb_env->flags = flags | PPC_DECR_UNDERFLOW_TRIGGERED;
+ } else {
+ tb_env->flags = flags | PPC_TIMER_BOOKE | PPC_DECR_ZERO_TRIGGERED;
+ }
tb_env->tb_freq = freq;
tb_env->decr_freq = freq;
diff --git a/hw/ppc/ppe42_machine.c b/hw/ppc/ppe42_machine.c
new file mode 100644
index 0000000000..0bc295da28
--- /dev/null
+++ b/hw/ppc/ppe42_machine.c
@@ -0,0 +1,69 @@
+
+/*
+ * Test Machine for the IBM PPE42 processor
+ *
+ * Copyright (c) 2025, IBM Corporation.
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "qemu/osdep.h"
+#include "qemu/units.h"
+#include "qemu/error-report.h"
+#include "system/address-spaces.h"
+#include "hw/boards.h"
+#include "hw/ppc/ppc.h"
+#include "system/system.h"
+#include "system/reset.h"
+#include "system/kvm.h"
+
+static void main_cpu_reset(void *opaque)
+{
+ PowerPCCPU *cpu = opaque;
+
+ cpu_reset(CPU(cpu));
+}
+
+static void ppe42_machine_init(MachineState *machine)
+{
+ PowerPCCPU *cpu;
+ CPUPPCState *env;
+
+ if (kvm_enabled()) {
+ error_report("machine %s does not support the KVM accelerator",
+ MACHINE_GET_CLASS(machine)->name);
+ exit(EXIT_FAILURE);
+ }
+
+ /* init CPU */
+ cpu = POWERPC_CPU(cpu_create(machine->cpu_type));
+ env = &cpu->env;
+ if (PPC_INPUT(env) != PPC_FLAGS_INPUT_PPE42) {
+ error_report("Incompatible CPU, only PPE42 bus supported");
+ exit(1);
+ }
+
+ qemu_register_reset(main_cpu_reset, cpu);
+
+ /* This sets the decrementer timebase */
+ ppc_booke_timers_init(cpu, 37500000, PPC_TIMER_PPE);
+
+ /* RAM */
+ if (machine->ram_size > 2 * GiB) {
+ error_report("RAM size more than 2 GiB is not supported");
+ exit(1);
+ }
+ memory_region_add_subregion(get_system_memory(), 0xfff80000, machine->ram);
+}
+
+
+static void ppe42_machine_class_init(MachineClass *mc)
+{
+ mc->desc = "PPE42 Test Machine";
+ mc->init = ppe42_machine_init;
+ mc->default_cpu_type = POWERPC_CPU_TYPE_NAME("PPE42XM");
+ mc->default_ram_id = "ram";
+ mc->default_ram_size = 1 * MiB;
+}
+
+DEFINE_MACHINE("ppe42_machine", ppe42_machine_class_init)
diff --git a/include/hw/ppc/ppc.h b/include/hw/ppc/ppc.h
index 8a14d623f8..cb51d704c6 100644
--- a/include/hw/ppc/ppc.h
+++ b/include/hw/ppc/ppc.h
@@ -52,6 +52,7 @@ struct ppc_tb_t {
#define PPC_DECR_UNDERFLOW_LEVEL (1 << 4) /* Decr interrupt active when
* the most significant bit is 1.
*/
+#define PPC_TIMER_PPE (1 << 5) /* Enable PPE support */
uint64_t cpu_ppc_get_tb(ppc_tb_t *tb_env, uint64_t vmclk, int64_t tb_offset);
void cpu_ppc_tb_init(CPUPPCState *env, uint32_t freq);
--
2.43.0
On 8/26/25 22:17, Glenn Miles wrote: > Adds a test machine for the IBM PPE42 processor, including a > DEC, FIT, WDT and 1MB of ram. > > The purpose of this machine is only to provide a generic platform > for testing instructions of the recently added PPE42 processor > model which is used extensively in the IBM Power9, Power10 and > future Power server processors. > > Signed-off-by: Glenn Miles <milesg@linux.ibm.com> > --- > > Changes from previous version > - Added ppe42_machine.c to MAINTAINERS file with self as maintainer > > MAINTAINERS | 6 ++++ > hw/ppc/Kconfig | 9 ++++++ > hw/ppc/meson.build | 2 ++ > hw/ppc/ppc_booke.c | 7 ++++- > hw/ppc/ppe42_machine.c | 69 ++++++++++++++++++++++++++++++++++++++++++ > include/hw/ppc/ppc.h | 1 + > 6 files changed, 93 insertions(+), 1 deletion(-) > create mode 100644 hw/ppc/ppe42_machine.c > > diff --git a/MAINTAINERS b/MAINTAINERS > index a07086ed76..52fa303e0a 100644 > --- a/MAINTAINERS > +++ b/MAINTAINERS > @@ -1531,6 +1531,12 @@ F: include/hw/pci-host/grackle.h > F: pc-bios/qemu_vga.ndrv > F: tests/functional/test_ppc_mac.py > > +PPE42 > +M: Glenn Miles <milesg@linux.ibm.com> > +L: qemu-ppc@nongnu.org > +S: Odd Fixes > +F: hw/ppc/ppe42_machine.c > + > PReP > M: Hervé Poussineau <hpoussin@reactos.org> > L: qemu-ppc@nongnu.org > diff --git a/hw/ppc/Kconfig b/hw/ppc/Kconfig > index ced6bbc740..3fdea5919c 100644 > --- a/hw/ppc/Kconfig > +++ b/hw/ppc/Kconfig > @@ -44,6 +44,15 @@ config POWERNV > select SSI_M25P80 > select PNV_SPI > > +config PPC405 > + bool > + default y > + depends on PPC > + select M48T59 > + select PFLASH_CFI02 > + select PPC4XX > + select SERIAL > + > config PPC440 > bool > default y > diff --git a/hw/ppc/meson.build b/hw/ppc/meson.build > index 9893f8adeb..170b90ae7d 100644 > --- a/hw/ppc/meson.build > +++ b/hw/ppc/meson.build > @@ -57,6 +57,8 @@ ppc_ss.add(when: 'CONFIG_POWERNV', if_true: files( > 'pnv_n1_chiplet.c', > )) > # PowerPC 4xx boards > +ppc_ss.add(when: 'CONFIG_PPC405', if_true: files( > + 'ppe42_machine.c')) > ppc_ss.add(when: 'CONFIG_PPC440', if_true: files( > 'ppc440_bamboo.c', > 'ppc440_uc.c')) > diff --git a/hw/ppc/ppc_booke.c b/hw/ppc/ppc_booke.c > index 3872ae2822..13403a56b1 100644 > --- a/hw/ppc/ppc_booke.c > +++ b/hw/ppc/ppc_booke.c > @@ -352,7 +352,12 @@ void ppc_booke_timers_init(PowerPCCPU *cpu, uint32_t freq, uint32_t flags) > booke_timer = g_new0(booke_timer_t, 1); > > cpu->env.tb_env = tb_env; > - tb_env->flags = flags | PPC_TIMER_BOOKE | PPC_DECR_ZERO_TRIGGERED; > + if (flags & PPC_TIMER_PPE) { PPC_TIMER_PPE definition should be introduced in its own patch. > + /* PPE's use a modified version of the booke behavior */ > + tb_env->flags = flags | PPC_DECR_UNDERFLOW_TRIGGERED; > + } else { > + tb_env->flags = flags | PPC_TIMER_BOOKE | PPC_DECR_ZERO_TRIGGERED; > + } > > tb_env->tb_freq = freq; > tb_env->decr_freq = freq; > diff --git a/hw/ppc/ppe42_machine.c b/hw/ppc/ppe42_machine.c > new file mode 100644 > index 0000000000..0bc295da28 > --- /dev/null > +++ b/hw/ppc/ppe42_machine.c > @@ -0,0 +1,69 @@ > + > +/* > + * Test Machine for the IBM PPE42 processor > + * > + * Copyright (c) 2025, IBM Corporation. > + * > + * SPDX-License-Identifier: GPL-2.0-or-later > + */ > + > +#include "qemu/osdep.h" > +#include "qemu/units.h" > +#include "qemu/error-report.h" > +#include "system/address-spaces.h" > +#include "hw/boards.h" > +#include "hw/ppc/ppc.h" > +#include "system/system.h" > +#include "system/reset.h" > +#include "system/kvm.h" > + > +static void main_cpu_reset(void *opaque) > +{ > + PowerPCCPU *cpu = opaque; > + > + cpu_reset(CPU(cpu)); There are no register settings ? Just asking > +} > + > +static void ppe42_machine_init(MachineState *machine) > +{ > + PowerPCCPU *cpu; > + CPUPPCState *env; > + > + if (kvm_enabled()) { > + error_report("machine %s does not support the KVM accelerator", > + MACHINE_GET_CLASS(machine)->name); > + exit(EXIT_FAILURE); > + } > + > + /* init CPU */ > + cpu = POWERPC_CPU(cpu_create(machine->cpu_type)); I would introduce a specific MachineState for the ppe42 Machine: struct Ppe42MachineState { /* Private */ MachineState parent_obj; /* Public */ PowerPCCPU cpu; }; and use qdev_realize() too. > + env = &cpu->env; > + if (PPC_INPUT(env) != PPC_FLAGS_INPUT_PPE42) { > + error_report("Incompatible CPU, only PPE42 bus supported"); Can't we define valid_cpu_types instead ? > + exit(1); > + } > + > + qemu_register_reset(main_cpu_reset, cpu); > + > + /* This sets the decrementer timebase */ > + ppc_booke_timers_init(cpu, 37500000, PPC_TIMER_PPE); > + > + /* RAM */ > + if (machine->ram_size > 2 * GiB) { 2GB RAM ? really ? > + error_report("RAM size more than 2 GiB is not supported"); > + exit(1); > + } > + memory_region_add_subregion(get_system_memory(), 0xfff80000, machine->ram); > +} > + > + > +static void ppe42_machine_class_init(MachineClass *mc) > +{ > + mc->desc = "PPE42 Test Machine"; > + mc->init = ppe42_machine_init; > + mc->default_cpu_type = POWERPC_CPU_TYPE_NAME("PPE42XM"); > + mc->default_ram_id = "ram"; > + mc->default_ram_size = 1 * MiB; > +} > + > +DEFINE_MACHINE("ppe42_machine", ppe42_machine_class_init) > diff --git a/include/hw/ppc/ppc.h b/include/hw/ppc/ppc.h > index 8a14d623f8..cb51d704c6 100644 > --- a/include/hw/ppc/ppc.h > +++ b/include/hw/ppc/ppc.h > @@ -52,6 +52,7 @@ struct ppc_tb_t { > #define PPC_DECR_UNDERFLOW_LEVEL (1 << 4) /* Decr interrupt active when > * the most significant bit is 1. > */ > +#define PPC_TIMER_PPE (1 << 5) /* Enable PPE support */ > > uint64_t cpu_ppc_get_tb(ppc_tb_t *tb_env, uint64_t vmclk, int64_t tb_offset); > void cpu_ppc_tb_init(CPUPPCState *env, uint32_t freq);
On Mon, 2025-09-01 at 09:25 +0200, Cédric Le Goater wrote: > On 8/26/25 22:17, Glenn Miles wrote: > > Adds a test machine for the IBM PPE42 processor, including a > > DEC, FIT, WDT and 1MB of ram. > > > > The purpose of this machine is only to provide a generic platform > > for testing instructions of the recently added PPE42 processor > > model which is used extensively in the IBM Power9, Power10 and > > future Power server processors. > > > > Signed-off-by: Glenn Miles <milesg@linux.ibm.com> > > --- > > > > Changes from previous version > > - Added ppe42_machine.c to MAINTAINERS file with self as maintainer > > > > MAINTAINERS | 6 ++++ > > hw/ppc/Kconfig | 9 ++++++ > > hw/ppc/meson.build | 2 ++ > > hw/ppc/ppc_booke.c | 7 ++++- > > hw/ppc/ppe42_machine.c | 69 ++++++++++++++++++++++++++++++++++++++++++ > > include/hw/ppc/ppc.h | 1 + > > 6 files changed, 93 insertions(+), 1 deletion(-) > > create mode 100644 hw/ppc/ppe42_machine.c > > > > diff --git a/MAINTAINERS b/MAINTAINERS > > index a07086ed76..52fa303e0a 100644 > > --- a/MAINTAINERS > > +++ b/MAINTAINERS > > @@ -1531,6 +1531,12 @@ F: include/hw/pci-host/grackle.h > > F: pc-bios/qemu_vga.ndrv > > F: tests/functional/test_ppc_mac.py > > > > +PPE42 > > +M: Glenn Miles <milesg@linux.ibm.com> > > +L: qemu-ppc@nongnu.org > > +S: Odd Fixes > > +F: hw/ppc/ppe42_machine.c > > + > > PReP > > M: Hervé Poussineau <hpoussin@reactos.org> > > L: qemu-ppc@nongnu.org > > diff --git a/hw/ppc/Kconfig b/hw/ppc/Kconfig > > index ced6bbc740..3fdea5919c 100644 > > --- a/hw/ppc/Kconfig > > +++ b/hw/ppc/Kconfig > > @@ -44,6 +44,15 @@ config POWERNV > > select SSI_M25P80 > > select PNV_SPI > > > > +config PPC405 > > + bool > > + default y > > + depends on PPC > > + select M48T59 > > + select PFLASH_CFI02 > > + select PPC4XX > > + select SERIAL > > + > > config PPC440 > > bool > > default y > > diff --git a/hw/ppc/meson.build b/hw/ppc/meson.build > > index 9893f8adeb..170b90ae7d 100644 > > --- a/hw/ppc/meson.build > > +++ b/hw/ppc/meson.build > > @@ -57,6 +57,8 @@ ppc_ss.add(when: 'CONFIG_POWERNV', if_true: files( > > 'pnv_n1_chiplet.c', > > )) > > # PowerPC 4xx boards > > +ppc_ss.add(when: 'CONFIG_PPC405', if_true: files( > > + 'ppe42_machine.c')) > > ppc_ss.add(when: 'CONFIG_PPC440', if_true: files( > > 'ppc440_bamboo.c', > > 'ppc440_uc.c')) > > diff --git a/hw/ppc/ppc_booke.c b/hw/ppc/ppc_booke.c > > index 3872ae2822..13403a56b1 100644 > > --- a/hw/ppc/ppc_booke.c > > +++ b/hw/ppc/ppc_booke.c > > @@ -352,7 +352,12 @@ void ppc_booke_timers_init(PowerPCCPU *cpu, uint32_t freq, uint32_t flags) > > booke_timer = g_new0(booke_timer_t, 1); > > > > cpu->env.tb_env = tb_env; > > - tb_env->flags = flags | PPC_TIMER_BOOKE | PPC_DECR_ZERO_TRIGGERED; > > + if (flags & PPC_TIMER_PPE) { > > PPC_TIMER_PPE definition should be introduced in its own patch. > Ok, will do. > > > + /* PPE's use a modified version of the booke behavior */ > > + tb_env->flags = flags | PPC_DECR_UNDERFLOW_TRIGGERED; > > + } else { > > + tb_env->flags = flags | PPC_TIMER_BOOKE | PPC_DECR_ZERO_TRIGGERED; > > + } > > > > tb_env->tb_freq = freq; > > tb_env->decr_freq = freq; > > diff --git a/hw/ppc/ppe42_machine.c b/hw/ppc/ppe42_machine.c > > new file mode 100644 > > index 0000000000..0bc295da28 > > --- /dev/null > > +++ b/hw/ppc/ppe42_machine.c > > @@ -0,0 +1,69 @@ > > + > > +/* > > + * Test Machine for the IBM PPE42 processor > > + * > > + * Copyright (c) 2025, IBM Corporation. > > + * > > + * SPDX-License-Identifier: GPL-2.0-or-later > > + */ > > + > > +#include "qemu/osdep.h" > > +#include "qemu/units.h" > > +#include "qemu/error-report.h" > > +#include "system/address-spaces.h" > > +#include "hw/boards.h" > > +#include "hw/ppc/ppc.h" > > +#include "system/system.h" > > +#include "system/reset.h" > > +#include "system/kvm.h" > > + > > +static void main_cpu_reset(void *opaque) > > +{ > > + PowerPCCPU *cpu = opaque; > > + > > + cpu_reset(CPU(cpu)); > > There are no register settings ? Just asking > Admittedly, I did not do much digging into the reset case. I'll look into it. > > +} > > + > > +static void ppe42_machine_init(MachineState *machine) > > +{ > > + PowerPCCPU *cpu; > > + CPUPPCState *env; > > + > > + if (kvm_enabled()) { > > + error_report("machine %s does not support the KVM accelerator", > > + MACHINE_GET_CLASS(machine)->name); > > + exit(EXIT_FAILURE); > > + } > > + > > + /* init CPU */ > > + cpu = POWERPC_CPU(cpu_create(machine->cpu_type)); > > I would introduce a specific MachineState for the ppe42 Machine: > > struct Ppe42MachineState { > /* Private */ > MachineState parent_obj; > /* Public */ > > PowerPCCPU cpu; > }; > > and use qdev_realize() too. > > Sure, no problem. > > + env = &cpu->env; > > + if (PPC_INPUT(env) != PPC_FLAGS_INPUT_PPE42) { > > + error_report("Incompatible CPU, only PPE42 bus supported"); > > Can't we define valid_cpu_types instead ? > Ah, I was not aware of that option. Thanks, I'll give that a try. > > + exit(1); > > + } > > + > > + qemu_register_reset(main_cpu_reset, cpu); > > + > > + /* This sets the decrementer timebase */ > > + ppc_booke_timers_init(cpu, 37500000, PPC_TIMER_PPE); > > + > > + /* RAM */ > > + if (machine->ram_size > 2 * GiB) { > > 2GB RAM ? really ? > Hmm, not sure what I was thinking there. Since the ram starts at address 0xfff80000, I'll change that to 512K. Thanks for taking a look! Glenn > > + error_report("RAM size more than 2 GiB is not supported"); > > + exit(1); > > + } > > + memory_region_add_subregion(get_system_memory(), 0xfff80000, machine->ram); > > +} > > + > > + > > +static void ppe42_machine_class_init(MachineClass *mc) > > +{ > > + mc->desc = "PPE42 Test Machine"; > > + mc->init = ppe42_machine_init; > > + mc->default_cpu_type = POWERPC_CPU_TYPE_NAME("PPE42XM"); > > + mc->default_ram_id = "ram"; > > + mc->default_ram_size = 1 * MiB; > > +} > > + > > +DEFINE_MACHINE("ppe42_machine", ppe42_machine_class_init) > > diff --git a/include/hw/ppc/ppc.h b/include/hw/ppc/ppc.h > > index 8a14d623f8..cb51d704c6 100644 > > --- a/include/hw/ppc/ppc.h > > +++ b/include/hw/ppc/ppc.h > > @@ -52,6 +52,7 @@ struct ppc_tb_t { > > #define PPC_DECR_UNDERFLOW_LEVEL (1 << 4) /* Decr interrupt active when > > * the most significant bit is 1. > > */ > > +#define PPC_TIMER_PPE (1 << 5) /* Enable PPE support */ > > > > uint64_t cpu_ppc_get_tb(ppc_tb_t *tb_env, uint64_t vmclk, int64_t tb_offset); > > void cpu_ppc_tb_init(CPUPPCState *env, uint32_t freq);
On Tue, 2025-09-02 at 11:56 -0500, Miles Glenn wrote: > > > + > > > +static void main_cpu_reset(void *opaque) > > > +{ > > > + PowerPCCPU *cpu = opaque; > > > + > > > + cpu_reset(CPU(cpu)); > > > > There are no register settings ? Just asking > > > > Admittedly, I did not do much digging into the reset case. I'll look > into it. Cédric, As far as I can tell, it doesn't look like we need to add any other registers here besides the cpu registers, which are already handled by cpu_reset(). Also, I did verify that executing a system_reset from the QEMU command line works as expected (resets to the correct vector and executes successfully). Thanks, Glenn
On Mon, 1 Sep 2025, Cédric Le Goater wrote: > On 8/26/25 22:17, Glenn Miles wrote: > I would introduce a specific MachineState for the ppe42 Machine: > > struct Ppe42MachineState { > /* Private */ > MachineState parent_obj; > /* Public */ Private/public comments are not recommended any more for QOM objects since commit 067109a11c8 just an empty line after parent_obj. Regards, BALATON Zoltan
© 2016 - 2025 Red Hat, Inc.