xref: /openbmc/qemu/hw/misc/max78000_gcr.c (revision 0edc2afe0c8197bbcb98f948c609fb74c9b1ffd5)
1a017f53eSJackson Donaldson /*
2a017f53eSJackson Donaldson  * MAX78000 Global Control Registers
3a017f53eSJackson Donaldson  *
4a017f53eSJackson Donaldson  * Copyright (c) 2025 Jackson Donaldson <jcksn@duck.com>
5a017f53eSJackson Donaldson  *
6a017f53eSJackson Donaldson  * SPDX-License-Identifier: GPL-2.0-or-later
7a017f53eSJackson Donaldson  */
8a017f53eSJackson Donaldson 
9a017f53eSJackson Donaldson #include "qemu/osdep.h"
10a017f53eSJackson Donaldson #include "qemu/log.h"
11a017f53eSJackson Donaldson #include "trace.h"
12a017f53eSJackson Donaldson #include "hw/irq.h"
13a017f53eSJackson Donaldson #include "system/runstate.h"
14a017f53eSJackson Donaldson #include "migration/vmstate.h"
15a017f53eSJackson Donaldson #include "hw/qdev-properties.h"
16a017f53eSJackson Donaldson #include "hw/char/max78000_uart.h"
17069852d1SJackson Donaldson #include "hw/misc/max78000_trng.h"
18*33dfff7eSJackson Donaldson #include "hw/misc/max78000_aes.h"
19a017f53eSJackson Donaldson #include "hw/misc/max78000_gcr.h"
20a017f53eSJackson Donaldson 
21a017f53eSJackson Donaldson 
max78000_gcr_reset_hold(Object * obj,ResetType type)22a017f53eSJackson Donaldson static void max78000_gcr_reset_hold(Object *obj, ResetType type)
23a017f53eSJackson Donaldson {
24a017f53eSJackson Donaldson     DeviceState *dev = DEVICE(obj);
25a017f53eSJackson Donaldson     Max78000GcrState *s = MAX78000_GCR(dev);
26a017f53eSJackson Donaldson     s->sysctrl = 0x21002;
27a017f53eSJackson Donaldson     s->rst0 = 0;
28a017f53eSJackson Donaldson     /* All clocks are always ready */
29a017f53eSJackson Donaldson     s->clkctrl = 0x3e140008;
30a017f53eSJackson Donaldson     s->pm = 0x3f000;
31a017f53eSJackson Donaldson     s->pclkdiv = 0;
32a017f53eSJackson Donaldson     s->pclkdis0 = 0xffffffff;
33a017f53eSJackson Donaldson     s->memctrl = 0x5;
34a017f53eSJackson Donaldson     s->memz = 0;
35a017f53eSJackson Donaldson     s->sysst = 0;
36a017f53eSJackson Donaldson     s->rst1 = 0;
37a017f53eSJackson Donaldson     s->pckdis1 = 0xffffffff;
38a017f53eSJackson Donaldson     s->eventen = 0;
39a017f53eSJackson Donaldson     s->revision = 0xa1;
40a017f53eSJackson Donaldson     s->sysie = 0;
41a017f53eSJackson Donaldson     s->eccerr = 0;
42a017f53eSJackson Donaldson     s->ecced = 0;
43a017f53eSJackson Donaldson     s->eccie = 0;
44a017f53eSJackson Donaldson     s->eccaddr = 0;
45a017f53eSJackson Donaldson }
46a017f53eSJackson Donaldson 
max78000_gcr_read(void * opaque,hwaddr addr,unsigned int size)47a017f53eSJackson Donaldson static uint64_t max78000_gcr_read(void *opaque, hwaddr addr,
48a017f53eSJackson Donaldson                                      unsigned int size)
49a017f53eSJackson Donaldson {
50a017f53eSJackson Donaldson     Max78000GcrState *s = opaque;
51a017f53eSJackson Donaldson 
52a017f53eSJackson Donaldson     switch (addr) {
53a017f53eSJackson Donaldson     case SYSCTRL:
54a017f53eSJackson Donaldson         return s->sysctrl;
55a017f53eSJackson Donaldson 
56a017f53eSJackson Donaldson     case RST0:
57a017f53eSJackson Donaldson         return s->rst0;
58a017f53eSJackson Donaldson 
59a017f53eSJackson Donaldson     case CLKCTRL:
60a017f53eSJackson Donaldson         return s->clkctrl;
61a017f53eSJackson Donaldson 
62a017f53eSJackson Donaldson     case PM:
63a017f53eSJackson Donaldson         return s->pm;
64a017f53eSJackson Donaldson 
65a017f53eSJackson Donaldson     case PCLKDIV:
66a017f53eSJackson Donaldson         return s->pclkdiv;
67a017f53eSJackson Donaldson 
68a017f53eSJackson Donaldson     case PCLKDIS0:
69a017f53eSJackson Donaldson         return s->pclkdis0;
70a017f53eSJackson Donaldson 
71a017f53eSJackson Donaldson     case MEMCTRL:
72a017f53eSJackson Donaldson         return s->memctrl;
73a017f53eSJackson Donaldson 
74a017f53eSJackson Donaldson     case MEMZ:
75a017f53eSJackson Donaldson         return s->memz;
76a017f53eSJackson Donaldson 
77a017f53eSJackson Donaldson     case SYSST:
78a017f53eSJackson Donaldson         return s->sysst;
79a017f53eSJackson Donaldson 
80a017f53eSJackson Donaldson     case RST1:
81a017f53eSJackson Donaldson         return s->rst1;
82a017f53eSJackson Donaldson 
83a017f53eSJackson Donaldson     case PCKDIS1:
84a017f53eSJackson Donaldson         return s->pckdis1;
85a017f53eSJackson Donaldson 
86a017f53eSJackson Donaldson     case EVENTEN:
87a017f53eSJackson Donaldson         return s->eventen;
88a017f53eSJackson Donaldson 
89a017f53eSJackson Donaldson     case REVISION:
90a017f53eSJackson Donaldson         return s->revision;
91a017f53eSJackson Donaldson 
92a017f53eSJackson Donaldson     case SYSIE:
93a017f53eSJackson Donaldson         return s->sysie;
94a017f53eSJackson Donaldson 
95a017f53eSJackson Donaldson     case ECCERR:
96a017f53eSJackson Donaldson         return s->eccerr;
97a017f53eSJackson Donaldson 
98a017f53eSJackson Donaldson     case ECCED:
99a017f53eSJackson Donaldson         return s->ecced;
100a017f53eSJackson Donaldson 
101a017f53eSJackson Donaldson     case ECCIE:
102a017f53eSJackson Donaldson         return s->eccie;
103a017f53eSJackson Donaldson 
104a017f53eSJackson Donaldson     case ECCADDR:
105a017f53eSJackson Donaldson         return s->eccaddr;
106a017f53eSJackson Donaldson 
107a017f53eSJackson Donaldson     default:
108a017f53eSJackson Donaldson         qemu_log_mask(LOG_GUEST_ERROR, "%s: Bad offset 0x%"
109a017f53eSJackson Donaldson             HWADDR_PRIx "\n", __func__, addr);
110a017f53eSJackson Donaldson         return 0;
111a017f53eSJackson Donaldson 
112a017f53eSJackson Donaldson     }
113a017f53eSJackson Donaldson }
114a017f53eSJackson Donaldson 
max78000_gcr_write(void * opaque,hwaddr addr,uint64_t val64,unsigned int size)115a017f53eSJackson Donaldson static void max78000_gcr_write(void *opaque, hwaddr addr,
116a017f53eSJackson Donaldson                        uint64_t val64, unsigned int size)
117a017f53eSJackson Donaldson {
118a017f53eSJackson Donaldson     Max78000GcrState *s = opaque;
119a017f53eSJackson Donaldson     uint32_t val = val64;
120a017f53eSJackson Donaldson     uint8_t zero[0xc000] = {0};
121a017f53eSJackson Donaldson     switch (addr) {
122a017f53eSJackson Donaldson     case SYSCTRL:
123a017f53eSJackson Donaldson         /* Checksum calculations always pass immediately */
124a017f53eSJackson Donaldson         s->sysctrl = (val & 0x30000) | 0x1002;
125a017f53eSJackson Donaldson         break;
126a017f53eSJackson Donaldson 
127a017f53eSJackson Donaldson     case RST0:
128a017f53eSJackson Donaldson         if (val & SYSTEM_RESET) {
129a017f53eSJackson Donaldson             qemu_system_reset_request(SHUTDOWN_CAUSE_GUEST_RESET);
130a017f53eSJackson Donaldson         }
131a017f53eSJackson Donaldson         if (val & PERIPHERAL_RESET) {
132a017f53eSJackson Donaldson             /*
133a017f53eSJackson Donaldson              * Peripheral reset resets all peripherals. The CPU
134a017f53eSJackson Donaldson              * retains its state. The GPIO, watchdog timers, AoD,
135a017f53eSJackson Donaldson              * RAM retention, and general control registers (GCR),
136a017f53eSJackson Donaldson              * including the clock configuration, are unaffected.
137a017f53eSJackson Donaldson              */
138a017f53eSJackson Donaldson             val = UART2_RESET | UART1_RESET | UART0_RESET |
139a017f53eSJackson Donaldson                     ADC_RESET | CNN_RESET | TRNG_RESET |
140a017f53eSJackson Donaldson                     RTC_RESET | I2C0_RESET | SPI1_RESET |
141a017f53eSJackson Donaldson                     TMR3_RESET | TMR2_RESET | TMR1_RESET |
142a017f53eSJackson Donaldson                     TMR0_RESET | WDT0_RESET | DMA_RESET;
143a017f53eSJackson Donaldson         }
144a017f53eSJackson Donaldson         if (val & SOFT_RESET) {
145a017f53eSJackson Donaldson             /* Soft reset also resets GPIO */
146a017f53eSJackson Donaldson             val = UART2_RESET | UART1_RESET | UART0_RESET |
147a017f53eSJackson Donaldson                     ADC_RESET | CNN_RESET | TRNG_RESET |
148a017f53eSJackson Donaldson                     RTC_RESET | I2C0_RESET | SPI1_RESET |
149a017f53eSJackson Donaldson                     TMR3_RESET | TMR2_RESET | TMR1_RESET |
150a017f53eSJackson Donaldson                     TMR0_RESET | GPIO1_RESET | GPIO0_RESET |
151a017f53eSJackson Donaldson                     DMA_RESET;
152a017f53eSJackson Donaldson         }
153a017f53eSJackson Donaldson         if (val & UART2_RESET) {
154a017f53eSJackson Donaldson             device_cold_reset(s->uart2);
155a017f53eSJackson Donaldson         }
156a017f53eSJackson Donaldson         if (val & UART1_RESET) {
157a017f53eSJackson Donaldson             device_cold_reset(s->uart1);
158a017f53eSJackson Donaldson         }
159a017f53eSJackson Donaldson         if (val & UART0_RESET) {
160a017f53eSJackson Donaldson             device_cold_reset(s->uart0);
161a017f53eSJackson Donaldson         }
162069852d1SJackson Donaldson         if (val & TRNG_RESET) {
163069852d1SJackson Donaldson             device_cold_reset(s->trng);
164069852d1SJackson Donaldson         }
165*33dfff7eSJackson Donaldson         if (val & AES_RESET) {
166*33dfff7eSJackson Donaldson             device_cold_reset(s->aes);
167*33dfff7eSJackson Donaldson         }
168a017f53eSJackson Donaldson         /* TODO: As other devices are implemented, add them here */
169a017f53eSJackson Donaldson         break;
170a017f53eSJackson Donaldson 
171a017f53eSJackson Donaldson     case CLKCTRL:
172a017f53eSJackson Donaldson         s->clkctrl = val | SYSCLK_RDY;
173a017f53eSJackson Donaldson         break;
174a017f53eSJackson Donaldson 
175a017f53eSJackson Donaldson     case PM:
176a017f53eSJackson Donaldson         s->pm = val;
177a017f53eSJackson Donaldson         break;
178a017f53eSJackson Donaldson 
179a017f53eSJackson Donaldson     case PCLKDIV:
180a017f53eSJackson Donaldson         s->pclkdiv = val;
181a017f53eSJackson Donaldson         break;
182a017f53eSJackson Donaldson 
183a017f53eSJackson Donaldson     case PCLKDIS0:
184a017f53eSJackson Donaldson         s->pclkdis0 = val;
185a017f53eSJackson Donaldson         break;
186a017f53eSJackson Donaldson 
187a017f53eSJackson Donaldson     case MEMCTRL:
188a017f53eSJackson Donaldson         s->memctrl = val;
189a017f53eSJackson Donaldson         break;
190a017f53eSJackson Donaldson 
191a017f53eSJackson Donaldson     case MEMZ:
192a017f53eSJackson Donaldson         if (val & ram0) {
193a017f53eSJackson Donaldson             address_space_write(&s->sram_as, SYSRAM0_START,
194a017f53eSJackson Donaldson                                 MEMTXATTRS_UNSPECIFIED, zero, 0x8000);
195a017f53eSJackson Donaldson         }
196a017f53eSJackson Donaldson         if (val & ram1) {
197a017f53eSJackson Donaldson             address_space_write(&s->sram_as, SYSRAM1_START,
198a017f53eSJackson Donaldson                                 MEMTXATTRS_UNSPECIFIED, zero, 0x8000);
199a017f53eSJackson Donaldson         }
200a017f53eSJackson Donaldson         if (val & ram2) {
201a017f53eSJackson Donaldson             address_space_write(&s->sram_as, SYSRAM2_START,
202a017f53eSJackson Donaldson                                 MEMTXATTRS_UNSPECIFIED, zero, 0xC000);
203a017f53eSJackson Donaldson         }
204a017f53eSJackson Donaldson         if (val & ram3) {
205a017f53eSJackson Donaldson             address_space_write(&s->sram_as, SYSRAM3_START,
206a017f53eSJackson Donaldson                                 MEMTXATTRS_UNSPECIFIED, zero, 0x4000);
207a017f53eSJackson Donaldson         }
208a017f53eSJackson Donaldson         break;
209a017f53eSJackson Donaldson 
210a017f53eSJackson Donaldson     case SYSST:
211a017f53eSJackson Donaldson         s->sysst = val;
212a017f53eSJackson Donaldson         break;
213a017f53eSJackson Donaldson 
214a017f53eSJackson Donaldson     case RST1:
215a017f53eSJackson Donaldson         /* TODO: As other devices are implemented, add them here */
216a017f53eSJackson Donaldson         s->rst1 = val;
217a017f53eSJackson Donaldson         break;
218a017f53eSJackson Donaldson 
219a017f53eSJackson Donaldson     case PCKDIS1:
220a017f53eSJackson Donaldson         s->pckdis1 = val;
221a017f53eSJackson Donaldson         break;
222a017f53eSJackson Donaldson 
223a017f53eSJackson Donaldson     case EVENTEN:
224a017f53eSJackson Donaldson         s->eventen = val;
225a017f53eSJackson Donaldson         break;
226a017f53eSJackson Donaldson 
227a017f53eSJackson Donaldson     case REVISION:
228a017f53eSJackson Donaldson         s->revision = val;
229a017f53eSJackson Donaldson         break;
230a017f53eSJackson Donaldson 
231a017f53eSJackson Donaldson     case SYSIE:
232a017f53eSJackson Donaldson         s->sysie = val;
233a017f53eSJackson Donaldson         break;
234a017f53eSJackson Donaldson 
235a017f53eSJackson Donaldson     case ECCERR:
236a017f53eSJackson Donaldson         s->eccerr = val;
237a017f53eSJackson Donaldson         break;
238a017f53eSJackson Donaldson 
239a017f53eSJackson Donaldson     case ECCED:
240a017f53eSJackson Donaldson         s->ecced = val;
241a017f53eSJackson Donaldson         break;
242a017f53eSJackson Donaldson 
243a017f53eSJackson Donaldson     case ECCIE:
244a017f53eSJackson Donaldson         s->eccie = val;
245a017f53eSJackson Donaldson         break;
246a017f53eSJackson Donaldson 
247a017f53eSJackson Donaldson     case ECCADDR:
248a017f53eSJackson Donaldson         s->eccaddr = val;
249a017f53eSJackson Donaldson         break;
250a017f53eSJackson Donaldson 
251a017f53eSJackson Donaldson     default:
252a017f53eSJackson Donaldson         qemu_log_mask(LOG_GUEST_ERROR, "%s: Bad offset 0x%" HWADDR_PRIx "\n",
253a017f53eSJackson Donaldson                       __func__, addr);
254a017f53eSJackson Donaldson         break;
255a017f53eSJackson Donaldson 
256a017f53eSJackson Donaldson     }
257a017f53eSJackson Donaldson }
258a017f53eSJackson Donaldson 
259a017f53eSJackson Donaldson static const Property max78000_gcr_properties[] = {
260a017f53eSJackson Donaldson     DEFINE_PROP_LINK("sram", Max78000GcrState, sram,
261a017f53eSJackson Donaldson                      TYPE_MEMORY_REGION, MemoryRegion*),
262a017f53eSJackson Donaldson     DEFINE_PROP_LINK("uart0", Max78000GcrState, uart0,
263a017f53eSJackson Donaldson                      TYPE_MAX78000_UART, DeviceState*),
264a017f53eSJackson Donaldson     DEFINE_PROP_LINK("uart1", Max78000GcrState, uart1,
265a017f53eSJackson Donaldson                      TYPE_MAX78000_UART, DeviceState*),
266a017f53eSJackson Donaldson     DEFINE_PROP_LINK("uart2", Max78000GcrState, uart2,
267a017f53eSJackson Donaldson                      TYPE_MAX78000_UART, DeviceState*),
268069852d1SJackson Donaldson     DEFINE_PROP_LINK("trng", Max78000GcrState, trng,
269069852d1SJackson Donaldson                         TYPE_MAX78000_TRNG, DeviceState*),
270*33dfff7eSJackson Donaldson     DEFINE_PROP_LINK("aes", Max78000GcrState, aes,
271*33dfff7eSJackson Donaldson                         TYPE_MAX78000_AES, DeviceState*),
272a017f53eSJackson Donaldson };
273a017f53eSJackson Donaldson 
274a017f53eSJackson Donaldson static const MemoryRegionOps max78000_gcr_ops = {
275a017f53eSJackson Donaldson     .read = max78000_gcr_read,
276a017f53eSJackson Donaldson     .write = max78000_gcr_write,
277a017f53eSJackson Donaldson     .endianness = DEVICE_LITTLE_ENDIAN,
278a017f53eSJackson Donaldson     .valid.min_access_size = 4,
279a017f53eSJackson Donaldson     .valid.max_access_size = 4,
280a017f53eSJackson Donaldson };
281a017f53eSJackson Donaldson 
282a017f53eSJackson Donaldson static const VMStateDescription vmstate_max78000_gcr = {
283a017f53eSJackson Donaldson     .name = TYPE_MAX78000_GCR,
284a017f53eSJackson Donaldson     .version_id = 1,
285a017f53eSJackson Donaldson     .minimum_version_id = 1,
286a017f53eSJackson Donaldson     .fields = (const VMStateField[]) {
287a017f53eSJackson Donaldson         VMSTATE_UINT32(sysctrl, Max78000GcrState),
288a017f53eSJackson Donaldson         VMSTATE_UINT32(rst0, Max78000GcrState),
289a017f53eSJackson Donaldson         VMSTATE_UINT32(clkctrl, Max78000GcrState),
290a017f53eSJackson Donaldson         VMSTATE_UINT32(pm, Max78000GcrState),
291a017f53eSJackson Donaldson         VMSTATE_UINT32(pclkdiv, Max78000GcrState),
292a017f53eSJackson Donaldson         VMSTATE_UINT32(pclkdis0, Max78000GcrState),
293a017f53eSJackson Donaldson         VMSTATE_UINT32(memctrl, Max78000GcrState),
294a017f53eSJackson Donaldson         VMSTATE_UINT32(memz, Max78000GcrState),
295a017f53eSJackson Donaldson         VMSTATE_UINT32(sysst, Max78000GcrState),
296a017f53eSJackson Donaldson         VMSTATE_UINT32(rst1, Max78000GcrState),
297a017f53eSJackson Donaldson         VMSTATE_UINT32(pckdis1, Max78000GcrState),
298a017f53eSJackson Donaldson         VMSTATE_UINT32(eventen, Max78000GcrState),
299a017f53eSJackson Donaldson         VMSTATE_UINT32(revision, Max78000GcrState),
300a017f53eSJackson Donaldson         VMSTATE_UINT32(sysie, Max78000GcrState),
301a017f53eSJackson Donaldson         VMSTATE_UINT32(eccerr, Max78000GcrState),
302a017f53eSJackson Donaldson         VMSTATE_UINT32(ecced, Max78000GcrState),
303a017f53eSJackson Donaldson         VMSTATE_UINT32(eccie, Max78000GcrState),
304a017f53eSJackson Donaldson         VMSTATE_UINT32(eccaddr, Max78000GcrState),
305a017f53eSJackson Donaldson         VMSTATE_END_OF_LIST()
306a017f53eSJackson Donaldson     }
307a017f53eSJackson Donaldson };
308a017f53eSJackson Donaldson 
max78000_gcr_init(Object * obj)309a017f53eSJackson Donaldson static void max78000_gcr_init(Object *obj)
310a017f53eSJackson Donaldson {
311a017f53eSJackson Donaldson     Max78000GcrState *s = MAX78000_GCR(obj);
312a017f53eSJackson Donaldson 
313a017f53eSJackson Donaldson     memory_region_init_io(&s->mmio, obj, &max78000_gcr_ops, s,
314a017f53eSJackson Donaldson                           TYPE_MAX78000_GCR, 0x400);
315a017f53eSJackson Donaldson     sysbus_init_mmio(SYS_BUS_DEVICE(obj), &s->mmio);
316a017f53eSJackson Donaldson 
317a017f53eSJackson Donaldson }
318a017f53eSJackson Donaldson 
max78000_gcr_realize(DeviceState * dev,Error ** errp)319a017f53eSJackson Donaldson static void max78000_gcr_realize(DeviceState *dev, Error **errp)
320a017f53eSJackson Donaldson {
321a017f53eSJackson Donaldson     Max78000GcrState *s = MAX78000_GCR(dev);
322a017f53eSJackson Donaldson 
323a017f53eSJackson Donaldson     address_space_init(&s->sram_as, s->sram, "sram");
324a017f53eSJackson Donaldson }
325a017f53eSJackson Donaldson 
max78000_gcr_class_init(ObjectClass * klass,const void * data)326a017f53eSJackson Donaldson static void max78000_gcr_class_init(ObjectClass *klass, const void *data)
327a017f53eSJackson Donaldson {
328a017f53eSJackson Donaldson     DeviceClass *dc = DEVICE_CLASS(klass);
329a017f53eSJackson Donaldson     ResettableClass *rc = RESETTABLE_CLASS(klass);
330a017f53eSJackson Donaldson 
331a017f53eSJackson Donaldson     device_class_set_props(dc, max78000_gcr_properties);
332a017f53eSJackson Donaldson 
333a017f53eSJackson Donaldson     dc->realize = max78000_gcr_realize;
334a017f53eSJackson Donaldson     dc->vmsd = &vmstate_max78000_gcr;
335a017f53eSJackson Donaldson     rc->phases.hold = max78000_gcr_reset_hold;
336a017f53eSJackson Donaldson }
337a017f53eSJackson Donaldson 
338a017f53eSJackson Donaldson static const TypeInfo max78000_gcr_info = {
339a017f53eSJackson Donaldson     .name          = TYPE_MAX78000_GCR,
340a017f53eSJackson Donaldson     .parent        = TYPE_SYS_BUS_DEVICE,
341a017f53eSJackson Donaldson     .instance_size = sizeof(Max78000GcrState),
342a017f53eSJackson Donaldson     .instance_init = max78000_gcr_init,
343a017f53eSJackson Donaldson     .class_init     = max78000_gcr_class_init,
344a017f53eSJackson Donaldson };
345a017f53eSJackson Donaldson 
max78000_gcr_register_types(void)346a017f53eSJackson Donaldson static void max78000_gcr_register_types(void)
347a017f53eSJackson Donaldson {
348a017f53eSJackson Donaldson     type_register_static(&max78000_gcr_info);
349a017f53eSJackson Donaldson }
350a017f53eSJackson Donaldson 
351a017f53eSJackson Donaldson type_init(max78000_gcr_register_types)
352