1*27407470SHelge Deller /*
2*27407470SHelge Deller * HP Diva GSP controller
3*27407470SHelge Deller *
4*27407470SHelge Deller * The Diva PCI boards are Remote Management cards for PA-RISC machines.
5*27407470SHelge Deller * They come with built-in 16550A multi UARTs for serial consoles
6*27407470SHelge Deller * and a mailbox-like memory area for hardware auto-reboot functionality.
7*27407470SHelge Deller * GSP stands for "Guardian Service Processor". Later products were marketed
8*27407470SHelge Deller * "Management Processor" (MP).
9*27407470SHelge Deller *
10*27407470SHelge Deller * Diva cards are multifunctional cards. The first part, the aux port,
11*27407470SHelge Deller * is on physical machines not useable but we still try to mimic it here.
12*27407470SHelge Deller *
13*27407470SHelge Deller * SPDX-License-Identifier: GPL-2.0-or-later
14*27407470SHelge Deller *
15*27407470SHelge Deller * Copyright (c) 2025 Helge Deller <deller@gmx.de>
16*27407470SHelge Deller */
17*27407470SHelge Deller
18*27407470SHelge Deller #include "qemu/osdep.h"
19*27407470SHelge Deller #include "qemu/units.h"
20*27407470SHelge Deller #include "hw/char/serial.h"
21*27407470SHelge Deller #include "hw/irq.h"
22*27407470SHelge Deller #include "hw/pci/pci_device.h"
23*27407470SHelge Deller #include "hw/qdev-properties.h"
24*27407470SHelge Deller #include "hw/qdev-properties-system.h"
25*27407470SHelge Deller #include "migration/vmstate.h"
26*27407470SHelge Deller
27*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA 0x1048
28*27407470SHelge Deller /* various DIVA GSP cards: */
29*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_TOSCA1 0x1049
30*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_TOSCA2 0x104A
31*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_MAESTRO 0x104B
32*27407470SHelge Deller #define PCI_DEVICE_ID_HP_REO_IOC 0x10f1
33*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_HALFDOME 0x1223
34*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_KEYSTONE 0x1226
35*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_POWERBAR 0x1227
36*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_EVEREST 0x1282
37*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_AUX 0x1290
38*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_RMP3 0x1301
39*27407470SHelge Deller #define PCI_DEVICE_ID_HP_DIVA_HURRICANE 0x132a
40*27407470SHelge Deller
41*27407470SHelge Deller
42*27407470SHelge Deller #define PCI_SERIAL_MAX_PORTS 4
43*27407470SHelge Deller
44*27407470SHelge Deller typedef struct PCIDivaSerialState {
45*27407470SHelge Deller PCIDevice dev;
46*27407470SHelge Deller MemoryRegion membar; /* for serial ports */
47*27407470SHelge Deller MemoryRegion mailboxbar; /* for hardware mailbox */
48*27407470SHelge Deller uint32_t subvendor;
49*27407470SHelge Deller uint32_t ports;
50*27407470SHelge Deller char *name[PCI_SERIAL_MAX_PORTS];
51*27407470SHelge Deller SerialState state[PCI_SERIAL_MAX_PORTS];
52*27407470SHelge Deller uint32_t level[PCI_SERIAL_MAX_PORTS];
53*27407470SHelge Deller qemu_irq *irqs;
54*27407470SHelge Deller uint8_t prog_if;
55*27407470SHelge Deller bool disable;
56*27407470SHelge Deller } PCIDivaSerialState;
57*27407470SHelge Deller
diva_pci_exit(PCIDevice * dev)58*27407470SHelge Deller static void diva_pci_exit(PCIDevice *dev)
59*27407470SHelge Deller {
60*27407470SHelge Deller PCIDivaSerialState *pci = DO_UPCAST(PCIDivaSerialState, dev, dev);
61*27407470SHelge Deller SerialState *s;
62*27407470SHelge Deller int i;
63*27407470SHelge Deller
64*27407470SHelge Deller for (i = 0; i < pci->ports; i++) {
65*27407470SHelge Deller s = pci->state + i;
66*27407470SHelge Deller qdev_unrealize(DEVICE(s));
67*27407470SHelge Deller memory_region_del_subregion(&pci->membar, &s->io);
68*27407470SHelge Deller g_free(pci->name[i]);
69*27407470SHelge Deller }
70*27407470SHelge Deller qemu_free_irqs(pci->irqs, pci->ports);
71*27407470SHelge Deller }
72*27407470SHelge Deller
multi_serial_irq_mux(void * opaque,int n,int level)73*27407470SHelge Deller static void multi_serial_irq_mux(void *opaque, int n, int level)
74*27407470SHelge Deller {
75*27407470SHelge Deller PCIDivaSerialState *pci = opaque;
76*27407470SHelge Deller int i, pending = 0;
77*27407470SHelge Deller
78*27407470SHelge Deller pci->level[n] = level;
79*27407470SHelge Deller for (i = 0; i < pci->ports; i++) {
80*27407470SHelge Deller if (pci->level[i]) {
81*27407470SHelge Deller pending = 1;
82*27407470SHelge Deller }
83*27407470SHelge Deller }
84*27407470SHelge Deller pci_set_irq(&pci->dev, pending);
85*27407470SHelge Deller }
86*27407470SHelge Deller
87*27407470SHelge Deller struct diva_info {
88*27407470SHelge Deller unsigned int nports:4; /* number of serial ports */
89*27407470SHelge Deller unsigned int omask:12; /* offset mask: BIT(1) -> offset 8 */
90*27407470SHelge Deller };
91*27407470SHelge Deller
diva_get_diva_info(PCIDeviceClass * pc)92*27407470SHelge Deller static struct diva_info diva_get_diva_info(PCIDeviceClass *pc)
93*27407470SHelge Deller {
94*27407470SHelge Deller switch (pc->subsystem_id) {
95*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_POWERBAR:
96*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_HURRICANE:
97*27407470SHelge Deller return (struct diva_info) { .nports = 1,
98*27407470SHelge Deller .omask = BIT(0) };
99*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_TOSCA2:
100*27407470SHelge Deller return (struct diva_info) { .nports = 2,
101*27407470SHelge Deller .omask = BIT(0) | BIT(1) };
102*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_TOSCA1:
103*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_HALFDOME:
104*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_KEYSTONE:
105*27407470SHelge Deller return (struct diva_info) { .nports = 3,
106*27407470SHelge Deller .omask = BIT(0) | BIT(1) | BIT(2) };
107*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_EVEREST: /* e.g. in rp3410 */
108*27407470SHelge Deller return (struct diva_info) { .nports = 3,
109*27407470SHelge Deller .omask = BIT(0) | BIT(2) | BIT(7) };
110*27407470SHelge Deller case PCI_DEVICE_ID_HP_DIVA_MAESTRO:
111*27407470SHelge Deller return (struct diva_info) { .nports = 4,
112*27407470SHelge Deller .omask = BIT(0) | BIT(1) | BIT(2) | BIT(7) };
113*27407470SHelge Deller }
114*27407470SHelge Deller g_assert_not_reached();
115*27407470SHelge Deller }
116*27407470SHelge Deller
117*27407470SHelge Deller
diva_pci_realize(PCIDevice * dev,Error ** errp)118*27407470SHelge Deller static void diva_pci_realize(PCIDevice *dev, Error **errp)
119*27407470SHelge Deller {
120*27407470SHelge Deller PCIDeviceClass *pc = PCI_DEVICE_GET_CLASS(dev);
121*27407470SHelge Deller PCIDivaSerialState *pci = DO_UPCAST(PCIDivaSerialState, dev, dev);
122*27407470SHelge Deller SerialState *s;
123*27407470SHelge Deller struct diva_info di = diva_get_diva_info(pc);
124*27407470SHelge Deller size_t i, offset = 0;
125*27407470SHelge Deller size_t portmask = di.omask;
126*27407470SHelge Deller
127*27407470SHelge Deller pci->dev.config[PCI_CLASS_PROG] = pci->prog_if;
128*27407470SHelge Deller pci->dev.config[PCI_INTERRUPT_PIN] = 0x01;
129*27407470SHelge Deller memory_region_init(&pci->membar, OBJECT(pci), "serial_ports", 4096);
130*27407470SHelge Deller pci_register_bar(&pci->dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &pci->membar);
131*27407470SHelge Deller pci->irqs = qemu_allocate_irqs(multi_serial_irq_mux, pci, di.nports);
132*27407470SHelge Deller
133*27407470SHelge Deller for (i = 0; i < di.nports; i++) {
134*27407470SHelge Deller s = pci->state + i;
135*27407470SHelge Deller if (!qdev_realize(DEVICE(s), NULL, errp)) {
136*27407470SHelge Deller diva_pci_exit(dev);
137*27407470SHelge Deller return;
138*27407470SHelge Deller }
139*27407470SHelge Deller s->irq = pci->irqs[i];
140*27407470SHelge Deller pci->name[i] = g_strdup_printf("uart #%zu", i + 1);
141*27407470SHelge Deller memory_region_init_io(&s->io, OBJECT(pci), &serial_io_ops, s,
142*27407470SHelge Deller pci->name[i], 8);
143*27407470SHelge Deller
144*27407470SHelge Deller /* calculate offset of given port based on bitmask */
145*27407470SHelge Deller while ((portmask & BIT(0)) == 0) {
146*27407470SHelge Deller offset += 8;
147*27407470SHelge Deller portmask >>= 1;
148*27407470SHelge Deller }
149*27407470SHelge Deller memory_region_add_subregion(&pci->membar, offset, &s->io);
150*27407470SHelge Deller offset += 8;
151*27407470SHelge Deller portmask >>= 1;
152*27407470SHelge Deller pci->ports++;
153*27407470SHelge Deller }
154*27407470SHelge Deller
155*27407470SHelge Deller /* mailbox bar */
156*27407470SHelge Deller memory_region_init(&pci->mailboxbar, OBJECT(pci), "mailbox", 128 * KiB);
157*27407470SHelge Deller pci_register_bar(&pci->dev, 1, PCI_BASE_ADDRESS_SPACE_MEMORY |
158*27407470SHelge Deller PCI_BASE_ADDRESS_MEM_PREFETCH, &pci->mailboxbar);
159*27407470SHelge Deller }
160*27407470SHelge Deller
161*27407470SHelge Deller static const VMStateDescription vmstate_pci_diva = {
162*27407470SHelge Deller .name = "pci-diva-serial",
163*27407470SHelge Deller .version_id = 1,
164*27407470SHelge Deller .minimum_version_id = 1,
165*27407470SHelge Deller .fields = (const VMStateField[]) {
166*27407470SHelge Deller VMSTATE_PCI_DEVICE(dev, PCIDivaSerialState),
167*27407470SHelge Deller VMSTATE_STRUCT_ARRAY(state, PCIDivaSerialState, PCI_SERIAL_MAX_PORTS,
168*27407470SHelge Deller 0, vmstate_serial, SerialState),
169*27407470SHelge Deller VMSTATE_UINT32_ARRAY(level, PCIDivaSerialState, PCI_SERIAL_MAX_PORTS),
170*27407470SHelge Deller VMSTATE_BOOL(disable, PCIDivaSerialState),
171*27407470SHelge Deller VMSTATE_END_OF_LIST()
172*27407470SHelge Deller }
173*27407470SHelge Deller };
174*27407470SHelge Deller
175*27407470SHelge Deller static const Property diva_serial_properties[] = {
176*27407470SHelge Deller DEFINE_PROP_BOOL("disable", PCIDivaSerialState, disable, false),
177*27407470SHelge Deller DEFINE_PROP_CHR("chardev1", PCIDivaSerialState, state[0].chr),
178*27407470SHelge Deller DEFINE_PROP_CHR("chardev2", PCIDivaSerialState, state[1].chr),
179*27407470SHelge Deller DEFINE_PROP_CHR("chardev3", PCIDivaSerialState, state[2].chr),
180*27407470SHelge Deller DEFINE_PROP_CHR("chardev4", PCIDivaSerialState, state[3].chr),
181*27407470SHelge Deller DEFINE_PROP_UINT8("prog_if", PCIDivaSerialState, prog_if, 0x02),
182*27407470SHelge Deller DEFINE_PROP_UINT32("subvendor", PCIDivaSerialState, subvendor,
183*27407470SHelge Deller PCI_DEVICE_ID_HP_DIVA_TOSCA1),
184*27407470SHelge Deller };
185*27407470SHelge Deller
diva_serial_class_initfn(ObjectClass * klass,void * data)186*27407470SHelge Deller static void diva_serial_class_initfn(ObjectClass *klass, void *data)
187*27407470SHelge Deller {
188*27407470SHelge Deller DeviceClass *dc = DEVICE_CLASS(klass);
189*27407470SHelge Deller PCIDeviceClass *pc = PCI_DEVICE_CLASS(klass);
190*27407470SHelge Deller pc->realize = diva_pci_realize;
191*27407470SHelge Deller pc->exit = diva_pci_exit;
192*27407470SHelge Deller pc->vendor_id = PCI_VENDOR_ID_HP;
193*27407470SHelge Deller pc->device_id = PCI_DEVICE_ID_HP_DIVA;
194*27407470SHelge Deller pc->subsystem_vendor_id = PCI_VENDOR_ID_HP;
195*27407470SHelge Deller pc->subsystem_id = PCI_DEVICE_ID_HP_DIVA_TOSCA1;
196*27407470SHelge Deller pc->revision = 3;
197*27407470SHelge Deller pc->class_id = PCI_CLASS_COMMUNICATION_SERIAL;
198*27407470SHelge Deller dc->vmsd = &vmstate_pci_diva;
199*27407470SHelge Deller device_class_set_props(dc, diva_serial_properties);
200*27407470SHelge Deller set_bit(DEVICE_CATEGORY_INPUT, dc->categories);
201*27407470SHelge Deller }
202*27407470SHelge Deller
diva_serial_init(Object * o)203*27407470SHelge Deller static void diva_serial_init(Object *o)
204*27407470SHelge Deller {
205*27407470SHelge Deller PCIDevice *dev = PCI_DEVICE(o);
206*27407470SHelge Deller PCIDivaSerialState *pms = DO_UPCAST(PCIDivaSerialState, dev, dev);
207*27407470SHelge Deller struct diva_info di = diva_get_diva_info(PCI_DEVICE_GET_CLASS(dev));
208*27407470SHelge Deller size_t i;
209*27407470SHelge Deller
210*27407470SHelge Deller for (i = 0; i < di.nports; i++) {
211*27407470SHelge Deller object_initialize_child(o, "serial[*]", &pms->state[i], TYPE_SERIAL);
212*27407470SHelge Deller }
213*27407470SHelge Deller }
214*27407470SHelge Deller
215*27407470SHelge Deller
216*27407470SHelge Deller /* Diva-aux is the driver for portion 0 of the multifunction PCI device */
217*27407470SHelge Deller
218*27407470SHelge Deller struct DivaAuxState {
219*27407470SHelge Deller PCIDevice dev;
220*27407470SHelge Deller MemoryRegion mem;
221*27407470SHelge Deller qemu_irq irq;
222*27407470SHelge Deller };
223*27407470SHelge Deller
224*27407470SHelge Deller #define TYPE_DIVA_AUX "diva-aux"
OBJECT_DECLARE_SIMPLE_TYPE(DivaAuxState,DIVA_AUX)225*27407470SHelge Deller OBJECT_DECLARE_SIMPLE_TYPE(DivaAuxState, DIVA_AUX)
226*27407470SHelge Deller
227*27407470SHelge Deller static void diva_aux_realize(PCIDevice *dev, Error **errp)
228*27407470SHelge Deller {
229*27407470SHelge Deller DivaAuxState *pci = DO_UPCAST(DivaAuxState, dev, dev);
230*27407470SHelge Deller
231*27407470SHelge Deller pci->dev.config[PCI_CLASS_PROG] = 0x02;
232*27407470SHelge Deller pci->dev.config[PCI_INTERRUPT_PIN] = 0x01;
233*27407470SHelge Deller pci->irq = pci_allocate_irq(&pci->dev);
234*27407470SHelge Deller
235*27407470SHelge Deller memory_region_init(&pci->mem, OBJECT(pci), "mem", 16);
236*27407470SHelge Deller pci_register_bar(&pci->dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &pci->mem);
237*27407470SHelge Deller }
238*27407470SHelge Deller
diva_aux_exit(PCIDevice * dev)239*27407470SHelge Deller static void diva_aux_exit(PCIDevice *dev)
240*27407470SHelge Deller {
241*27407470SHelge Deller DivaAuxState *pci = DO_UPCAST(DivaAuxState, dev, dev);
242*27407470SHelge Deller qemu_free_irq(pci->irq);
243*27407470SHelge Deller }
244*27407470SHelge Deller
diva_aux_class_initfn(ObjectClass * klass,void * data)245*27407470SHelge Deller static void diva_aux_class_initfn(ObjectClass *klass, void *data)
246*27407470SHelge Deller {
247*27407470SHelge Deller DeviceClass *dc = DEVICE_CLASS(klass);
248*27407470SHelge Deller PCIDeviceClass *pc = PCI_DEVICE_CLASS(klass);
249*27407470SHelge Deller pc->realize = diva_aux_realize;
250*27407470SHelge Deller pc->exit = diva_aux_exit;
251*27407470SHelge Deller pc->vendor_id = PCI_VENDOR_ID_HP;
252*27407470SHelge Deller pc->device_id = PCI_DEVICE_ID_HP_DIVA_AUX;
253*27407470SHelge Deller pc->subsystem_vendor_id = PCI_VENDOR_ID_HP;
254*27407470SHelge Deller pc->subsystem_id = 0x1291;
255*27407470SHelge Deller pc->revision = 1;
256*27407470SHelge Deller pc->class_id = PCI_CLASS_COMMUNICATION_MULTISERIAL;
257*27407470SHelge Deller set_bit(DEVICE_CATEGORY_MISC, dc->categories);
258*27407470SHelge Deller dc->user_creatable = false;
259*27407470SHelge Deller }
260*27407470SHelge Deller
diva_aux_init(Object * o)261*27407470SHelge Deller static void diva_aux_init(Object *o)
262*27407470SHelge Deller {
263*27407470SHelge Deller }
264*27407470SHelge Deller
265*27407470SHelge Deller static const TypeInfo diva_aux_info = {
266*27407470SHelge Deller .name = TYPE_DIVA_AUX,
267*27407470SHelge Deller .parent = TYPE_PCI_DEVICE,
268*27407470SHelge Deller .instance_size = sizeof(DivaAuxState),
269*27407470SHelge Deller .instance_init = diva_aux_init,
270*27407470SHelge Deller .class_init = diva_aux_class_initfn,
271*27407470SHelge Deller .interfaces = (InterfaceInfo[]) {
272*27407470SHelge Deller { INTERFACE_CONVENTIONAL_PCI_DEVICE },
273*27407470SHelge Deller { },
274*27407470SHelge Deller },
275*27407470SHelge Deller };
276*27407470SHelge Deller
277*27407470SHelge Deller
278*27407470SHelge Deller
279*27407470SHelge Deller static const TypeInfo diva_serial_pci_info = {
280*27407470SHelge Deller .name = "diva-gsp",
281*27407470SHelge Deller .parent = TYPE_PCI_DEVICE,
282*27407470SHelge Deller .instance_size = sizeof(PCIDivaSerialState),
283*27407470SHelge Deller .instance_init = diva_serial_init,
284*27407470SHelge Deller .class_init = diva_serial_class_initfn,
285*27407470SHelge Deller .interfaces = (InterfaceInfo[]) {
286*27407470SHelge Deller { INTERFACE_CONVENTIONAL_PCI_DEVICE },
287*27407470SHelge Deller { },
288*27407470SHelge Deller },
289*27407470SHelge Deller };
290*27407470SHelge Deller
diva_pci_register_type(void)291*27407470SHelge Deller static void diva_pci_register_type(void)
292*27407470SHelge Deller {
293*27407470SHelge Deller type_register_static(&diva_serial_pci_info);
294*27407470SHelge Deller type_register_static(&diva_aux_info);
295*27407470SHelge Deller }
296*27407470SHelge Deller
297*27407470SHelge Deller type_init(diva_pci_register_type)
298