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 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 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 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 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 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 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 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