1 /* 2 * Arm IoT Kit security controller 3 * 4 * Copyright (c) 2018 Linaro Limited 5 * Written by Peter Maydell 6 * 7 * This program is free software; you can redistribute it and/or modify 8 * it under the terms of the GNU General Public License version 2 or 9 * (at your option) any later version. 10 */ 11 12 #include "qemu/osdep.h" 13 #include "qemu/log.h" 14 #include "qapi/error.h" 15 #include "trace.h" 16 #include "hw/sysbus.h" 17 #include "hw/registerfields.h" 18 #include "hw/misc/iotkit-secctl.h" 19 20 /* Registers in the secure privilege control block */ 21 REG32(SECRESPCFG, 0x10) 22 REG32(NSCCFG, 0x14) 23 REG32(SECMPCINTSTATUS, 0x1c) 24 REG32(SECPPCINTSTAT, 0x20) 25 REG32(SECPPCINTCLR, 0x24) 26 REG32(SECPPCINTEN, 0x28) 27 REG32(SECMSCINTSTAT, 0x30) 28 REG32(SECMSCINTCLR, 0x34) 29 REG32(SECMSCINTEN, 0x38) 30 REG32(BRGINTSTAT, 0x40) 31 REG32(BRGINTCLR, 0x44) 32 REG32(BRGINTEN, 0x48) 33 REG32(AHBNSPPC0, 0x50) 34 REG32(AHBNSPPCEXP0, 0x60) 35 REG32(AHBNSPPCEXP1, 0x64) 36 REG32(AHBNSPPCEXP2, 0x68) 37 REG32(AHBNSPPCEXP3, 0x6c) 38 REG32(APBNSPPC0, 0x70) 39 REG32(APBNSPPC1, 0x74) 40 REG32(APBNSPPCEXP0, 0x80) 41 REG32(APBNSPPCEXP1, 0x84) 42 REG32(APBNSPPCEXP2, 0x88) 43 REG32(APBNSPPCEXP3, 0x8c) 44 REG32(AHBSPPPC0, 0x90) 45 REG32(AHBSPPPCEXP0, 0xa0) 46 REG32(AHBSPPPCEXP1, 0xa4) 47 REG32(AHBSPPPCEXP2, 0xa8) 48 REG32(AHBSPPPCEXP3, 0xac) 49 REG32(APBSPPPC0, 0xb0) 50 REG32(APBSPPPC1, 0xb4) 51 REG32(APBSPPPCEXP0, 0xc0) 52 REG32(APBSPPPCEXP1, 0xc4) 53 REG32(APBSPPPCEXP2, 0xc8) 54 REG32(APBSPPPCEXP3, 0xcc) 55 REG32(NSMSCEXP, 0xd0) 56 REG32(PID4, 0xfd0) 57 REG32(PID5, 0xfd4) 58 REG32(PID6, 0xfd8) 59 REG32(PID7, 0xfdc) 60 REG32(PID0, 0xfe0) 61 REG32(PID1, 0xfe4) 62 REG32(PID2, 0xfe8) 63 REG32(PID3, 0xfec) 64 REG32(CID0, 0xff0) 65 REG32(CID1, 0xff4) 66 REG32(CID2, 0xff8) 67 REG32(CID3, 0xffc) 68 69 /* Registers in the non-secure privilege control block */ 70 REG32(AHBNSPPPC0, 0x90) 71 REG32(AHBNSPPPCEXP0, 0xa0) 72 REG32(AHBNSPPPCEXP1, 0xa4) 73 REG32(AHBNSPPPCEXP2, 0xa8) 74 REG32(AHBNSPPPCEXP3, 0xac) 75 REG32(APBNSPPPC0, 0xb0) 76 REG32(APBNSPPPC1, 0xb4) 77 REG32(APBNSPPPCEXP0, 0xc0) 78 REG32(APBNSPPPCEXP1, 0xc4) 79 REG32(APBNSPPPCEXP2, 0xc8) 80 REG32(APBNSPPPCEXP3, 0xcc) 81 /* PID and CID registers are also present in the NS block */ 82 83 static const uint8_t iotkit_secctl_s_idregs[] = { 84 0x04, 0x00, 0x00, 0x00, 85 0x52, 0xb8, 0x0b, 0x00, 86 0x0d, 0xf0, 0x05, 0xb1, 87 }; 88 89 static const uint8_t iotkit_secctl_ns_idregs[] = { 90 0x04, 0x00, 0x00, 0x00, 91 0x53, 0xb8, 0x0b, 0x00, 92 0x0d, 0xf0, 0x05, 0xb1, 93 }; 94 95 /* The register sets for the various PPCs (AHB internal, APB internal, 96 * AHB expansion, APB expansion) are all set up so that they are 97 * in 16-aligned blocks so offsets 0xN0, 0xN4, 0xN8, 0xNC are PPCs 98 * 0, 1, 2, 3 of that type, so we can convert a register address offset 99 * into an an index into a PPC array easily. 100 */ 101 static inline int offset_to_ppc_idx(uint32_t offset) 102 { 103 return extract32(offset, 2, 2); 104 } 105 106 typedef void PerPPCFunction(IoTKitSecCtlPPC *ppc); 107 108 static void foreach_ppc(IoTKitSecCtl *s, PerPPCFunction *fn) 109 { 110 int i; 111 112 for (i = 0; i < IOTS_NUM_APB_PPC; i++) { 113 fn(&s->apb[i]); 114 } 115 for (i = 0; i < IOTS_NUM_APB_EXP_PPC; i++) { 116 fn(&s->apbexp[i]); 117 } 118 for (i = 0; i < IOTS_NUM_AHB_EXP_PPC; i++) { 119 fn(&s->ahbexp[i]); 120 } 121 } 122 123 static MemTxResult iotkit_secctl_s_read(void *opaque, hwaddr addr, 124 uint64_t *pdata, 125 unsigned size, MemTxAttrs attrs) 126 { 127 uint64_t r; 128 uint32_t offset = addr & ~0x3; 129 IoTKitSecCtl *s = IOTKIT_SECCTL(opaque); 130 131 switch (offset) { 132 case A_AHBNSPPC0: 133 case A_AHBSPPPC0: 134 r = 0; 135 break; 136 case A_SECRESPCFG: 137 r = s->secrespcfg; 138 break; 139 case A_NSCCFG: 140 r = s->nsccfg; 141 break; 142 case A_SECMPCINTSTATUS: 143 r = s->mpcintstatus; 144 break; 145 case A_SECPPCINTSTAT: 146 r = s->secppcintstat; 147 break; 148 case A_SECPPCINTEN: 149 r = s->secppcinten; 150 break; 151 case A_BRGINTSTAT: 152 /* QEMU's bus fabric can never report errors as it doesn't buffer 153 * writes, so we never report bridge interrupts. 154 */ 155 r = 0; 156 break; 157 case A_BRGINTEN: 158 r = s->brginten; 159 break; 160 case A_AHBNSPPCEXP0: 161 case A_AHBNSPPCEXP1: 162 case A_AHBNSPPCEXP2: 163 case A_AHBNSPPCEXP3: 164 r = s->ahbexp[offset_to_ppc_idx(offset)].ns; 165 break; 166 case A_APBNSPPC0: 167 case A_APBNSPPC1: 168 r = s->apb[offset_to_ppc_idx(offset)].ns; 169 break; 170 case A_APBNSPPCEXP0: 171 case A_APBNSPPCEXP1: 172 case A_APBNSPPCEXP2: 173 case A_APBNSPPCEXP3: 174 r = s->apbexp[offset_to_ppc_idx(offset)].ns; 175 break; 176 case A_AHBSPPPCEXP0: 177 case A_AHBSPPPCEXP1: 178 case A_AHBSPPPCEXP2: 179 case A_AHBSPPPCEXP3: 180 r = s->apbexp[offset_to_ppc_idx(offset)].sp; 181 break; 182 case A_APBSPPPC0: 183 case A_APBSPPPC1: 184 r = s->apb[offset_to_ppc_idx(offset)].sp; 185 break; 186 case A_APBSPPPCEXP0: 187 case A_APBSPPPCEXP1: 188 case A_APBSPPPCEXP2: 189 case A_APBSPPPCEXP3: 190 r = s->apbexp[offset_to_ppc_idx(offset)].sp; 191 break; 192 case A_SECMSCINTSTAT: 193 r = s->secmscintstat; 194 break; 195 case A_SECMSCINTEN: 196 r = s->secmscinten; 197 break; 198 case A_NSMSCEXP: 199 r = s->nsmscexp; 200 break; 201 case A_PID4: 202 case A_PID5: 203 case A_PID6: 204 case A_PID7: 205 case A_PID0: 206 case A_PID1: 207 case A_PID2: 208 case A_PID3: 209 case A_CID0: 210 case A_CID1: 211 case A_CID2: 212 case A_CID3: 213 r = iotkit_secctl_s_idregs[(offset - A_PID4) / 4]; 214 break; 215 case A_SECPPCINTCLR: 216 case A_SECMSCINTCLR: 217 case A_BRGINTCLR: 218 qemu_log_mask(LOG_GUEST_ERROR, 219 "IotKit SecCtl S block read: write-only offset 0x%x\n", 220 offset); 221 r = 0; 222 break; 223 default: 224 qemu_log_mask(LOG_GUEST_ERROR, 225 "IotKit SecCtl S block read: bad offset 0x%x\n", offset); 226 r = 0; 227 break; 228 } 229 230 if (size != 4) { 231 /* None of our registers are access-sensitive, so just pull the right 232 * byte out of the word read result. 233 */ 234 r = extract32(r, (addr & 3) * 8, size * 8); 235 } 236 237 trace_iotkit_secctl_s_read(offset, r, size); 238 *pdata = r; 239 return MEMTX_OK; 240 } 241 242 static void iotkit_secctl_update_ppc_ap(IoTKitSecCtlPPC *ppc) 243 { 244 int i; 245 246 for (i = 0; i < ppc->numports; i++) { 247 bool v; 248 249 if (extract32(ppc->ns, i, 1)) { 250 v = extract32(ppc->nsp, i, 1); 251 } else { 252 v = extract32(ppc->sp, i, 1); 253 } 254 qemu_set_irq(ppc->ap[i], v); 255 } 256 } 257 258 static void iotkit_secctl_ppc_ns_write(IoTKitSecCtlPPC *ppc, uint32_t value) 259 { 260 int i; 261 262 ppc->ns = value & MAKE_64BIT_MASK(0, ppc->numports); 263 for (i = 0; i < ppc->numports; i++) { 264 qemu_set_irq(ppc->nonsec[i], extract32(ppc->ns, i, 1)); 265 } 266 iotkit_secctl_update_ppc_ap(ppc); 267 } 268 269 static void iotkit_secctl_ppc_sp_write(IoTKitSecCtlPPC *ppc, uint32_t value) 270 { 271 ppc->sp = value & MAKE_64BIT_MASK(0, ppc->numports); 272 iotkit_secctl_update_ppc_ap(ppc); 273 } 274 275 static void iotkit_secctl_ppc_nsp_write(IoTKitSecCtlPPC *ppc, uint32_t value) 276 { 277 ppc->nsp = value & MAKE_64BIT_MASK(0, ppc->numports); 278 iotkit_secctl_update_ppc_ap(ppc); 279 } 280 281 static void iotkit_secctl_ppc_update_irq_clear(IoTKitSecCtlPPC *ppc) 282 { 283 uint32_t value = ppc->parent->secppcintstat; 284 285 qemu_set_irq(ppc->irq_clear, extract32(value, ppc->irq_bit_offset, 1)); 286 } 287 288 static void iotkit_secctl_ppc_update_irq_enable(IoTKitSecCtlPPC *ppc) 289 { 290 uint32_t value = ppc->parent->secppcinten; 291 292 qemu_set_irq(ppc->irq_enable, extract32(value, ppc->irq_bit_offset, 1)); 293 } 294 295 static void iotkit_secctl_update_mscexp_irqs(qemu_irq *msc_irqs, uint32_t value) 296 { 297 int i; 298 299 for (i = 0; i < IOTS_NUM_EXP_MSC; i++) { 300 qemu_set_irq(msc_irqs[i], extract32(value, i + 16, 1)); 301 } 302 } 303 304 static void iotkit_secctl_update_msc_irq(IoTKitSecCtl *s) 305 { 306 /* Update the combined MSC IRQ, based on S_MSCEXP_STATUS and S_MSCEXP_EN */ 307 bool level = s->secmscintstat & s->secmscinten; 308 309 qemu_set_irq(s->msc_irq, level); 310 } 311 312 static MemTxResult iotkit_secctl_s_write(void *opaque, hwaddr addr, 313 uint64_t value, 314 unsigned size, MemTxAttrs attrs) 315 { 316 IoTKitSecCtl *s = IOTKIT_SECCTL(opaque); 317 uint32_t offset = addr; 318 IoTKitSecCtlPPC *ppc; 319 320 trace_iotkit_secctl_s_write(offset, value, size); 321 322 if (size != 4) { 323 /* Byte and halfword writes are ignored */ 324 qemu_log_mask(LOG_GUEST_ERROR, 325 "IotKit SecCtl S block write: bad size, ignored\n"); 326 return MEMTX_OK; 327 } 328 329 switch (offset) { 330 case A_NSCCFG: 331 s->nsccfg = value & 3; 332 qemu_set_irq(s->nsc_cfg_irq, s->nsccfg); 333 break; 334 case A_SECRESPCFG: 335 value &= 1; 336 s->secrespcfg = value; 337 qemu_set_irq(s->sec_resp_cfg, s->secrespcfg); 338 break; 339 case A_SECPPCINTCLR: 340 value &= 0x00f000f3; 341 foreach_ppc(s, iotkit_secctl_ppc_update_irq_clear); 342 break; 343 case A_SECPPCINTEN: 344 s->secppcinten = value & 0x00f000f3; 345 foreach_ppc(s, iotkit_secctl_ppc_update_irq_enable); 346 break; 347 case A_BRGINTCLR: 348 break; 349 case A_BRGINTEN: 350 s->brginten = value & 0xffff0000; 351 break; 352 case A_AHBNSPPCEXP0: 353 case A_AHBNSPPCEXP1: 354 case A_AHBNSPPCEXP2: 355 case A_AHBNSPPCEXP3: 356 ppc = &s->ahbexp[offset_to_ppc_idx(offset)]; 357 iotkit_secctl_ppc_ns_write(ppc, value); 358 break; 359 case A_APBNSPPC0: 360 case A_APBNSPPC1: 361 ppc = &s->apb[offset_to_ppc_idx(offset)]; 362 iotkit_secctl_ppc_ns_write(ppc, value); 363 break; 364 case A_APBNSPPCEXP0: 365 case A_APBNSPPCEXP1: 366 case A_APBNSPPCEXP2: 367 case A_APBNSPPCEXP3: 368 ppc = &s->apbexp[offset_to_ppc_idx(offset)]; 369 iotkit_secctl_ppc_ns_write(ppc, value); 370 break; 371 case A_AHBSPPPCEXP0: 372 case A_AHBSPPPCEXP1: 373 case A_AHBSPPPCEXP2: 374 case A_AHBSPPPCEXP3: 375 ppc = &s->ahbexp[offset_to_ppc_idx(offset)]; 376 iotkit_secctl_ppc_sp_write(ppc, value); 377 break; 378 case A_APBSPPPC0: 379 case A_APBSPPPC1: 380 ppc = &s->apb[offset_to_ppc_idx(offset)]; 381 iotkit_secctl_ppc_sp_write(ppc, value); 382 break; 383 case A_APBSPPPCEXP0: 384 case A_APBSPPPCEXP1: 385 case A_APBSPPPCEXP2: 386 case A_APBSPPPCEXP3: 387 ppc = &s->apbexp[offset_to_ppc_idx(offset)]; 388 iotkit_secctl_ppc_sp_write(ppc, value); 389 break; 390 case A_SECMSCINTCLR: 391 iotkit_secctl_update_mscexp_irqs(s->mscexp_clear, value); 392 break; 393 case A_SECMSCINTEN: 394 s->secmscinten = value; 395 iotkit_secctl_update_msc_irq(s); 396 break; 397 case A_NSMSCEXP: 398 s->nsmscexp = value; 399 iotkit_secctl_update_mscexp_irqs(s->mscexp_ns, value); 400 break; 401 case A_SECMPCINTSTATUS: 402 case A_SECPPCINTSTAT: 403 case A_SECMSCINTSTAT: 404 case A_BRGINTSTAT: 405 case A_AHBNSPPC0: 406 case A_AHBSPPPC0: 407 case A_PID4: 408 case A_PID5: 409 case A_PID6: 410 case A_PID7: 411 case A_PID0: 412 case A_PID1: 413 case A_PID2: 414 case A_PID3: 415 case A_CID0: 416 case A_CID1: 417 case A_CID2: 418 case A_CID3: 419 qemu_log_mask(LOG_GUEST_ERROR, 420 "IoTKit SecCtl S block write: " 421 "read-only offset 0x%x\n", offset); 422 break; 423 default: 424 qemu_log_mask(LOG_GUEST_ERROR, 425 "IotKit SecCtl S block write: bad offset 0x%x\n", 426 offset); 427 break; 428 } 429 430 return MEMTX_OK; 431 } 432 433 static MemTxResult iotkit_secctl_ns_read(void *opaque, hwaddr addr, 434 uint64_t *pdata, 435 unsigned size, MemTxAttrs attrs) 436 { 437 IoTKitSecCtl *s = IOTKIT_SECCTL(opaque); 438 uint64_t r; 439 uint32_t offset = addr & ~0x3; 440 441 switch (offset) { 442 case A_AHBNSPPPC0: 443 r = 0; 444 break; 445 case A_AHBNSPPPCEXP0: 446 case A_AHBNSPPPCEXP1: 447 case A_AHBNSPPPCEXP2: 448 case A_AHBNSPPPCEXP3: 449 r = s->ahbexp[offset_to_ppc_idx(offset)].nsp; 450 break; 451 case A_APBNSPPPC0: 452 case A_APBNSPPPC1: 453 r = s->apb[offset_to_ppc_idx(offset)].nsp; 454 break; 455 case A_APBNSPPPCEXP0: 456 case A_APBNSPPPCEXP1: 457 case A_APBNSPPPCEXP2: 458 case A_APBNSPPPCEXP3: 459 r = s->apbexp[offset_to_ppc_idx(offset)].nsp; 460 break; 461 case A_PID4: 462 case A_PID5: 463 case A_PID6: 464 case A_PID7: 465 case A_PID0: 466 case A_PID1: 467 case A_PID2: 468 case A_PID3: 469 case A_CID0: 470 case A_CID1: 471 case A_CID2: 472 case A_CID3: 473 r = iotkit_secctl_ns_idregs[(offset - A_PID4) / 4]; 474 break; 475 default: 476 qemu_log_mask(LOG_GUEST_ERROR, 477 "IotKit SecCtl NS block write: bad offset 0x%x\n", 478 offset); 479 r = 0; 480 break; 481 } 482 483 if (size != 4) { 484 /* None of our registers are access-sensitive, so just pull the right 485 * byte out of the word read result. 486 */ 487 r = extract32(r, (addr & 3) * 8, size * 8); 488 } 489 490 trace_iotkit_secctl_ns_read(offset, r, size); 491 *pdata = r; 492 return MEMTX_OK; 493 } 494 495 static MemTxResult iotkit_secctl_ns_write(void *opaque, hwaddr addr, 496 uint64_t value, 497 unsigned size, MemTxAttrs attrs) 498 { 499 IoTKitSecCtl *s = IOTKIT_SECCTL(opaque); 500 uint32_t offset = addr; 501 IoTKitSecCtlPPC *ppc; 502 503 trace_iotkit_secctl_ns_write(offset, value, size); 504 505 if (size != 4) { 506 /* Byte and halfword writes are ignored */ 507 qemu_log_mask(LOG_GUEST_ERROR, 508 "IotKit SecCtl NS block write: bad size, ignored\n"); 509 return MEMTX_OK; 510 } 511 512 switch (offset) { 513 case A_AHBNSPPPCEXP0: 514 case A_AHBNSPPPCEXP1: 515 case A_AHBNSPPPCEXP2: 516 case A_AHBNSPPPCEXP3: 517 ppc = &s->ahbexp[offset_to_ppc_idx(offset)]; 518 iotkit_secctl_ppc_nsp_write(ppc, value); 519 break; 520 case A_APBNSPPPC0: 521 case A_APBNSPPPC1: 522 ppc = &s->apb[offset_to_ppc_idx(offset)]; 523 iotkit_secctl_ppc_nsp_write(ppc, value); 524 break; 525 case A_APBNSPPPCEXP0: 526 case A_APBNSPPPCEXP1: 527 case A_APBNSPPPCEXP2: 528 case A_APBNSPPPCEXP3: 529 ppc = &s->apbexp[offset_to_ppc_idx(offset)]; 530 iotkit_secctl_ppc_nsp_write(ppc, value); 531 break; 532 case A_AHBNSPPPC0: 533 case A_PID4: 534 case A_PID5: 535 case A_PID6: 536 case A_PID7: 537 case A_PID0: 538 case A_PID1: 539 case A_PID2: 540 case A_PID3: 541 case A_CID0: 542 case A_CID1: 543 case A_CID2: 544 case A_CID3: 545 qemu_log_mask(LOG_GUEST_ERROR, 546 "IoTKit SecCtl NS block write: " 547 "read-only offset 0x%x\n", offset); 548 break; 549 default: 550 qemu_log_mask(LOG_GUEST_ERROR, 551 "IotKit SecCtl NS block write: bad offset 0x%x\n", 552 offset); 553 break; 554 } 555 556 return MEMTX_OK; 557 } 558 559 static const MemoryRegionOps iotkit_secctl_s_ops = { 560 .read_with_attrs = iotkit_secctl_s_read, 561 .write_with_attrs = iotkit_secctl_s_write, 562 .endianness = DEVICE_LITTLE_ENDIAN, 563 .valid.min_access_size = 1, 564 .valid.max_access_size = 4, 565 .impl.min_access_size = 1, 566 .impl.max_access_size = 4, 567 }; 568 569 static const MemoryRegionOps iotkit_secctl_ns_ops = { 570 .read_with_attrs = iotkit_secctl_ns_read, 571 .write_with_attrs = iotkit_secctl_ns_write, 572 .endianness = DEVICE_LITTLE_ENDIAN, 573 .valid.min_access_size = 1, 574 .valid.max_access_size = 4, 575 .impl.min_access_size = 1, 576 .impl.max_access_size = 4, 577 }; 578 579 static void iotkit_secctl_reset_ppc(IoTKitSecCtlPPC *ppc) 580 { 581 ppc->ns = 0; 582 ppc->sp = 0; 583 ppc->nsp = 0; 584 } 585 586 static void iotkit_secctl_reset(DeviceState *dev) 587 { 588 IoTKitSecCtl *s = IOTKIT_SECCTL(dev); 589 590 s->secppcintstat = 0; 591 s->secppcinten = 0; 592 s->secrespcfg = 0; 593 s->nsccfg = 0; 594 s->brginten = 0; 595 596 foreach_ppc(s, iotkit_secctl_reset_ppc); 597 } 598 599 static void iotkit_secctl_mpc_status(void *opaque, int n, int level) 600 { 601 IoTKitSecCtl *s = IOTKIT_SECCTL(opaque); 602 603 s->mpcintstatus = deposit32(s->mpcintstatus, 0, 1, !!level); 604 } 605 606 static void iotkit_secctl_mpcexp_status(void *opaque, int n, int level) 607 { 608 IoTKitSecCtl *s = IOTKIT_SECCTL(opaque); 609 610 s->mpcintstatus = deposit32(s->mpcintstatus, n + 16, 1, !!level); 611 } 612 613 static void iotkit_secctl_mscexp_status(void *opaque, int n, int level) 614 { 615 IoTKitSecCtl *s = IOTKIT_SECCTL(opaque); 616 617 s->secmscintstat = deposit32(s->secmscintstat, n + 16, 1, !!level); 618 iotkit_secctl_update_msc_irq(s); 619 } 620 621 static void iotkit_secctl_ppc_irqstatus(void *opaque, int n, int level) 622 { 623 IoTKitSecCtlPPC *ppc = opaque; 624 IoTKitSecCtl *s = IOTKIT_SECCTL(ppc->parent); 625 int irqbit = ppc->irq_bit_offset + n; 626 627 s->secppcintstat = deposit32(s->secppcintstat, irqbit, 1, level); 628 } 629 630 static void iotkit_secctl_init_ppc(IoTKitSecCtl *s, 631 IoTKitSecCtlPPC *ppc, 632 const char *name, 633 int numports, 634 int irq_bit_offset) 635 { 636 char *gpioname; 637 DeviceState *dev = DEVICE(s); 638 639 ppc->numports = numports; 640 ppc->irq_bit_offset = irq_bit_offset; 641 ppc->parent = s; 642 643 gpioname = g_strdup_printf("%s_nonsec", name); 644 qdev_init_gpio_out_named(dev, ppc->nonsec, gpioname, numports); 645 g_free(gpioname); 646 gpioname = g_strdup_printf("%s_ap", name); 647 qdev_init_gpio_out_named(dev, ppc->ap, gpioname, numports); 648 g_free(gpioname); 649 gpioname = g_strdup_printf("%s_irq_enable", name); 650 qdev_init_gpio_out_named(dev, &ppc->irq_enable, gpioname, 1); 651 g_free(gpioname); 652 gpioname = g_strdup_printf("%s_irq_clear", name); 653 qdev_init_gpio_out_named(dev, &ppc->irq_clear, gpioname, 1); 654 g_free(gpioname); 655 gpioname = g_strdup_printf("%s_irq_status", name); 656 qdev_init_gpio_in_named_with_opaque(dev, iotkit_secctl_ppc_irqstatus, 657 ppc, gpioname, 1); 658 g_free(gpioname); 659 } 660 661 static void iotkit_secctl_init(Object *obj) 662 { 663 IoTKitSecCtl *s = IOTKIT_SECCTL(obj); 664 SysBusDevice *sbd = SYS_BUS_DEVICE(obj); 665 DeviceState *dev = DEVICE(obj); 666 int i; 667 668 iotkit_secctl_init_ppc(s, &s->apb[0], "apb_ppc0", 669 IOTS_APB_PPC0_NUM_PORTS, 0); 670 iotkit_secctl_init_ppc(s, &s->apb[1], "apb_ppc1", 671 IOTS_APB_PPC1_NUM_PORTS, 1); 672 673 for (i = 0; i < IOTS_NUM_APB_EXP_PPC; i++) { 674 IoTKitSecCtlPPC *ppc = &s->apbexp[i]; 675 char *ppcname = g_strdup_printf("apb_ppcexp%d", i); 676 iotkit_secctl_init_ppc(s, ppc, ppcname, IOTS_PPC_NUM_PORTS, 4 + i); 677 g_free(ppcname); 678 } 679 for (i = 0; i < IOTS_NUM_AHB_EXP_PPC; i++) { 680 IoTKitSecCtlPPC *ppc = &s->ahbexp[i]; 681 char *ppcname = g_strdup_printf("ahb_ppcexp%d", i); 682 iotkit_secctl_init_ppc(s, ppc, ppcname, IOTS_PPC_NUM_PORTS, 20 + i); 683 g_free(ppcname); 684 } 685 686 qdev_init_gpio_out_named(dev, &s->sec_resp_cfg, "sec_resp_cfg", 1); 687 qdev_init_gpio_out_named(dev, &s->nsc_cfg_irq, "nsc_cfg", 1); 688 689 qdev_init_gpio_in_named(dev, iotkit_secctl_mpc_status, "mpc_status", 1); 690 qdev_init_gpio_in_named(dev, iotkit_secctl_mpcexp_status, 691 "mpcexp_status", IOTS_NUM_EXP_MPC); 692 693 qdev_init_gpio_in_named(dev, iotkit_secctl_mscexp_status, 694 "mscexp_status", IOTS_NUM_EXP_MSC); 695 qdev_init_gpio_out_named(dev, s->mscexp_clear, "mscexp_clear", 696 IOTS_NUM_EXP_MSC); 697 qdev_init_gpio_out_named(dev, s->mscexp_ns, "mscexp_ns", 698 IOTS_NUM_EXP_MSC); 699 qdev_init_gpio_out_named(dev, &s->msc_irq, "msc_irq", 1); 700 701 memory_region_init_io(&s->s_regs, obj, &iotkit_secctl_s_ops, 702 s, "iotkit-secctl-s-regs", 0x1000); 703 memory_region_init_io(&s->ns_regs, obj, &iotkit_secctl_ns_ops, 704 s, "iotkit-secctl-ns-regs", 0x1000); 705 sysbus_init_mmio(sbd, &s->s_regs); 706 sysbus_init_mmio(sbd, &s->ns_regs); 707 } 708 709 static const VMStateDescription iotkit_secctl_ppc_vmstate = { 710 .name = "iotkit-secctl-ppc", 711 .version_id = 1, 712 .minimum_version_id = 1, 713 .fields = (VMStateField[]) { 714 VMSTATE_UINT32(ns, IoTKitSecCtlPPC), 715 VMSTATE_UINT32(sp, IoTKitSecCtlPPC), 716 VMSTATE_UINT32(nsp, IoTKitSecCtlPPC), 717 VMSTATE_END_OF_LIST() 718 } 719 }; 720 721 static const VMStateDescription iotkit_secctl_mpcintstatus_vmstate = { 722 .name = "iotkit-secctl-mpcintstatus", 723 .version_id = 1, 724 .minimum_version_id = 1, 725 .fields = (VMStateField[]) { 726 VMSTATE_UINT32(mpcintstatus, IoTKitSecCtl), 727 VMSTATE_END_OF_LIST() 728 } 729 }; 730 731 static bool needed_always(void *opaque) 732 { 733 return true; 734 } 735 736 static const VMStateDescription iotkit_secctl_msc_vmstate = { 737 .name = "iotkit-secctl/msc", 738 .version_id = 1, 739 .minimum_version_id = 1, 740 .needed = needed_always, 741 .fields = (VMStateField[]) { 742 VMSTATE_UINT32(secmscintstat, IoTKitSecCtl), 743 VMSTATE_UINT32(secmscinten, IoTKitSecCtl), 744 VMSTATE_UINT32(nsmscexp, IoTKitSecCtl), 745 VMSTATE_END_OF_LIST() 746 } 747 }; 748 749 static const VMStateDescription iotkit_secctl_vmstate = { 750 .name = "iotkit-secctl", 751 .version_id = 1, 752 .minimum_version_id = 1, 753 .fields = (VMStateField[]) { 754 VMSTATE_UINT32(secppcintstat, IoTKitSecCtl), 755 VMSTATE_UINT32(secppcinten, IoTKitSecCtl), 756 VMSTATE_UINT32(secrespcfg, IoTKitSecCtl), 757 VMSTATE_UINT32(nsccfg, IoTKitSecCtl), 758 VMSTATE_UINT32(brginten, IoTKitSecCtl), 759 VMSTATE_STRUCT_ARRAY(apb, IoTKitSecCtl, IOTS_NUM_APB_PPC, 1, 760 iotkit_secctl_ppc_vmstate, IoTKitSecCtlPPC), 761 VMSTATE_STRUCT_ARRAY(apbexp, IoTKitSecCtl, IOTS_NUM_APB_EXP_PPC, 1, 762 iotkit_secctl_ppc_vmstate, IoTKitSecCtlPPC), 763 VMSTATE_STRUCT_ARRAY(ahbexp, IoTKitSecCtl, IOTS_NUM_AHB_EXP_PPC, 1, 764 iotkit_secctl_ppc_vmstate, IoTKitSecCtlPPC), 765 VMSTATE_END_OF_LIST() 766 }, 767 .subsections = (const VMStateDescription*[]) { 768 &iotkit_secctl_mpcintstatus_vmstate, 769 &iotkit_secctl_msc_vmstate, 770 NULL 771 }, 772 }; 773 774 static void iotkit_secctl_class_init(ObjectClass *klass, void *data) 775 { 776 DeviceClass *dc = DEVICE_CLASS(klass); 777 778 dc->vmsd = &iotkit_secctl_vmstate; 779 dc->reset = iotkit_secctl_reset; 780 } 781 782 static const TypeInfo iotkit_secctl_info = { 783 .name = TYPE_IOTKIT_SECCTL, 784 .parent = TYPE_SYS_BUS_DEVICE, 785 .instance_size = sizeof(IoTKitSecCtl), 786 .instance_init = iotkit_secctl_init, 787 .class_init = iotkit_secctl_class_init, 788 }; 789 790 static void iotkit_secctl_register_types(void) 791 { 792 type_register_static(&iotkit_secctl_info); 793 } 794 795 type_init(iotkit_secctl_register_types); 796