xref: /openbmc/qemu/hw/intc/riscv_aplic.c (revision dd84028f)
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 writable 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_new0(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(cpu_by_arch_id(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 = cpu_by_arch_id(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 }
979