Anup Patel | e8f7934 | 2022-02-04 23:16:54 +0530 | [diff] [blame] | 1 | /* |
| 2 | * RISC-V APLIC (Advanced Platform Level Interrupt Controller) |
| 3 | * |
| 4 | * Copyright (c) 2021 Western Digital Corporation or its affiliates. |
| 5 | * |
| 6 | * This program is free software; you can redistribute it and/or modify it |
| 7 | * under the terms and conditions of the GNU General Public License, |
| 8 | * version 2 or later, as published by the Free Software Foundation. |
| 9 | * |
| 10 | * This program is distributed in the hope it will be useful, but WITHOUT |
| 11 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| 12 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for |
| 13 | * more details. |
| 14 | * |
| 15 | * You should have received a copy of the GNU General Public License along with |
| 16 | * this program. If not, see <http://www.gnu.org/licenses/>. |
| 17 | */ |
| 18 | |
| 19 | #include "qemu/osdep.h" |
| 20 | #include "qapi/error.h" |
| 21 | #include "qemu/log.h" |
| 22 | #include "qemu/module.h" |
| 23 | #include "qemu/error-report.h" |
| 24 | #include "qemu/bswap.h" |
| 25 | #include "exec/address-spaces.h" |
| 26 | #include "hw/sysbus.h" |
| 27 | #include "hw/pci/msi.h" |
| 28 | #include "hw/boards.h" |
| 29 | #include "hw/qdev-properties.h" |
| 30 | #include "hw/intc/riscv_aplic.h" |
| 31 | #include "hw/irq.h" |
| 32 | #include "target/riscv/cpu.h" |
| 33 | #include "sysemu/sysemu.h" |
| 34 | #include "migration/vmstate.h" |
| 35 | |
| 36 | #define APLIC_MAX_IDC (1UL << 14) |
| 37 | #define APLIC_MAX_SOURCE 1024 |
| 38 | #define APLIC_MIN_IPRIO_BITS 1 |
| 39 | #define APLIC_MAX_IPRIO_BITS 8 |
| 40 | #define APLIC_MAX_CHILDREN 1024 |
| 41 | |
| 42 | #define APLIC_DOMAINCFG 0x0000 |
| 43 | #define APLIC_DOMAINCFG_RDONLY 0x80000000 |
| 44 | #define APLIC_DOMAINCFG_IE (1 << 8) |
| 45 | #define APLIC_DOMAINCFG_DM (1 << 2) |
| 46 | #define APLIC_DOMAINCFG_BE (1 << 0) |
| 47 | |
| 48 | #define APLIC_SOURCECFG_BASE 0x0004 |
| 49 | #define APLIC_SOURCECFG_D (1 << 10) |
| 50 | #define APLIC_SOURCECFG_CHILDIDX_MASK 0x000003ff |
| 51 | #define APLIC_SOURCECFG_SM_MASK 0x00000007 |
| 52 | #define APLIC_SOURCECFG_SM_INACTIVE 0x0 |
| 53 | #define APLIC_SOURCECFG_SM_DETACH 0x1 |
| 54 | #define APLIC_SOURCECFG_SM_EDGE_RISE 0x4 |
| 55 | #define APLIC_SOURCECFG_SM_EDGE_FALL 0x5 |
| 56 | #define APLIC_SOURCECFG_SM_LEVEL_HIGH 0x6 |
| 57 | #define APLIC_SOURCECFG_SM_LEVEL_LOW 0x7 |
| 58 | |
| 59 | #define APLIC_MMSICFGADDR 0x1bc0 |
| 60 | #define APLIC_MMSICFGADDRH 0x1bc4 |
| 61 | #define APLIC_SMSICFGADDR 0x1bc8 |
| 62 | #define APLIC_SMSICFGADDRH 0x1bcc |
| 63 | |
| 64 | #define APLIC_xMSICFGADDRH_L (1UL << 31) |
| 65 | #define APLIC_xMSICFGADDRH_HHXS_MASK 0x1f |
| 66 | #define APLIC_xMSICFGADDRH_HHXS_SHIFT 24 |
| 67 | #define APLIC_xMSICFGADDRH_LHXS_MASK 0x7 |
| 68 | #define APLIC_xMSICFGADDRH_LHXS_SHIFT 20 |
| 69 | #define APLIC_xMSICFGADDRH_HHXW_MASK 0x7 |
| 70 | #define APLIC_xMSICFGADDRH_HHXW_SHIFT 16 |
| 71 | #define APLIC_xMSICFGADDRH_LHXW_MASK 0xf |
| 72 | #define APLIC_xMSICFGADDRH_LHXW_SHIFT 12 |
| 73 | #define APLIC_xMSICFGADDRH_BAPPN_MASK 0xfff |
| 74 | |
| 75 | #define APLIC_xMSICFGADDR_PPN_SHIFT 12 |
| 76 | |
| 77 | #define APLIC_xMSICFGADDR_PPN_HART(__lhxs) \ |
| 78 | ((1UL << (__lhxs)) - 1) |
| 79 | |
| 80 | #define APLIC_xMSICFGADDR_PPN_LHX_MASK(__lhxw) \ |
| 81 | ((1UL << (__lhxw)) - 1) |
| 82 | #define APLIC_xMSICFGADDR_PPN_LHX_SHIFT(__lhxs) \ |
| 83 | ((__lhxs)) |
| 84 | #define APLIC_xMSICFGADDR_PPN_LHX(__lhxw, __lhxs) \ |
| 85 | (APLIC_xMSICFGADDR_PPN_LHX_MASK(__lhxw) << \ |
| 86 | APLIC_xMSICFGADDR_PPN_LHX_SHIFT(__lhxs)) |
| 87 | |
| 88 | #define APLIC_xMSICFGADDR_PPN_HHX_MASK(__hhxw) \ |
| 89 | ((1UL << (__hhxw)) - 1) |
| 90 | #define APLIC_xMSICFGADDR_PPN_HHX_SHIFT(__hhxs) \ |
| 91 | ((__hhxs) + APLIC_xMSICFGADDR_PPN_SHIFT) |
| 92 | #define APLIC_xMSICFGADDR_PPN_HHX(__hhxw, __hhxs) \ |
| 93 | (APLIC_xMSICFGADDR_PPN_HHX_MASK(__hhxw) << \ |
| 94 | APLIC_xMSICFGADDR_PPN_HHX_SHIFT(__hhxs)) |
| 95 | |
| 96 | #define APLIC_xMSICFGADDRH_VALID_MASK \ |
| 97 | (APLIC_xMSICFGADDRH_L | \ |
| 98 | (APLIC_xMSICFGADDRH_HHXS_MASK << APLIC_xMSICFGADDRH_HHXS_SHIFT) | \ |
| 99 | (APLIC_xMSICFGADDRH_LHXS_MASK << APLIC_xMSICFGADDRH_LHXS_SHIFT) | \ |
| 100 | (APLIC_xMSICFGADDRH_HHXW_MASK << APLIC_xMSICFGADDRH_HHXW_SHIFT) | \ |
| 101 | (APLIC_xMSICFGADDRH_LHXW_MASK << APLIC_xMSICFGADDRH_LHXW_SHIFT) | \ |
| 102 | APLIC_xMSICFGADDRH_BAPPN_MASK) |
| 103 | |
| 104 | #define APLIC_SETIP_BASE 0x1c00 |
| 105 | #define APLIC_SETIPNUM 0x1cdc |
| 106 | |
| 107 | #define APLIC_CLRIP_BASE 0x1d00 |
| 108 | #define APLIC_CLRIPNUM 0x1ddc |
| 109 | |
| 110 | #define APLIC_SETIE_BASE 0x1e00 |
| 111 | #define APLIC_SETIENUM 0x1edc |
| 112 | |
| 113 | #define APLIC_CLRIE_BASE 0x1f00 |
| 114 | #define APLIC_CLRIENUM 0x1fdc |
| 115 | |
| 116 | #define APLIC_SETIPNUM_LE 0x2000 |
| 117 | #define APLIC_SETIPNUM_BE 0x2004 |
| 118 | |
| 119 | #define APLIC_ISTATE_PENDING (1U << 0) |
| 120 | #define APLIC_ISTATE_ENABLED (1U << 1) |
| 121 | #define APLIC_ISTATE_ENPEND (APLIC_ISTATE_ENABLED | \ |
| 122 | APLIC_ISTATE_PENDING) |
| 123 | #define APLIC_ISTATE_INPUT (1U << 8) |
| 124 | |
| 125 | #define APLIC_GENMSI 0x3000 |
| 126 | |
| 127 | #define APLIC_TARGET_BASE 0x3004 |
| 128 | #define APLIC_TARGET_HART_IDX_SHIFT 18 |
| 129 | #define APLIC_TARGET_HART_IDX_MASK 0x3fff |
| 130 | #define APLIC_TARGET_GUEST_IDX_SHIFT 12 |
| 131 | #define APLIC_TARGET_GUEST_IDX_MASK 0x3f |
| 132 | #define APLIC_TARGET_IPRIO_MASK 0xff |
| 133 | #define APLIC_TARGET_EIID_MASK 0x7ff |
| 134 | |
| 135 | #define APLIC_IDC_BASE 0x4000 |
| 136 | #define APLIC_IDC_SIZE 32 |
| 137 | |
| 138 | #define APLIC_IDC_IDELIVERY 0x00 |
| 139 | |
| 140 | #define APLIC_IDC_IFORCE 0x04 |
| 141 | |
| 142 | #define APLIC_IDC_ITHRESHOLD 0x08 |
| 143 | |
| 144 | #define APLIC_IDC_TOPI 0x18 |
| 145 | #define APLIC_IDC_TOPI_ID_SHIFT 16 |
| 146 | #define APLIC_IDC_TOPI_ID_MASK 0x3ff |
| 147 | #define APLIC_IDC_TOPI_PRIO_MASK 0xff |
| 148 | |
| 149 | #define APLIC_IDC_CLAIMI 0x1c |
| 150 | |
| 151 | static uint32_t riscv_aplic_read_input_word(RISCVAPLICState *aplic, |
| 152 | uint32_t word) |
| 153 | { |
| 154 | uint32_t i, irq, ret = 0; |
| 155 | |
| 156 | for (i = 0; i < 32; i++) { |
| 157 | irq = word * 32 + i; |
| 158 | if (!irq || aplic->num_irqs <= irq) { |
| 159 | continue; |
| 160 | } |
| 161 | |
| 162 | ret |= ((aplic->state[irq] & APLIC_ISTATE_INPUT) ? 1 : 0) << i; |
| 163 | } |
| 164 | |
| 165 | return ret; |
| 166 | } |
| 167 | |
| 168 | static uint32_t riscv_aplic_read_pending_word(RISCVAPLICState *aplic, |
| 169 | uint32_t word) |
| 170 | { |
| 171 | uint32_t i, irq, ret = 0; |
| 172 | |
| 173 | for (i = 0; i < 32; i++) { |
| 174 | irq = word * 32 + i; |
| 175 | if (!irq || aplic->num_irqs <= irq) { |
| 176 | continue; |
| 177 | } |
| 178 | |
| 179 | ret |= ((aplic->state[irq] & APLIC_ISTATE_PENDING) ? 1 : 0) << i; |
| 180 | } |
| 181 | |
| 182 | return ret; |
| 183 | } |
| 184 | |
| 185 | static void riscv_aplic_set_pending_raw(RISCVAPLICState *aplic, |
| 186 | uint32_t irq, bool pending) |
| 187 | { |
| 188 | if (pending) { |
| 189 | aplic->state[irq] |= APLIC_ISTATE_PENDING; |
| 190 | } else { |
| 191 | aplic->state[irq] &= ~APLIC_ISTATE_PENDING; |
| 192 | } |
| 193 | } |
| 194 | |
| 195 | static void riscv_aplic_set_pending(RISCVAPLICState *aplic, |
| 196 | uint32_t irq, bool pending) |
| 197 | { |
| 198 | uint32_t sourcecfg, sm; |
| 199 | |
| 200 | if ((irq <= 0) || (aplic->num_irqs <= irq)) { |
| 201 | return; |
| 202 | } |
| 203 | |
| 204 | sourcecfg = aplic->sourcecfg[irq]; |
| 205 | if (sourcecfg & APLIC_SOURCECFG_D) { |
| 206 | return; |
| 207 | } |
| 208 | |
| 209 | sm = sourcecfg & APLIC_SOURCECFG_SM_MASK; |
| 210 | if ((sm == APLIC_SOURCECFG_SM_INACTIVE) || |
| 211 | ((!aplic->msimode || (aplic->msimode && !pending)) && |
| 212 | ((sm == APLIC_SOURCECFG_SM_LEVEL_HIGH) || |
| 213 | (sm == APLIC_SOURCECFG_SM_LEVEL_LOW)))) { |
| 214 | return; |
| 215 | } |
| 216 | |
| 217 | riscv_aplic_set_pending_raw(aplic, irq, pending); |
| 218 | } |
| 219 | |
| 220 | static void riscv_aplic_set_pending_word(RISCVAPLICState *aplic, |
| 221 | uint32_t word, uint32_t value, |
| 222 | bool pending) |
| 223 | { |
| 224 | uint32_t i, irq; |
| 225 | |
| 226 | for (i = 0; i < 32; i++) { |
| 227 | irq = word * 32 + i; |
| 228 | if (!irq || aplic->num_irqs <= irq) { |
| 229 | continue; |
| 230 | } |
| 231 | |
| 232 | if (value & (1U << i)) { |
| 233 | riscv_aplic_set_pending(aplic, irq, pending); |
| 234 | } |
| 235 | } |
| 236 | } |
| 237 | |
| 238 | static uint32_t riscv_aplic_read_enabled_word(RISCVAPLICState *aplic, |
| 239 | int word) |
| 240 | { |
| 241 | uint32_t i, irq, ret = 0; |
| 242 | |
| 243 | for (i = 0; i < 32; i++) { |
| 244 | irq = word * 32 + i; |
| 245 | if (!irq || aplic->num_irqs <= irq) { |
| 246 | continue; |
| 247 | } |
| 248 | |
| 249 | ret |= ((aplic->state[irq] & APLIC_ISTATE_ENABLED) ? 1 : 0) << i; |
| 250 | } |
| 251 | |
| 252 | return ret; |
| 253 | } |
| 254 | |
| 255 | static void riscv_aplic_set_enabled_raw(RISCVAPLICState *aplic, |
| 256 | uint32_t irq, bool enabled) |
| 257 | { |
| 258 | if (enabled) { |
| 259 | aplic->state[irq] |= APLIC_ISTATE_ENABLED; |
| 260 | } else { |
| 261 | aplic->state[irq] &= ~APLIC_ISTATE_ENABLED; |
| 262 | } |
| 263 | } |
| 264 | |
| 265 | static void riscv_aplic_set_enabled(RISCVAPLICState *aplic, |
| 266 | uint32_t irq, bool enabled) |
| 267 | { |
| 268 | uint32_t sourcecfg, sm; |
| 269 | |
| 270 | if ((irq <= 0) || (aplic->num_irqs <= irq)) { |
| 271 | return; |
| 272 | } |
| 273 | |
| 274 | sourcecfg = aplic->sourcecfg[irq]; |
| 275 | if (sourcecfg & APLIC_SOURCECFG_D) { |
| 276 | return; |
| 277 | } |
| 278 | |
| 279 | sm = sourcecfg & APLIC_SOURCECFG_SM_MASK; |
| 280 | if (sm == APLIC_SOURCECFG_SM_INACTIVE) { |
| 281 | return; |
| 282 | } |
| 283 | |
| 284 | riscv_aplic_set_enabled_raw(aplic, irq, enabled); |
| 285 | } |
| 286 | |
| 287 | static void riscv_aplic_set_enabled_word(RISCVAPLICState *aplic, |
| 288 | uint32_t word, uint32_t value, |
| 289 | bool enabled) |
| 290 | { |
| 291 | uint32_t i, irq; |
| 292 | |
| 293 | for (i = 0; i < 32; i++) { |
| 294 | irq = word * 32 + i; |
| 295 | if (!irq || aplic->num_irqs <= irq) { |
| 296 | continue; |
| 297 | } |
| 298 | |
| 299 | if (value & (1U << i)) { |
| 300 | riscv_aplic_set_enabled(aplic, irq, enabled); |
| 301 | } |
| 302 | } |
| 303 | } |
| 304 | |
| 305 | static void riscv_aplic_msi_send(RISCVAPLICState *aplic, |
| 306 | uint32_t hart_idx, uint32_t guest_idx, |
| 307 | uint32_t eiid) |
| 308 | { |
| 309 | uint64_t addr; |
| 310 | MemTxResult result; |
| 311 | RISCVAPLICState *aplic_m; |
| 312 | uint32_t lhxs, lhxw, hhxs, hhxw, group_idx, msicfgaddr, msicfgaddrH; |
| 313 | |
| 314 | aplic_m = aplic; |
| 315 | while (aplic_m && !aplic_m->mmode) { |
| 316 | aplic_m = aplic_m->parent; |
| 317 | } |
| 318 | if (!aplic_m) { |
| 319 | qemu_log_mask(LOG_GUEST_ERROR, "%s: m-level APLIC not found\n", |
| 320 | __func__); |
| 321 | return; |
| 322 | } |
| 323 | |
| 324 | if (aplic->mmode) { |
| 325 | msicfgaddr = aplic_m->mmsicfgaddr; |
| 326 | msicfgaddrH = aplic_m->mmsicfgaddrH; |
| 327 | } else { |
| 328 | msicfgaddr = aplic_m->smsicfgaddr; |
| 329 | msicfgaddrH = aplic_m->smsicfgaddrH; |
| 330 | } |
| 331 | |
| 332 | lhxs = (msicfgaddrH >> APLIC_xMSICFGADDRH_LHXS_SHIFT) & |
| 333 | APLIC_xMSICFGADDRH_LHXS_MASK; |
| 334 | lhxw = (msicfgaddrH >> APLIC_xMSICFGADDRH_LHXW_SHIFT) & |
| 335 | APLIC_xMSICFGADDRH_LHXW_MASK; |
| 336 | hhxs = (msicfgaddrH >> APLIC_xMSICFGADDRH_HHXS_SHIFT) & |
| 337 | APLIC_xMSICFGADDRH_HHXS_MASK; |
| 338 | hhxw = (msicfgaddrH >> APLIC_xMSICFGADDRH_HHXW_SHIFT) & |
| 339 | APLIC_xMSICFGADDRH_HHXW_MASK; |
| 340 | |
| 341 | group_idx = hart_idx >> lhxw; |
| 342 | hart_idx &= APLIC_xMSICFGADDR_PPN_LHX_MASK(lhxw); |
| 343 | |
| 344 | addr = msicfgaddr; |
| 345 | addr |= ((uint64_t)(msicfgaddrH & APLIC_xMSICFGADDRH_BAPPN_MASK)) << 32; |
| 346 | addr |= ((uint64_t)(group_idx & APLIC_xMSICFGADDR_PPN_HHX_MASK(hhxw))) << |
| 347 | APLIC_xMSICFGADDR_PPN_HHX_SHIFT(hhxs); |
| 348 | addr |= ((uint64_t)(hart_idx & APLIC_xMSICFGADDR_PPN_LHX_MASK(lhxw))) << |
| 349 | APLIC_xMSICFGADDR_PPN_LHX_SHIFT(lhxs); |
| 350 | addr |= (uint64_t)(guest_idx & APLIC_xMSICFGADDR_PPN_HART(lhxs)); |
| 351 | addr <<= APLIC_xMSICFGADDR_PPN_SHIFT; |
| 352 | |
| 353 | address_space_stl_le(&address_space_memory, addr, |
| 354 | eiid, MEMTXATTRS_UNSPECIFIED, &result); |
| 355 | if (result != MEMTX_OK) { |
| 356 | qemu_log_mask(LOG_GUEST_ERROR, "%s: MSI write failed for " |
| 357 | "hart_index=%d guest_index=%d eiid=%d\n", |
| 358 | __func__, hart_idx, guest_idx, eiid); |
| 359 | } |
| 360 | } |
| 361 | |
| 362 | static void riscv_aplic_msi_irq_update(RISCVAPLICState *aplic, uint32_t irq) |
| 363 | { |
| 364 | uint32_t hart_idx, guest_idx, eiid; |
| 365 | |
| 366 | if (!aplic->msimode || (aplic->num_irqs <= irq) || |
| 367 | !(aplic->domaincfg & APLIC_DOMAINCFG_IE)) { |
| 368 | return; |
| 369 | } |
| 370 | |
| 371 | if ((aplic->state[irq] & APLIC_ISTATE_ENPEND) != APLIC_ISTATE_ENPEND) { |
| 372 | return; |
| 373 | } |
| 374 | |
| 375 | riscv_aplic_set_pending_raw(aplic, irq, false); |
| 376 | |
| 377 | hart_idx = aplic->target[irq] >> APLIC_TARGET_HART_IDX_SHIFT; |
| 378 | hart_idx &= APLIC_TARGET_HART_IDX_MASK; |
| 379 | if (aplic->mmode) { |
| 380 | /* M-level APLIC ignores guest_index */ |
| 381 | guest_idx = 0; |
| 382 | } else { |
| 383 | guest_idx = aplic->target[irq] >> APLIC_TARGET_GUEST_IDX_SHIFT; |
| 384 | guest_idx &= APLIC_TARGET_GUEST_IDX_MASK; |
| 385 | } |
| 386 | eiid = aplic->target[irq] & APLIC_TARGET_EIID_MASK; |
| 387 | riscv_aplic_msi_send(aplic, hart_idx, guest_idx, eiid); |
| 388 | } |
| 389 | |
| 390 | static uint32_t riscv_aplic_idc_topi(RISCVAPLICState *aplic, uint32_t idc) |
| 391 | { |
| 392 | uint32_t best_irq, best_iprio; |
| 393 | uint32_t irq, iprio, ihartidx, ithres; |
| 394 | |
| 395 | if (aplic->num_harts <= idc) { |
| 396 | return 0; |
| 397 | } |
| 398 | |
| 399 | ithres = aplic->ithreshold[idc]; |
| 400 | best_irq = best_iprio = UINT32_MAX; |
| 401 | for (irq = 1; irq < aplic->num_irqs; irq++) { |
| 402 | if ((aplic->state[irq] & APLIC_ISTATE_ENPEND) != |
| 403 | APLIC_ISTATE_ENPEND) { |
| 404 | continue; |
| 405 | } |
| 406 | |
| 407 | ihartidx = aplic->target[irq] >> APLIC_TARGET_HART_IDX_SHIFT; |
| 408 | ihartidx &= APLIC_TARGET_HART_IDX_MASK; |
| 409 | if (ihartidx != idc) { |
| 410 | continue; |
| 411 | } |
| 412 | |
| 413 | iprio = aplic->target[irq] & aplic->iprio_mask; |
| 414 | if (ithres && iprio >= ithres) { |
| 415 | continue; |
| 416 | } |
| 417 | |
| 418 | if (iprio < best_iprio) { |
| 419 | best_irq = irq; |
| 420 | best_iprio = iprio; |
| 421 | } |
| 422 | } |
| 423 | |
| 424 | if (best_irq < aplic->num_irqs && best_iprio <= aplic->iprio_mask) { |
| 425 | return (best_irq << APLIC_IDC_TOPI_ID_SHIFT) | best_iprio; |
| 426 | } |
| 427 | |
| 428 | return 0; |
| 429 | } |
| 430 | |
| 431 | static void riscv_aplic_idc_update(RISCVAPLICState *aplic, uint32_t idc) |
| 432 | { |
| 433 | uint32_t topi; |
| 434 | |
| 435 | if (aplic->msimode || aplic->num_harts <= idc) { |
| 436 | return; |
| 437 | } |
| 438 | |
| 439 | topi = riscv_aplic_idc_topi(aplic, idc); |
| 440 | if ((aplic->domaincfg & APLIC_DOMAINCFG_IE) && |
| 441 | aplic->idelivery[idc] && |
| 442 | (aplic->iforce[idc] || topi)) { |
| 443 | qemu_irq_raise(aplic->external_irqs[idc]); |
| 444 | } else { |
| 445 | qemu_irq_lower(aplic->external_irqs[idc]); |
| 446 | } |
| 447 | } |
| 448 | |
| 449 | static uint32_t riscv_aplic_idc_claimi(RISCVAPLICState *aplic, uint32_t idc) |
| 450 | { |
| 451 | uint32_t irq, state, sm, topi = riscv_aplic_idc_topi(aplic, idc); |
| 452 | |
| 453 | if (!topi) { |
| 454 | aplic->iforce[idc] = 0; |
| 455 | return 0; |
| 456 | } |
| 457 | |
| 458 | irq = (topi >> APLIC_IDC_TOPI_ID_SHIFT) & APLIC_IDC_TOPI_ID_MASK; |
| 459 | sm = aplic->sourcecfg[irq] & APLIC_SOURCECFG_SM_MASK; |
| 460 | state = aplic->state[irq]; |
| 461 | riscv_aplic_set_pending_raw(aplic, irq, false); |
| 462 | if ((sm == APLIC_SOURCECFG_SM_LEVEL_HIGH) && |
| 463 | (state & APLIC_ISTATE_INPUT)) { |
| 464 | riscv_aplic_set_pending_raw(aplic, irq, true); |
| 465 | } else if ((sm == APLIC_SOURCECFG_SM_LEVEL_LOW) && |
| 466 | !(state & APLIC_ISTATE_INPUT)) { |
| 467 | riscv_aplic_set_pending_raw(aplic, irq, true); |
| 468 | } |
| 469 | riscv_aplic_idc_update(aplic, idc); |
| 470 | |
| 471 | return topi; |
| 472 | } |
| 473 | |
| 474 | static void riscv_aplic_request(void *opaque, int irq, int level) |
| 475 | { |
| 476 | bool update = false; |
| 477 | RISCVAPLICState *aplic = opaque; |
| 478 | uint32_t sourcecfg, childidx, state, idc; |
| 479 | |
| 480 | assert((0 < irq) && (irq < aplic->num_irqs)); |
| 481 | |
| 482 | sourcecfg = aplic->sourcecfg[irq]; |
| 483 | if (sourcecfg & APLIC_SOURCECFG_D) { |
| 484 | childidx = sourcecfg & APLIC_SOURCECFG_CHILDIDX_MASK; |
| 485 | if (childidx < aplic->num_children) { |
| 486 | riscv_aplic_request(aplic->children[childidx], irq, level); |
| 487 | } |
| 488 | return; |
| 489 | } |
| 490 | |
| 491 | state = aplic->state[irq]; |
| 492 | switch (sourcecfg & APLIC_SOURCECFG_SM_MASK) { |
| 493 | case APLIC_SOURCECFG_SM_EDGE_RISE: |
| 494 | if ((level > 0) && !(state & APLIC_ISTATE_INPUT) && |
| 495 | !(state & APLIC_ISTATE_PENDING)) { |
| 496 | riscv_aplic_set_pending_raw(aplic, irq, true); |
| 497 | update = true; |
| 498 | } |
| 499 | break; |
| 500 | case APLIC_SOURCECFG_SM_EDGE_FALL: |
| 501 | if ((level <= 0) && (state & APLIC_ISTATE_INPUT) && |
| 502 | !(state & APLIC_ISTATE_PENDING)) { |
| 503 | riscv_aplic_set_pending_raw(aplic, irq, true); |
| 504 | update = true; |
| 505 | } |
| 506 | break; |
| 507 | case APLIC_SOURCECFG_SM_LEVEL_HIGH: |
| 508 | if ((level > 0) && !(state & APLIC_ISTATE_PENDING)) { |
| 509 | riscv_aplic_set_pending_raw(aplic, irq, true); |
| 510 | update = true; |
| 511 | } |
| 512 | break; |
| 513 | case APLIC_SOURCECFG_SM_LEVEL_LOW: |
| 514 | if ((level <= 0) && !(state & APLIC_ISTATE_PENDING)) { |
| 515 | riscv_aplic_set_pending_raw(aplic, irq, true); |
| 516 | update = true; |
| 517 | } |
| 518 | break; |
| 519 | default: |
| 520 | break; |
| 521 | } |
| 522 | |
| 523 | if (level <= 0) { |
| 524 | aplic->state[irq] &= ~APLIC_ISTATE_INPUT; |
| 525 | } else { |
| 526 | aplic->state[irq] |= APLIC_ISTATE_INPUT; |
| 527 | } |
| 528 | |
| 529 | if (update) { |
| 530 | if (aplic->msimode) { |
| 531 | riscv_aplic_msi_irq_update(aplic, irq); |
| 532 | } else { |
| 533 | idc = aplic->target[irq] >> APLIC_TARGET_HART_IDX_SHIFT; |
| 534 | idc &= APLIC_TARGET_HART_IDX_MASK; |
| 535 | riscv_aplic_idc_update(aplic, idc); |
| 536 | } |
| 537 | } |
| 538 | } |
| 539 | |
| 540 | static uint64_t riscv_aplic_read(void *opaque, hwaddr addr, unsigned size) |
| 541 | { |
| 542 | uint32_t irq, word, idc; |
| 543 | RISCVAPLICState *aplic = opaque; |
| 544 | |
| 545 | /* Reads must be 4 byte words */ |
| 546 | if ((addr & 0x3) != 0) { |
| 547 | goto err; |
| 548 | } |
| 549 | |
| 550 | if (addr == APLIC_DOMAINCFG) { |
| 551 | return APLIC_DOMAINCFG_RDONLY | aplic->domaincfg | |
| 552 | (aplic->msimode ? APLIC_DOMAINCFG_DM : 0); |
| 553 | } else if ((APLIC_SOURCECFG_BASE <= addr) && |
| 554 | (addr < (APLIC_SOURCECFG_BASE + (aplic->num_irqs - 1) * 4))) { |
| 555 | irq = ((addr - APLIC_SOURCECFG_BASE) >> 2) + 1; |
| 556 | return aplic->sourcecfg[irq]; |
| 557 | } else if (aplic->mmode && aplic->msimode && |
| 558 | (addr == APLIC_MMSICFGADDR)) { |
| 559 | return aplic->mmsicfgaddr; |
| 560 | } else if (aplic->mmode && aplic->msimode && |
| 561 | (addr == APLIC_MMSICFGADDRH)) { |
| 562 | return aplic->mmsicfgaddrH; |
| 563 | } else if (aplic->mmode && aplic->msimode && |
| 564 | (addr == APLIC_SMSICFGADDR)) { |
| 565 | /* |
| 566 | * Registers SMSICFGADDR and SMSICFGADDRH are implemented only if: |
| 567 | * (a) the interrupt domain is at machine level |
| 568 | * (b) the domain's harts implement supervisor mode |
| 569 | * (c) the domain has one or more child supervisor-level domains |
| 570 | * that support MSI delivery mode (domaincfg.DM is not read- |
| 571 | * only zero in at least one of the supervisor-level child |
| 572 | * domains). |
| 573 | */ |
| 574 | return (aplic->num_children) ? aplic->smsicfgaddr : 0; |
| 575 | } else if (aplic->mmode && aplic->msimode && |
| 576 | (addr == APLIC_SMSICFGADDRH)) { |
| 577 | return (aplic->num_children) ? aplic->smsicfgaddrH : 0; |
| 578 | } else if ((APLIC_SETIP_BASE <= addr) && |
| 579 | (addr < (APLIC_SETIP_BASE + aplic->bitfield_words * 4))) { |
| 580 | word = (addr - APLIC_SETIP_BASE) >> 2; |
| 581 | return riscv_aplic_read_pending_word(aplic, word); |
| 582 | } else if (addr == APLIC_SETIPNUM) { |
| 583 | return 0; |
| 584 | } else if ((APLIC_CLRIP_BASE <= addr) && |
| 585 | (addr < (APLIC_CLRIP_BASE + aplic->bitfield_words * 4))) { |
| 586 | word = (addr - APLIC_CLRIP_BASE) >> 2; |
| 587 | return riscv_aplic_read_input_word(aplic, word); |
| 588 | } else if (addr == APLIC_CLRIPNUM) { |
| 589 | return 0; |
| 590 | } else if ((APLIC_SETIE_BASE <= addr) && |
| 591 | (addr < (APLIC_SETIE_BASE + aplic->bitfield_words * 4))) { |
| 592 | word = (addr - APLIC_SETIE_BASE) >> 2; |
| 593 | return riscv_aplic_read_enabled_word(aplic, word); |
| 594 | } else if (addr == APLIC_SETIENUM) { |
| 595 | return 0; |
| 596 | } else if ((APLIC_CLRIE_BASE <= addr) && |
| 597 | (addr < (APLIC_CLRIE_BASE + aplic->bitfield_words * 4))) { |
| 598 | return 0; |
| 599 | } else if (addr == APLIC_CLRIENUM) { |
| 600 | return 0; |
| 601 | } else if (addr == APLIC_SETIPNUM_LE) { |
| 602 | return 0; |
| 603 | } else if (addr == APLIC_SETIPNUM_BE) { |
| 604 | return 0; |
| 605 | } else if (addr == APLIC_GENMSI) { |
| 606 | return (aplic->msimode) ? aplic->genmsi : 0; |
| 607 | } else if ((APLIC_TARGET_BASE <= addr) && |
| 608 | (addr < (APLIC_TARGET_BASE + (aplic->num_irqs - 1) * 4))) { |
| 609 | irq = ((addr - APLIC_TARGET_BASE) >> 2) + 1; |
| 610 | return aplic->target[irq]; |
| 611 | } else if (!aplic->msimode && (APLIC_IDC_BASE <= addr) && |
| 612 | (addr < (APLIC_IDC_BASE + aplic->num_harts * APLIC_IDC_SIZE))) { |
| 613 | idc = (addr - APLIC_IDC_BASE) / APLIC_IDC_SIZE; |
| 614 | switch (addr - (APLIC_IDC_BASE + idc * APLIC_IDC_SIZE)) { |
| 615 | case APLIC_IDC_IDELIVERY: |
| 616 | return aplic->idelivery[idc]; |
| 617 | case APLIC_IDC_IFORCE: |
| 618 | return aplic->iforce[idc]; |
| 619 | case APLIC_IDC_ITHRESHOLD: |
| 620 | return aplic->ithreshold[idc]; |
| 621 | case APLIC_IDC_TOPI: |
| 622 | return riscv_aplic_idc_topi(aplic, idc); |
| 623 | case APLIC_IDC_CLAIMI: |
| 624 | return riscv_aplic_idc_claimi(aplic, idc); |
| 625 | default: |
| 626 | goto err; |
| 627 | }; |
| 628 | } |
| 629 | |
| 630 | err: |
| 631 | qemu_log_mask(LOG_GUEST_ERROR, |
| 632 | "%s: Invalid register read 0x%" HWADDR_PRIx "\n", |
| 633 | __func__, addr); |
| 634 | return 0; |
| 635 | } |
| 636 | |
| 637 | static void riscv_aplic_write(void *opaque, hwaddr addr, uint64_t value, |
| 638 | unsigned size) |
| 639 | { |
| 640 | RISCVAPLICState *aplic = opaque; |
| 641 | uint32_t irq, word, idc = UINT32_MAX; |
| 642 | |
| 643 | /* Writes must be 4 byte words */ |
| 644 | if ((addr & 0x3) != 0) { |
| 645 | goto err; |
| 646 | } |
| 647 | |
| 648 | if (addr == APLIC_DOMAINCFG) { |
| 649 | /* Only IE bit writeable at the moment */ |
| 650 | value &= APLIC_DOMAINCFG_IE; |
| 651 | aplic->domaincfg = value; |
| 652 | } else if ((APLIC_SOURCECFG_BASE <= addr) && |
| 653 | (addr < (APLIC_SOURCECFG_BASE + (aplic->num_irqs - 1) * 4))) { |
| 654 | irq = ((addr - APLIC_SOURCECFG_BASE) >> 2) + 1; |
| 655 | if (!aplic->num_children && (value & APLIC_SOURCECFG_D)) { |
| 656 | value = 0; |
| 657 | } |
| 658 | if (value & APLIC_SOURCECFG_D) { |
| 659 | value &= (APLIC_SOURCECFG_D | APLIC_SOURCECFG_CHILDIDX_MASK); |
| 660 | } else { |
| 661 | value &= (APLIC_SOURCECFG_D | APLIC_SOURCECFG_SM_MASK); |
| 662 | } |
| 663 | aplic->sourcecfg[irq] = value; |
| 664 | if ((aplic->sourcecfg[irq] & APLIC_SOURCECFG_D) || |
| 665 | (aplic->sourcecfg[irq] == 0)) { |
| 666 | riscv_aplic_set_pending_raw(aplic, irq, false); |
| 667 | riscv_aplic_set_enabled_raw(aplic, irq, false); |
| 668 | } |
| 669 | } else if (aplic->mmode && aplic->msimode && |
| 670 | (addr == APLIC_MMSICFGADDR)) { |
| 671 | if (!(aplic->mmsicfgaddrH & APLIC_xMSICFGADDRH_L)) { |
| 672 | aplic->mmsicfgaddr = value; |
| 673 | } |
| 674 | } else if (aplic->mmode && aplic->msimode && |
| 675 | (addr == APLIC_MMSICFGADDRH)) { |
| 676 | if (!(aplic->mmsicfgaddrH & APLIC_xMSICFGADDRH_L)) { |
| 677 | aplic->mmsicfgaddrH = value & APLIC_xMSICFGADDRH_VALID_MASK; |
| 678 | } |
| 679 | } else if (aplic->mmode && aplic->msimode && |
| 680 | (addr == APLIC_SMSICFGADDR)) { |
| 681 | /* |
| 682 | * Registers SMSICFGADDR and SMSICFGADDRH are implemented only if: |
| 683 | * (a) the interrupt domain is at machine level |
| 684 | * (b) the domain's harts implement supervisor mode |
| 685 | * (c) the domain has one or more child supervisor-level domains |
| 686 | * that support MSI delivery mode (domaincfg.DM is not read- |
| 687 | * only zero in at least one of the supervisor-level child |
| 688 | * domains). |
| 689 | */ |
| 690 | if (aplic->num_children && |
| 691 | !(aplic->smsicfgaddrH & APLIC_xMSICFGADDRH_L)) { |
| 692 | aplic->smsicfgaddr = value; |
| 693 | } |
| 694 | } else if (aplic->mmode && aplic->msimode && |
| 695 | (addr == APLIC_SMSICFGADDRH)) { |
| 696 | if (aplic->num_children && |
| 697 | !(aplic->smsicfgaddrH & APLIC_xMSICFGADDRH_L)) { |
| 698 | aplic->smsicfgaddrH = value & APLIC_xMSICFGADDRH_VALID_MASK; |
| 699 | } |
| 700 | } else if ((APLIC_SETIP_BASE <= addr) && |
| 701 | (addr < (APLIC_SETIP_BASE + aplic->bitfield_words * 4))) { |
| 702 | word = (addr - APLIC_SETIP_BASE) >> 2; |
| 703 | riscv_aplic_set_pending_word(aplic, word, value, true); |
| 704 | } else if (addr == APLIC_SETIPNUM) { |
| 705 | riscv_aplic_set_pending(aplic, value, true); |
| 706 | } else if ((APLIC_CLRIP_BASE <= addr) && |
| 707 | (addr < (APLIC_CLRIP_BASE + aplic->bitfield_words * 4))) { |
| 708 | word = (addr - APLIC_CLRIP_BASE) >> 2; |
| 709 | riscv_aplic_set_pending_word(aplic, word, value, false); |
| 710 | } else if (addr == APLIC_CLRIPNUM) { |
| 711 | riscv_aplic_set_pending(aplic, value, false); |
| 712 | } else if ((APLIC_SETIE_BASE <= addr) && |
| 713 | (addr < (APLIC_SETIE_BASE + aplic->bitfield_words * 4))) { |
| 714 | word = (addr - APLIC_SETIE_BASE) >> 2; |
| 715 | riscv_aplic_set_enabled_word(aplic, word, value, true); |
| 716 | } else if (addr == APLIC_SETIENUM) { |
| 717 | riscv_aplic_set_enabled(aplic, value, true); |
| 718 | } else if ((APLIC_CLRIE_BASE <= addr) && |
| 719 | (addr < (APLIC_CLRIE_BASE + aplic->bitfield_words * 4))) { |
| 720 | word = (addr - APLIC_CLRIE_BASE) >> 2; |
| 721 | riscv_aplic_set_enabled_word(aplic, word, value, false); |
| 722 | } else if (addr == APLIC_CLRIENUM) { |
| 723 | riscv_aplic_set_enabled(aplic, value, false); |
| 724 | } else if (addr == APLIC_SETIPNUM_LE) { |
| 725 | riscv_aplic_set_pending(aplic, value, true); |
| 726 | } else if (addr == APLIC_SETIPNUM_BE) { |
| 727 | riscv_aplic_set_pending(aplic, bswap32(value), true); |
| 728 | } else if (addr == APLIC_GENMSI) { |
| 729 | if (aplic->msimode) { |
| 730 | aplic->genmsi = value & ~(APLIC_TARGET_GUEST_IDX_MASK << |
| 731 | APLIC_TARGET_GUEST_IDX_SHIFT); |
| 732 | riscv_aplic_msi_send(aplic, |
| 733 | value >> APLIC_TARGET_HART_IDX_SHIFT, |
| 734 | 0, |
| 735 | value & APLIC_TARGET_EIID_MASK); |
| 736 | } |
| 737 | } else if ((APLIC_TARGET_BASE <= addr) && |
| 738 | (addr < (APLIC_TARGET_BASE + (aplic->num_irqs - 1) * 4))) { |
| 739 | irq = ((addr - APLIC_TARGET_BASE) >> 2) + 1; |
| 740 | if (aplic->msimode) { |
| 741 | aplic->target[irq] = value; |
| 742 | } else { |
| 743 | aplic->target[irq] = (value & ~APLIC_TARGET_IPRIO_MASK) | |
| 744 | ((value & aplic->iprio_mask) ? |
| 745 | (value & aplic->iprio_mask) : 1); |
| 746 | } |
| 747 | } else if (!aplic->msimode && (APLIC_IDC_BASE <= addr) && |
| 748 | (addr < (APLIC_IDC_BASE + aplic->num_harts * APLIC_IDC_SIZE))) { |
| 749 | idc = (addr - APLIC_IDC_BASE) / APLIC_IDC_SIZE; |
| 750 | switch (addr - (APLIC_IDC_BASE + idc * APLIC_IDC_SIZE)) { |
| 751 | case APLIC_IDC_IDELIVERY: |
| 752 | aplic->idelivery[idc] = value & 0x1; |
| 753 | break; |
| 754 | case APLIC_IDC_IFORCE: |
| 755 | aplic->iforce[idc] = value & 0x1; |
| 756 | break; |
| 757 | case APLIC_IDC_ITHRESHOLD: |
| 758 | aplic->ithreshold[idc] = value & aplic->iprio_mask; |
| 759 | break; |
| 760 | default: |
| 761 | goto err; |
| 762 | }; |
| 763 | } else { |
| 764 | goto err; |
| 765 | } |
| 766 | |
| 767 | if (aplic->msimode) { |
| 768 | for (irq = 1; irq < aplic->num_irqs; irq++) { |
| 769 | riscv_aplic_msi_irq_update(aplic, irq); |
| 770 | } |
| 771 | } else { |
| 772 | if (idc == UINT32_MAX) { |
| 773 | for (idc = 0; idc < aplic->num_harts; idc++) { |
| 774 | riscv_aplic_idc_update(aplic, idc); |
| 775 | } |
| 776 | } else { |
| 777 | riscv_aplic_idc_update(aplic, idc); |
| 778 | } |
| 779 | } |
| 780 | |
| 781 | return; |
| 782 | |
| 783 | err: |
| 784 | qemu_log_mask(LOG_GUEST_ERROR, |
| 785 | "%s: Invalid register write 0x%" HWADDR_PRIx "\n", |
| 786 | __func__, addr); |
| 787 | } |
| 788 | |
| 789 | static const MemoryRegionOps riscv_aplic_ops = { |
| 790 | .read = riscv_aplic_read, |
| 791 | .write = riscv_aplic_write, |
| 792 | .endianness = DEVICE_LITTLE_ENDIAN, |
| 793 | .valid = { |
| 794 | .min_access_size = 4, |
| 795 | .max_access_size = 4 |
| 796 | } |
| 797 | }; |
| 798 | |
| 799 | static void riscv_aplic_realize(DeviceState *dev, Error **errp) |
| 800 | { |
| 801 | uint32_t i; |
| 802 | RISCVAPLICState *aplic = RISCV_APLIC(dev); |
| 803 | |
| 804 | aplic->bitfield_words = (aplic->num_irqs + 31) >> 5; |
| 805 | aplic->sourcecfg = g_new0(uint32_t, aplic->num_irqs); |
| 806 | aplic->state = g_new(uint32_t, aplic->num_irqs); |
| 807 | aplic->target = g_new0(uint32_t, aplic->num_irqs); |
| 808 | if (!aplic->msimode) { |
| 809 | for (i = 0; i < aplic->num_irqs; i++) { |
| 810 | aplic->target[i] = 1; |
| 811 | } |
| 812 | } |
| 813 | aplic->idelivery = g_new0(uint32_t, aplic->num_harts); |
| 814 | aplic->iforce = g_new0(uint32_t, aplic->num_harts); |
| 815 | aplic->ithreshold = g_new0(uint32_t, aplic->num_harts); |
| 816 | |
| 817 | memory_region_init_io(&aplic->mmio, OBJECT(dev), &riscv_aplic_ops, aplic, |
| 818 | TYPE_RISCV_APLIC, aplic->aperture_size); |
| 819 | sysbus_init_mmio(SYS_BUS_DEVICE(dev), &aplic->mmio); |
| 820 | |
| 821 | /* |
| 822 | * Only root APLICs have hardware IRQ lines. All non-root APLICs |
| 823 | * have IRQ lines delegated by their parent APLIC. |
| 824 | */ |
| 825 | if (!aplic->parent) { |
| 826 | qdev_init_gpio_in(dev, riscv_aplic_request, aplic->num_irqs); |
| 827 | } |
| 828 | |
| 829 | /* Create output IRQ lines for non-MSI mode */ |
| 830 | if (!aplic->msimode) { |
| 831 | aplic->external_irqs = g_malloc(sizeof(qemu_irq) * aplic->num_harts); |
| 832 | qdev_init_gpio_out(dev, aplic->external_irqs, aplic->num_harts); |
| 833 | |
| 834 | /* Claim the CPU interrupt to be triggered by this APLIC */ |
| 835 | for (i = 0; i < aplic->num_harts; i++) { |
| 836 | RISCVCPU *cpu = RISCV_CPU(qemu_get_cpu(aplic->hartid_base + i)); |
| 837 | if (riscv_cpu_claim_interrupts(cpu, |
| 838 | (aplic->mmode) ? MIP_MEIP : MIP_SEIP) < 0) { |
| 839 | error_report("%s already claimed", |
| 840 | (aplic->mmode) ? "MEIP" : "SEIP"); |
| 841 | exit(1); |
| 842 | } |
| 843 | } |
| 844 | } |
| 845 | |
| 846 | msi_nonbroken = true; |
| 847 | } |
| 848 | |
| 849 | static Property riscv_aplic_properties[] = { |
| 850 | DEFINE_PROP_UINT32("aperture-size", RISCVAPLICState, aperture_size, 0), |
| 851 | DEFINE_PROP_UINT32("hartid-base", RISCVAPLICState, hartid_base, 0), |
| 852 | DEFINE_PROP_UINT32("num-harts", RISCVAPLICState, num_harts, 0), |
| 853 | DEFINE_PROP_UINT32("iprio-mask", RISCVAPLICState, iprio_mask, 0), |
| 854 | DEFINE_PROP_UINT32("num-irqs", RISCVAPLICState, num_irqs, 0), |
| 855 | DEFINE_PROP_BOOL("msimode", RISCVAPLICState, msimode, 0), |
| 856 | DEFINE_PROP_BOOL("mmode", RISCVAPLICState, mmode, 0), |
| 857 | DEFINE_PROP_END_OF_LIST(), |
| 858 | }; |
| 859 | |
| 860 | static const VMStateDescription vmstate_riscv_aplic = { |
| 861 | .name = "riscv_aplic", |
| 862 | .version_id = 1, |
| 863 | .minimum_version_id = 1, |
| 864 | .fields = (VMStateField[]) { |
| 865 | VMSTATE_UINT32(domaincfg, RISCVAPLICState), |
| 866 | VMSTATE_UINT32(mmsicfgaddr, RISCVAPLICState), |
| 867 | VMSTATE_UINT32(mmsicfgaddrH, RISCVAPLICState), |
| 868 | VMSTATE_UINT32(smsicfgaddr, RISCVAPLICState), |
| 869 | VMSTATE_UINT32(smsicfgaddrH, RISCVAPLICState), |
| 870 | VMSTATE_UINT32(genmsi, RISCVAPLICState), |
| 871 | VMSTATE_VARRAY_UINT32(sourcecfg, RISCVAPLICState, |
| 872 | num_irqs, 0, |
| 873 | vmstate_info_uint32, uint32_t), |
| 874 | VMSTATE_VARRAY_UINT32(state, RISCVAPLICState, |
| 875 | num_irqs, 0, |
| 876 | vmstate_info_uint32, uint32_t), |
| 877 | VMSTATE_VARRAY_UINT32(target, RISCVAPLICState, |
| 878 | num_irqs, 0, |
| 879 | vmstate_info_uint32, uint32_t), |
| 880 | VMSTATE_VARRAY_UINT32(idelivery, RISCVAPLICState, |
| 881 | num_harts, 0, |
| 882 | vmstate_info_uint32, uint32_t), |
| 883 | VMSTATE_VARRAY_UINT32(iforce, RISCVAPLICState, |
| 884 | num_harts, 0, |
| 885 | vmstate_info_uint32, uint32_t), |
| 886 | VMSTATE_VARRAY_UINT32(ithreshold, RISCVAPLICState, |
| 887 | num_harts, 0, |
| 888 | vmstate_info_uint32, uint32_t), |
| 889 | VMSTATE_END_OF_LIST() |
| 890 | } |
| 891 | }; |
| 892 | |
| 893 | static void riscv_aplic_class_init(ObjectClass *klass, void *data) |
| 894 | { |
| 895 | DeviceClass *dc = DEVICE_CLASS(klass); |
| 896 | |
| 897 | device_class_set_props(dc, riscv_aplic_properties); |
| 898 | dc->realize = riscv_aplic_realize; |
| 899 | dc->vmsd = &vmstate_riscv_aplic; |
| 900 | } |
| 901 | |
| 902 | static const TypeInfo riscv_aplic_info = { |
| 903 | .name = TYPE_RISCV_APLIC, |
| 904 | .parent = TYPE_SYS_BUS_DEVICE, |
| 905 | .instance_size = sizeof(RISCVAPLICState), |
| 906 | .class_init = riscv_aplic_class_init, |
| 907 | }; |
| 908 | |
| 909 | static void riscv_aplic_register_types(void) |
| 910 | { |
| 911 | type_register_static(&riscv_aplic_info); |
| 912 | } |
| 913 | |
| 914 | type_init(riscv_aplic_register_types) |
| 915 | |
| 916 | /* |
| 917 | * Add a APLIC device to another APLIC device as child for |
| 918 | * interrupt delegation. |
| 919 | */ |
| 920 | void riscv_aplic_add_child(DeviceState *parent, DeviceState *child) |
| 921 | { |
| 922 | RISCVAPLICState *caplic, *paplic; |
| 923 | |
| 924 | assert(parent && child); |
| 925 | caplic = RISCV_APLIC(child); |
| 926 | paplic = RISCV_APLIC(parent); |
| 927 | |
| 928 | assert(paplic->num_irqs == caplic->num_irqs); |
| 929 | assert(paplic->num_children <= QEMU_APLIC_MAX_CHILDREN); |
| 930 | |
| 931 | caplic->parent = paplic; |
| 932 | paplic->children[paplic->num_children] = caplic; |
| 933 | paplic->num_children++; |
| 934 | } |
| 935 | |
| 936 | /* |
| 937 | * Create APLIC device. |
| 938 | */ |
| 939 | DeviceState *riscv_aplic_create(hwaddr addr, hwaddr size, |
| 940 | uint32_t hartid_base, uint32_t num_harts, uint32_t num_sources, |
| 941 | uint32_t iprio_bits, bool msimode, bool mmode, DeviceState *parent) |
| 942 | { |
| 943 | DeviceState *dev = qdev_new(TYPE_RISCV_APLIC); |
| 944 | uint32_t i; |
| 945 | |
| 946 | assert(num_harts < APLIC_MAX_IDC); |
| 947 | assert((APLIC_IDC_BASE + (num_harts * APLIC_IDC_SIZE)) <= size); |
| 948 | assert(num_sources < APLIC_MAX_SOURCE); |
| 949 | assert(APLIC_MIN_IPRIO_BITS <= iprio_bits); |
| 950 | assert(iprio_bits <= APLIC_MAX_IPRIO_BITS); |
| 951 | |
| 952 | qdev_prop_set_uint32(dev, "aperture-size", size); |
| 953 | qdev_prop_set_uint32(dev, "hartid-base", hartid_base); |
| 954 | qdev_prop_set_uint32(dev, "num-harts", num_harts); |
| 955 | qdev_prop_set_uint32(dev, "iprio-mask", ((1U << iprio_bits) - 1)); |
| 956 | qdev_prop_set_uint32(dev, "num-irqs", num_sources + 1); |
| 957 | qdev_prop_set_bit(dev, "msimode", msimode); |
| 958 | qdev_prop_set_bit(dev, "mmode", mmode); |
| 959 | |
| 960 | sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal); |
| 961 | sysbus_mmio_map(SYS_BUS_DEVICE(dev), 0, addr); |
| 962 | |
| 963 | if (parent) { |
| 964 | riscv_aplic_add_child(parent, dev); |
| 965 | } |
| 966 | |
| 967 | if (!msimode) { |
| 968 | for (i = 0; i < num_harts; i++) { |
| 969 | CPUState *cpu = qemu_get_cpu(hartid_base + i); |
| 970 | |
| 971 | qdev_connect_gpio_out_named(dev, NULL, i, |
| 972 | qdev_get_gpio_in(DEVICE(cpu), |
| 973 | (mmode) ? IRQ_M_EXT : IRQ_S_EXT)); |
| 974 | } |
| 975 | } |
| 976 | |
| 977 | return dev; |
| 978 | } |