David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 1 | /* |
| 2 | * QEMU PowerPC pSeries Logical Partition (aka sPAPR) hardware System Emulator |
| 3 | * |
| 4 | * PAPR Virtualized Interrupt System, aka ICS/ICP aka xics |
| 5 | * |
| 6 | * Copyright (c) 2010,2011 David Gibson, IBM Corporation. |
| 7 | * |
| 8 | * Permission is hereby granted, free of charge, to any person obtaining a copy |
| 9 | * of this software and associated documentation files (the "Software"), to deal |
| 10 | * in the Software without restriction, including without limitation the rights |
| 11 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
| 12 | * copies of the Software, and to permit persons to whom the Software is |
| 13 | * furnished to do so, subject to the following conditions: |
| 14 | * |
| 15 | * The above copyright notice and this permission notice shall be included in |
| 16 | * all copies or substantial portions of the Software. |
| 17 | * |
| 18 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| 19 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| 20 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| 21 | * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| 22 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
| 23 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
| 24 | * THE SOFTWARE. |
| 25 | * |
| 26 | */ |
| 27 | |
Peter Maydell | 0d75590 | 2016-01-26 18:16:58 +0000 | [diff] [blame] | 28 | #include "qemu/osdep.h" |
Markus Armbruster | da34e65 | 2016-03-14 09:01:28 +0100 | [diff] [blame] | 29 | #include "qapi/error.h" |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 30 | #include "trace.h" |
Benjamin Herrenschmidt | 5d87e4b | 2013-09-26 16:18:46 +1000 | [diff] [blame] | 31 | #include "qemu/timer.h" |
Paolo Bonzini | 0d09e41 | 2013-02-05 17:06:20 +0100 | [diff] [blame] | 32 | #include "hw/ppc/xics.h" |
Markus Armbruster | a27bd6c | 2019-08-12 07:23:51 +0200 | [diff] [blame] | 33 | #include "hw/qdev-properties.h" |
Alexey Kardashevskiy | 9ccff2a | 2013-09-26 16:18:38 +1000 | [diff] [blame] | 34 | #include "qemu/error-report.h" |
Markus Armbruster | 0b8fa32 | 2019-05-23 16:35:07 +0200 | [diff] [blame] | 35 | #include "qemu/module.h" |
Alexey Kardashevskiy | 5a3d7b2 | 2013-09-26 16:18:42 +1000 | [diff] [blame] | 36 | #include "qapi/visitor.h" |
Markus Armbruster | d645427 | 2019-08-12 07:23:45 +0200 | [diff] [blame] | 37 | #include "migration/vmstate.h" |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 38 | #include "hw/intc/intc.h" |
Markus Armbruster | 64552b6 | 2019-08-12 07:23:42 +0200 | [diff] [blame] | 39 | #include "hw/irq.h" |
Greg Kurz | 0e5c7fa | 2019-02-15 12:39:48 +0100 | [diff] [blame] | 40 | #include "sysemu/kvm.h" |
Markus Armbruster | 71e8a91 | 2019-08-12 07:23:38 +0200 | [diff] [blame] | 41 | #include "sysemu/reset.h" |
Philippe Mathieu-Daudé | 0a77a76 | 2024-01-29 18:05:29 +0100 | [diff] [blame] | 42 | #include "target/ppc/cpu.h" |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 43 | |
Philippe Mathieu-Daudé | 5242494 | 2024-06-07 13:14:20 +0200 | [diff] [blame] | 44 | void icp_pic_print_info(ICPState *icp, GString *buf) |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 45 | { |
Greg Kurz | 0a83b47 | 2019-10-24 16:27:33 +0200 | [diff] [blame] | 46 | int cpu_index; |
| 47 | |
| 48 | /* Skip partially initialized vCPUs. This can happen on sPAPR when vCPUs |
| 49 | * are hot plugged or unplugged. |
| 50 | */ |
| 51 | if (!icp) { |
| 52 | return; |
| 53 | } |
| 54 | |
| 55 | cpu_index = icp->cs ? icp->cs->cpu_index : -1; |
Cédric Le Goater | b9038e7 | 2017-02-27 15:29:13 +0100 | [diff] [blame] | 56 | |
| 57 | if (!icp->output) { |
| 58 | return; |
| 59 | } |
Greg Kurz | dcb556f | 2017-11-13 20:42:39 +0100 | [diff] [blame] | 60 | |
Greg Kurz | 0e5c7fa | 2019-02-15 12:39:48 +0100 | [diff] [blame] | 61 | if (kvm_irqchip_in_kernel()) { |
| 62 | icp_synchronize_state(icp); |
Greg Kurz | dcb556f | 2017-11-13 20:42:39 +0100 | [diff] [blame] | 63 | } |
| 64 | |
Philippe Mathieu-Daudé | 5242494 | 2024-06-07 13:14:20 +0200 | [diff] [blame] | 65 | g_string_append_printf(buf, "CPU %d XIRR=%08x (%p) PP=%02x MFRR=%02x\n", |
| 66 | cpu_index, icp->xirr, icp->xirr_owner, |
| 67 | icp->pending_priority, icp->mfrr); |
Cédric Le Goater | b9038e7 | 2017-02-27 15:29:13 +0100 | [diff] [blame] | 68 | } |
| 69 | |
Philippe Mathieu-Daudé | dd77c49 | 2024-06-07 13:23:47 +0200 | [diff] [blame] | 70 | void ics_pic_print_info(ICSState *ics, GString *buf) |
Cédric Le Goater | b9038e7 | 2017-02-27 15:29:13 +0100 | [diff] [blame] | 71 | { |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 72 | uint32_t i; |
| 73 | |
Philippe Mathieu-Daudé | dd77c49 | 2024-06-07 13:23:47 +0200 | [diff] [blame] | 74 | g_string_append_printf(buf, "ICS %4x..%4x %p\n", |
| 75 | ics->offset, ics->offset + ics->nr_irqs - 1, ics); |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 76 | |
Cédric Le Goater | b9038e7 | 2017-02-27 15:29:13 +0100 | [diff] [blame] | 77 | if (!ics->irqs) { |
| 78 | return; |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 79 | } |
| 80 | |
Greg Kurz | d80b2cc | 2019-02-15 12:40:18 +0100 | [diff] [blame] | 81 | if (kvm_irqchip_in_kernel()) { |
| 82 | ics_synchronize_state(ics); |
Greg Kurz | dcb556f | 2017-11-13 20:42:39 +0100 | [diff] [blame] | 83 | } |
| 84 | |
Cédric Le Goater | b9038e7 | 2017-02-27 15:29:13 +0100 | [diff] [blame] | 85 | for (i = 0; i < ics->nr_irqs; i++) { |
| 86 | ICSIRQState *irq = ics->irqs + i; |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 87 | |
Cédric Le Goater | b9038e7 | 2017-02-27 15:29:13 +0100 | [diff] [blame] | 88 | if (!(irq->flags & XICS_FLAGS_IRQ_MASK)) { |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 89 | continue; |
| 90 | } |
Philippe Mathieu-Daudé | dd77c49 | 2024-06-07 13:23:47 +0200 | [diff] [blame] | 91 | g_string_append_printf(buf, " %4x %s %02x %02x\n", |
| 92 | ics->offset + i, |
| 93 | (irq->flags & XICS_FLAGS_IRQ_LSI) ? |
| 94 | "LSI" : "MSI", |
| 95 | irq->priority, irq->status); |
Benjamin Herrenschmidt | b1fc72f | 2016-10-17 22:33:14 +0200 | [diff] [blame] | 96 | } |
| 97 | } |
| 98 | |
Alexey Kardashevskiy | 5a3d7b2 | 2013-09-26 16:18:42 +1000 | [diff] [blame] | 99 | /* |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 100 | * ICP: Presentation layer |
| 101 | */ |
| 102 | |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 103 | #define XISR_MASK 0x00ffffff |
| 104 | #define CPPR_MASK 0xff000000 |
| 105 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 106 | #define XISR(icp) (((icp)->xirr) & XISR_MASK) |
| 107 | #define CPPR(icp) (((icp)->xirr) >> 24) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 108 | |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 109 | static void ics_reject(ICSState *ics, uint32_t nr); |
| 110 | static void ics_eoi(ICSState *ics, uint32_t nr); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 111 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 112 | static void icp_check_ipi(ICPState *icp) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 113 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 114 | if (XISR(icp) && (icp->pending_priority <= icp->mfrr)) { |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 115 | return; |
| 116 | } |
| 117 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 118 | trace_xics_icp_check_ipi(icp->cs->cpu_index, icp->mfrr); |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 119 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 120 | if (XISR(icp) && icp->xirr_owner) { |
| 121 | ics_reject(icp->xirr_owner, XISR(icp)); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 122 | } |
| 123 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 124 | icp->xirr = (icp->xirr & ~XISR_MASK) | XICS_IPI; |
| 125 | icp->pending_priority = icp->mfrr; |
| 126 | icp->xirr_owner = NULL; |
| 127 | qemu_irq_raise(icp->output); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 128 | } |
| 129 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 130 | void icp_resend(ICPState *icp) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 131 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 132 | XICSFabric *xi = icp->xics; |
Cédric Le Goater | 2cd908d | 2017-02-27 15:29:17 +0100 | [diff] [blame] | 133 | XICSFabricClass *xic = XICS_FABRIC_GET_CLASS(xi); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 134 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 135 | if (icp->mfrr < CPPR(icp)) { |
| 136 | icp_check_ipi(icp); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 137 | } |
Cédric Le Goater | 2cd908d | 2017-02-27 15:29:17 +0100 | [diff] [blame] | 138 | |
| 139 | xic->ics_resend(xi); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 140 | } |
| 141 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 142 | void icp_set_cppr(ICPState *icp, uint8_t cppr) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 143 | { |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 144 | uint8_t old_cppr; |
| 145 | uint32_t old_xisr; |
| 146 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 147 | old_cppr = CPPR(icp); |
| 148 | icp->xirr = (icp->xirr & ~CPPR_MASK) | (cppr << 24); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 149 | |
| 150 | if (cppr < old_cppr) { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 151 | if (XISR(icp) && (cppr <= icp->pending_priority)) { |
| 152 | old_xisr = XISR(icp); |
| 153 | icp->xirr &= ~XISR_MASK; /* Clear XISR */ |
| 154 | icp->pending_priority = 0xff; |
| 155 | qemu_irq_lower(icp->output); |
| 156 | if (icp->xirr_owner) { |
| 157 | ics_reject(icp->xirr_owner, old_xisr); |
| 158 | icp->xirr_owner = NULL; |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 159 | } |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 160 | } |
| 161 | } else { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 162 | if (!XISR(icp)) { |
| 163 | icp_resend(icp); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 164 | } |
| 165 | } |
| 166 | } |
| 167 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 168 | void icp_set_mfrr(ICPState *icp, uint8_t mfrr) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 169 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 170 | icp->mfrr = mfrr; |
| 171 | if (mfrr < CPPR(icp)) { |
| 172 | icp_check_ipi(icp); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 173 | } |
| 174 | } |
| 175 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 176 | uint32_t icp_accept(ICPState *icp) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 177 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 178 | uint32_t xirr = icp->xirr; |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 179 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 180 | qemu_irq_lower(icp->output); |
| 181 | icp->xirr = icp->pending_priority << 24; |
| 182 | icp->pending_priority = 0xff; |
| 183 | icp->xirr_owner = NULL; |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 184 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 185 | trace_xics_icp_accept(xirr, icp->xirr); |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 186 | |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 187 | return xirr; |
| 188 | } |
| 189 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 190 | uint32_t icp_ipoll(ICPState *icp, uint32_t *mfrr) |
Benjamin Herrenschmidt | 1cbd222 | 2016-06-29 00:35:14 +0530 | [diff] [blame] | 191 | { |
| 192 | if (mfrr) { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 193 | *mfrr = icp->mfrr; |
Benjamin Herrenschmidt | 1cbd222 | 2016-06-29 00:35:14 +0530 | [diff] [blame] | 194 | } |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 195 | return icp->xirr; |
Benjamin Herrenschmidt | 1cbd222 | 2016-06-29 00:35:14 +0530 | [diff] [blame] | 196 | } |
| 197 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 198 | void icp_eoi(ICPState *icp, uint32_t xirr) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 199 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 200 | XICSFabric *xi = icp->xics; |
Cédric Le Goater | 2cd908d | 2017-02-27 15:29:17 +0100 | [diff] [blame] | 201 | XICSFabricClass *xic = XICS_FABRIC_GET_CLASS(xi); |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 202 | ICSState *ics; |
| 203 | uint32_t irq; |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 204 | |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 205 | /* Send EOI -> ICS */ |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 206 | icp->xirr = (icp->xirr & ~CPPR_MASK) | (xirr & CPPR_MASK); |
| 207 | trace_xics_icp_eoi(icp->cs->cpu_index, xirr, icp->xirr); |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 208 | irq = xirr & XISR_MASK; |
Cédric Le Goater | 2cd908d | 2017-02-27 15:29:17 +0100 | [diff] [blame] | 209 | |
| 210 | ics = xic->ics_get(xi, irq); |
| 211 | if (ics) { |
| 212 | ics_eoi(ics, irq); |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 213 | } |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 214 | if (!XISR(icp)) { |
| 215 | icp_resend(icp); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 216 | } |
| 217 | } |
| 218 | |
Cédric Le Goater | 9ae1329 | 2020-01-27 15:45:06 +0100 | [diff] [blame] | 219 | void icp_irq(ICSState *ics, int server, int nr, uint8_t priority) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 220 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 221 | ICPState *icp = xics_icp_get(ics->xics, server); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 222 | |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 223 | trace_xics_icp_irq(server, nr, priority); |
| 224 | |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 225 | if ((priority >= CPPR(icp)) |
| 226 | || (XISR(icp) && (icp->pending_priority <= priority))) { |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 227 | ics_reject(ics, nr); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 228 | } else { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 229 | if (XISR(icp) && icp->xirr_owner) { |
| 230 | ics_reject(icp->xirr_owner, XISR(icp)); |
| 231 | icp->xirr_owner = NULL; |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 232 | } |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 233 | icp->xirr = (icp->xirr & ~XISR_MASK) | (nr & XISR_MASK); |
| 234 | icp->xirr_owner = ics; |
| 235 | icp->pending_priority = priority; |
| 236 | trace_xics_icp_raise(icp->xirr, icp->pending_priority); |
| 237 | qemu_irq_raise(icp->output); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 238 | } |
| 239 | } |
| 240 | |
Greg Kurz | 0e5c7fa | 2019-02-15 12:39:48 +0100 | [diff] [blame] | 241 | static int icp_pre_save(void *opaque) |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 242 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 243 | ICPState *icp = opaque; |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 244 | |
Greg Kurz | 0e5c7fa | 2019-02-15 12:39:48 +0100 | [diff] [blame] | 245 | if (kvm_irqchip_in_kernel()) { |
| 246 | icp_get_kvm_state(icp); |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 247 | } |
Dr. David Alan Gilbert | 44b1ff3 | 2017-09-25 12:29:12 +0100 | [diff] [blame] | 248 | |
| 249 | return 0; |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 250 | } |
| 251 | |
Greg Kurz | 0e5c7fa | 2019-02-15 12:39:48 +0100 | [diff] [blame] | 252 | static int icp_post_load(void *opaque, int version_id) |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 253 | { |
Cédric Le Goater | 8e4fba2 | 2017-02-27 15:29:33 +0100 | [diff] [blame] | 254 | ICPState *icp = opaque; |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 255 | |
Greg Kurz | 0e5c7fa | 2019-02-15 12:39:48 +0100 | [diff] [blame] | 256 | if (kvm_irqchip_in_kernel()) { |
Greg Kurz | 330a21e | 2019-06-17 15:46:57 +0200 | [diff] [blame] | 257 | Error *local_err = NULL; |
| 258 | int ret; |
| 259 | |
| 260 | ret = icp_set_kvm_state(icp, &local_err); |
| 261 | if (ret < 0) { |
| 262 | error_report_err(local_err); |
| 263 | return ret; |
| 264 | } |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 265 | } |
| 266 | |
| 267 | return 0; |
| 268 | } |
| 269 | |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 270 | static const VMStateDescription vmstate_icp_server = { |
| 271 | .name = "icp/server", |
| 272 | .version_id = 1, |
| 273 | .minimum_version_id = 1, |
Greg Kurz | 0e5c7fa | 2019-02-15 12:39:48 +0100 | [diff] [blame] | 274 | .pre_save = icp_pre_save, |
| 275 | .post_load = icp_post_load, |
Richard Henderson | 45b1f81 | 2023-12-21 14:16:15 +1100 | [diff] [blame] | 276 | .fields = (const VMStateField[]) { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 277 | /* Sanity check */ |
| 278 | VMSTATE_UINT32(xirr, ICPState), |
| 279 | VMSTATE_UINT8(pending_priority, ICPState), |
| 280 | VMSTATE_UINT8(mfrr, ICPState), |
| 281 | VMSTATE_END_OF_LIST() |
| 282 | }, |
| 283 | }; |
| 284 | |
Cédric Le Goater | d49e8a9 | 2019-10-22 18:38:10 +0200 | [diff] [blame] | 285 | void icp_reset(ICPState *icp) |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 286 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 287 | icp->xirr = 0; |
| 288 | icp->pending_priority = 0xff; |
| 289 | icp->mfrr = 0xff; |
| 290 | |
Greg Kurz | d82f397 | 2019-02-15 12:39:54 +0100 | [diff] [blame] | 291 | if (kvm_irqchip_in_kernel()) { |
Greg Kurz | 330a21e | 2019-06-17 15:46:57 +0200 | [diff] [blame] | 292 | Error *local_err = NULL; |
| 293 | |
Cédric Le Goater | d49e8a9 | 2019-10-22 18:38:10 +0200 | [diff] [blame] | 294 | icp_set_kvm_state(icp, &local_err); |
Greg Kurz | 330a21e | 2019-06-17 15:46:57 +0200 | [diff] [blame] | 295 | if (local_err) { |
| 296 | error_report_err(local_err); |
| 297 | } |
Greg Kurz | d82f397 | 2019-02-15 12:39:54 +0100 | [diff] [blame] | 298 | } |
Greg Kurz | b585395 | 2018-07-12 12:01:49 +0200 | [diff] [blame] | 299 | } |
| 300 | |
Cédric Le Goater | 817bb6a | 2017-02-27 15:29:11 +0100 | [diff] [blame] | 301 | static void icp_realize(DeviceState *dev, Error **errp) |
| 302 | { |
| 303 | ICPState *icp = ICP(dev); |
Cédric Le Goater | 9fd0122 | 2022-07-05 16:58:10 +0200 | [diff] [blame] | 304 | PowerPCCPU *cpu; |
Greg Kurz | 9ed6566 | 2017-06-08 15:42:59 +0200 | [diff] [blame] | 305 | CPUPPCState *env; |
Cédric Le Goater | 817bb6a | 2017-02-27 15:29:11 +0100 | [diff] [blame] | 306 | Error *err = NULL; |
| 307 | |
Greg Kurz | b4a378a | 2019-11-18 00:20:41 +0100 | [diff] [blame] | 308 | assert(icp->xics); |
Greg Kurz | e388d66 | 2019-11-18 00:20:47 +0100 | [diff] [blame] | 309 | assert(icp->cs); |
Cédric Le Goater | 7ea6e06 | 2017-03-03 13:51:03 +0100 | [diff] [blame] | 310 | |
Cédric Le Goater | 9fd0122 | 2022-07-05 16:58:10 +0200 | [diff] [blame] | 311 | cpu = POWERPC_CPU(icp->cs); |
| 312 | env = &cpu->env; |
Greg Kurz | 9ed6566 | 2017-06-08 15:42:59 +0200 | [diff] [blame] | 313 | switch (PPC_INPUT(env)) { |
| 314 | case PPC_FLAGS_INPUT_POWER7: |
Cédric Le Goater | 9fd0122 | 2022-07-05 16:58:10 +0200 | [diff] [blame] | 315 | icp->output = qdev_get_gpio_in(DEVICE(cpu), POWER7_INPUT_INT); |
Greg Kurz | 9ed6566 | 2017-06-08 15:42:59 +0200 | [diff] [blame] | 316 | break; |
Benjamin Herrenschmidt | 67afe77 | 2019-02-15 17:16:47 +0100 | [diff] [blame] | 317 | case PPC_FLAGS_INPUT_POWER9: /* For SPAPR xics emulation */ |
Cédric Le Goater | 9fd0122 | 2022-07-05 16:58:10 +0200 | [diff] [blame] | 318 | icp->output = qdev_get_gpio_in(DEVICE(cpu), POWER9_INPUT_INT); |
Benjamin Herrenschmidt | 67afe77 | 2019-02-15 17:16:47 +0100 | [diff] [blame] | 319 | break; |
Greg Kurz | 9ed6566 | 2017-06-08 15:42:59 +0200 | [diff] [blame] | 320 | |
| 321 | case PPC_FLAGS_INPUT_970: |
Cédric Le Goater | 9fd0122 | 2022-07-05 16:58:10 +0200 | [diff] [blame] | 322 | icp->output = qdev_get_gpio_in(DEVICE(cpu), PPC970_INPUT_INT); |
Greg Kurz | 9ed6566 | 2017-06-08 15:42:59 +0200 | [diff] [blame] | 323 | break; |
| 324 | |
| 325 | default: |
| 326 | error_setg(errp, "XICS interrupt controller does not support this CPU bus model"); |
| 327 | return; |
| 328 | } |
| 329 | |
Greg Kurz | d9b9e6f | 2019-06-17 16:10:33 +0200 | [diff] [blame] | 330 | /* Connect the presenter to the VCPU (required for CPU hotplug) */ |
Greg Kurz | 8e6e6ef | 2019-02-15 12:40:00 +0100 | [diff] [blame] | 331 | if (kvm_irqchip_in_kernel()) { |
| 332 | icp_kvm_realize(dev, &err); |
| 333 | if (err) { |
| 334 | error_propagate(errp, err); |
| 335 | return; |
| 336 | } |
| 337 | } |
Juan Quintela | 485fb95 | 2023-10-20 11:07:25 +0200 | [diff] [blame] | 338 | /* |
| 339 | * The way that pre_2_10_icp is handling is really, really hacky. |
| 340 | * We used to have here this call: |
| 341 | * |
| 342 | * vmstate_register(NULL, icp->cs->cpu_index, &vmstate_icp_server, icp); |
| 343 | * |
| 344 | * But we were doing: |
| 345 | * pre_2_10_vmstate_register_dummy_icp() |
| 346 | * this vmstate_register() |
| 347 | * pre_2_10_vmstate_unregister_dummy_icp() |
| 348 | * |
| 349 | * So for a short amount of time we had to vmstate entries with |
| 350 | * the same name. This fixes it. |
| 351 | */ |
| 352 | vmstate_replace_hack_for_ppc(NULL, icp->cs->cpu_index, |
| 353 | &vmstate_icp_server, icp); |
Cédric Le Goater | 817bb6a | 2017-02-27 15:29:11 +0100 | [diff] [blame] | 354 | } |
| 355 | |
Markus Armbruster | b69c3c2 | 2020-05-05 17:29:24 +0200 | [diff] [blame] | 356 | static void icp_unrealize(DeviceState *dev) |
Greg Kurz | 62f94fc | 2017-05-24 19:40:43 +0200 | [diff] [blame] | 357 | { |
Greg Kurz | c95f616 | 2017-06-14 15:29:10 +0200 | [diff] [blame] | 358 | ICPState *icp = ICP(dev); |
| 359 | |
| 360 | vmstate_unregister(NULL, &vmstate_icp_server, icp); |
Greg Kurz | 62f94fc | 2017-05-24 19:40:43 +0200 | [diff] [blame] | 361 | } |
Cédric Le Goater | 817bb6a | 2017-02-27 15:29:11 +0100 | [diff] [blame] | 362 | |
Greg Kurz | b4a378a | 2019-11-18 00:20:41 +0100 | [diff] [blame] | 363 | static Property icp_properties[] = { |
| 364 | DEFINE_PROP_LINK(ICP_PROP_XICS, ICPState, xics, TYPE_XICS_FABRIC, |
| 365 | XICSFabric *), |
Greg Kurz | e388d66 | 2019-11-18 00:20:47 +0100 | [diff] [blame] | 366 | DEFINE_PROP_LINK(ICP_PROP_CPU, ICPState, cs, TYPE_CPU, CPUState *), |
Greg Kurz | b4a378a | 2019-11-18 00:20:41 +0100 | [diff] [blame] | 367 | DEFINE_PROP_END_OF_LIST(), |
| 368 | }; |
| 369 | |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 370 | static void icp_class_init(ObjectClass *klass, void *data) |
| 371 | { |
| 372 | DeviceClass *dc = DEVICE_CLASS(klass); |
| 373 | |
Cédric Le Goater | 817bb6a | 2017-02-27 15:29:11 +0100 | [diff] [blame] | 374 | dc->realize = icp_realize; |
Greg Kurz | 62f94fc | 2017-05-24 19:40:43 +0200 | [diff] [blame] | 375 | dc->unrealize = icp_unrealize; |
Marc-André Lureau | 4f67d30 | 2020-01-10 19:30:32 +0400 | [diff] [blame] | 376 | device_class_set_props(dc, icp_properties); |
Greg Kurz | e6144bf | 2019-10-04 10:37:47 +0200 | [diff] [blame] | 377 | /* |
| 378 | * Reason: part of XICS interrupt controller, needs to be wired up |
| 379 | * by icp_create(). |
| 380 | */ |
| 381 | dc->user_creatable = false; |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 382 | } |
| 383 | |
Alexey Kardashevskiy | 456df19 | 2013-09-26 16:18:41 +1000 | [diff] [blame] | 384 | static const TypeInfo icp_info = { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 385 | .name = TYPE_ICP, |
| 386 | .parent = TYPE_DEVICE, |
| 387 | .instance_size = sizeof(ICPState), |
| 388 | .class_init = icp_class_init, |
Alexey Kardashevskiy | d1b5682 | 2013-09-26 16:18:39 +1000 | [diff] [blame] | 389 | .class_size = sizeof(ICPStateClass), |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 390 | }; |
| 391 | |
Cédric Le Goater | 4f7a47b | 2017-12-01 17:06:00 +0100 | [diff] [blame] | 392 | Object *icp_create(Object *cpu, const char *type, XICSFabric *xi, Error **errp) |
| 393 | { |
Cédric Le Goater | 4f7a47b | 2017-12-01 17:06:00 +0100 | [diff] [blame] | 394 | Object *obj; |
| 395 | |
| 396 | obj = object_new(type); |
Markus Armbruster | d262312 | 2020-05-05 17:29:22 +0200 | [diff] [blame] | 397 | object_property_add_child(cpu, type, obj); |
Cédric Le Goater | 4f7a47b | 2017-12-01 17:06:00 +0100 | [diff] [blame] | 398 | object_unref(obj); |
Markus Armbruster | 5325cc3 | 2020-07-07 18:05:54 +0200 | [diff] [blame] | 399 | object_property_set_link(obj, ICP_PROP_XICS, OBJECT(xi), &error_abort); |
| 400 | object_property_set_link(obj, ICP_PROP_CPU, cpu, &error_abort); |
Markus Armbruster | af175e8 | 2020-07-07 18:06:03 +0200 | [diff] [blame] | 401 | if (!qdev_realize(DEVICE(obj), NULL, errp)) { |
Cédric Le Goater | 4f7a47b | 2017-12-01 17:06:00 +0100 | [diff] [blame] | 402 | object_unparent(obj); |
Cédric Le Goater | 4f7a47b | 2017-12-01 17:06:00 +0100 | [diff] [blame] | 403 | obj = NULL; |
| 404 | } |
| 405 | |
| 406 | return obj; |
| 407 | } |
| 408 | |
Greg Kurz | 0990ce6 | 2019-10-24 16:27:22 +0200 | [diff] [blame] | 409 | void icp_destroy(ICPState *icp) |
| 410 | { |
Greg Kurz | 35886de | 2019-10-24 16:27:27 +0200 | [diff] [blame] | 411 | Object *obj = OBJECT(icp); |
| 412 | |
Greg Kurz | 35886de | 2019-10-24 16:27:27 +0200 | [diff] [blame] | 413 | object_unparent(obj); |
Greg Kurz | 0990ce6 | 2019-10-24 16:27:22 +0200 | [diff] [blame] | 414 | } |
| 415 | |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 416 | /* |
| 417 | * ICS: Source layer |
| 418 | */ |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 419 | static void ics_resend_msi(ICSState *ics, int srcno) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 420 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 421 | ICSIRQState *irq = ics->irqs + srcno; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 422 | |
| 423 | /* FIXME: filter by server#? */ |
David Gibson | 98ca8c0 | 2012-09-12 16:57:17 +0000 | [diff] [blame] | 424 | if (irq->status & XICS_STATUS_REJECTED) { |
| 425 | irq->status &= ~XICS_STATUS_REJECTED; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 426 | if (irq->priority != 0xff) { |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 427 | icp_irq(ics, irq->server, srcno + ics->offset, irq->priority); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 428 | } |
| 429 | } |
| 430 | } |
| 431 | |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 432 | static void ics_resend_lsi(ICSState *ics, int srcno) |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 433 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 434 | ICSIRQState *irq = ics->irqs + srcno; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 435 | |
David Gibson | 98ca8c0 | 2012-09-12 16:57:17 +0000 | [diff] [blame] | 436 | if ((irq->priority != 0xff) |
| 437 | && (irq->status & XICS_STATUS_ASSERTED) |
| 438 | && !(irq->status & XICS_STATUS_SENT)) { |
| 439 | irq->status |= XICS_STATUS_SENT; |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 440 | icp_irq(ics, irq->server, srcno + ics->offset, irq->priority); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 441 | } |
| 442 | } |
| 443 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 444 | static void ics_set_irq_msi(ICSState *ics, int srcno, int val) |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 445 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 446 | ICSIRQState *irq = ics->irqs + srcno; |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 447 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 448 | trace_xics_ics_set_irq_msi(srcno, srcno + ics->offset); |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 449 | |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 450 | if (val) { |
| 451 | if (irq->priority == 0xff) { |
David Gibson | 98ca8c0 | 2012-09-12 16:57:17 +0000 | [diff] [blame] | 452 | irq->status |= XICS_STATUS_MASKED_PENDING; |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 453 | trace_xics_masked_pending(); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 454 | } else { |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 455 | icp_irq(ics, irq->server, srcno + ics->offset, irq->priority); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 456 | } |
| 457 | } |
| 458 | } |
| 459 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 460 | static void ics_set_irq_lsi(ICSState *ics, int srcno, int val) |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 461 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 462 | ICSIRQState *irq = ics->irqs + srcno; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 463 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 464 | trace_xics_ics_set_irq_lsi(srcno, srcno + ics->offset); |
David Gibson | 98ca8c0 | 2012-09-12 16:57:17 +0000 | [diff] [blame] | 465 | if (val) { |
| 466 | irq->status |= XICS_STATUS_ASSERTED; |
| 467 | } else { |
| 468 | irq->status &= ~XICS_STATUS_ASSERTED; |
| 469 | } |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 470 | ics_resend_lsi(ics, srcno); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 471 | } |
| 472 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 473 | void ics_set_irq(void *opaque, int srcno, int val) |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 474 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 475 | ICSState *ics = (ICSState *)opaque; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 476 | |
Greg Kurz | 557b456 | 2019-02-15 12:40:30 +0100 | [diff] [blame] | 477 | if (kvm_irqchip_in_kernel()) { |
| 478 | ics_kvm_set_irq(ics, srcno, val); |
| 479 | return; |
| 480 | } |
| 481 | |
Alexey Kardashevskiy | 4af8894 | 2014-05-30 19:34:12 +1000 | [diff] [blame] | 482 | if (ics->irqs[srcno].flags & XICS_FLAGS_IRQ_LSI) { |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 483 | ics_set_irq_lsi(ics, srcno, val); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 484 | } else { |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 485 | ics_set_irq_msi(ics, srcno, val); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 486 | } |
| 487 | } |
| 488 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 489 | static void ics_write_xive_msi(ICSState *ics, int srcno) |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 490 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 491 | ICSIRQState *irq = ics->irqs + srcno; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 492 | |
David Gibson | 98ca8c0 | 2012-09-12 16:57:17 +0000 | [diff] [blame] | 493 | if (!(irq->status & XICS_STATUS_MASKED_PENDING) |
| 494 | || (irq->priority == 0xff)) { |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 495 | return; |
| 496 | } |
| 497 | |
David Gibson | 98ca8c0 | 2012-09-12 16:57:17 +0000 | [diff] [blame] | 498 | irq->status &= ~XICS_STATUS_MASKED_PENDING; |
Benjamin Herrenschmidt | cc706a5 | 2016-10-03 09:24:46 +0200 | [diff] [blame] | 499 | icp_irq(ics, irq->server, srcno + ics->offset, irq->priority); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 500 | } |
| 501 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 502 | static void ics_write_xive_lsi(ICSState *ics, int srcno) |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 503 | { |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 504 | ics_resend_lsi(ics, srcno); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 505 | } |
| 506 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 507 | void ics_write_xive(ICSState *ics, int srcno, int server, |
| 508 | uint8_t priority, uint8_t saved_priority) |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 509 | { |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 510 | ICSIRQState *irq = ics->irqs + srcno; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 511 | |
| 512 | irq->server = server; |
| 513 | irq->priority = priority; |
David Gibson | 3fe719f | 2012-09-12 16:57:21 +0000 | [diff] [blame] | 514 | irq->saved_priority = saved_priority; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 515 | |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 516 | trace_xics_ics_write_xive(ics->offset + srcno, srcno, server, priority); |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 517 | |
Alexey Kardashevskiy | 4af8894 | 2014-05-30 19:34:12 +1000 | [diff] [blame] | 518 | if (ics->irqs[srcno].flags & XICS_FLAGS_IRQ_LSI) { |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 519 | ics_write_xive_lsi(ics, srcno); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 520 | } else { |
David Gibson | 28976c9 | 2019-09-24 14:13:39 +1000 | [diff] [blame] | 521 | ics_write_xive_msi(ics, srcno); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 522 | } |
| 523 | } |
| 524 | |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 525 | static void ics_reject(ICSState *ics, uint32_t nr) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 526 | { |
Cédric Le Goater | 9ae1329 | 2020-01-27 15:45:06 +0100 | [diff] [blame] | 527 | ICSStateClass *isc = ICS_GET_CLASS(ics); |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 528 | ICSIRQState *irq = ics->irqs + nr - ics->offset; |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 529 | |
Cédric Le Goater | 9ae1329 | 2020-01-27 15:45:06 +0100 | [diff] [blame] | 530 | if (isc->reject) { |
| 531 | isc->reject(ics, nr); |
| 532 | return; |
| 533 | } |
| 534 | |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 535 | trace_xics_ics_reject(nr, nr - ics->offset); |
Nikunj A Dadhania | 056b977 | 2016-09-19 11:59:29 +0530 | [diff] [blame] | 536 | if (irq->flags & XICS_FLAGS_IRQ_MSI) { |
| 537 | irq->status |= XICS_STATUS_REJECTED; |
| 538 | } else if (irq->flags & XICS_FLAGS_IRQ_LSI) { |
| 539 | irq->status &= ~XICS_STATUS_SENT; |
| 540 | } |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 541 | } |
| 542 | |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 543 | void ics_resend(ICSState *ics) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 544 | { |
Cédric Le Goater | 9ae1329 | 2020-01-27 15:45:06 +0100 | [diff] [blame] | 545 | ICSStateClass *isc = ICS_GET_CLASS(ics); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 546 | int i; |
| 547 | |
Cédric Le Goater | 9ae1329 | 2020-01-27 15:45:06 +0100 | [diff] [blame] | 548 | if (isc->resend) { |
| 549 | isc->resend(ics); |
| 550 | return; |
| 551 | } |
| 552 | |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 553 | for (i = 0; i < ics->nr_irqs; i++) { |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 554 | /* FIXME: filter by server#? */ |
Alexey Kardashevskiy | 4af8894 | 2014-05-30 19:34:12 +1000 | [diff] [blame] | 555 | if (ics->irqs[i].flags & XICS_FLAGS_IRQ_LSI) { |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 556 | ics_resend_lsi(ics, i); |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 557 | } else { |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 558 | ics_resend_msi(ics, i); |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 559 | } |
| 560 | } |
| 561 | } |
| 562 | |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 563 | static void ics_eoi(ICSState *ics, uint32_t nr) |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 564 | { |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 565 | int srcno = nr - ics->offset; |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 566 | ICSIRQState *irq = ics->irqs + srcno; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 567 | |
David Gibson | d5803c7 | 2019-09-24 13:56:47 +1000 | [diff] [blame] | 568 | trace_xics_ics_eoi(nr); |
David Gibson | 500efa2 | 2012-11-12 16:46:54 +0000 | [diff] [blame] | 569 | |
Alexey Kardashevskiy | 4af8894 | 2014-05-30 19:34:12 +1000 | [diff] [blame] | 570 | if (ics->irqs[srcno].flags & XICS_FLAGS_IRQ_LSI) { |
David Gibson | 98ca8c0 | 2012-09-12 16:57:17 +0000 | [diff] [blame] | 571 | irq->status &= ~XICS_STATUS_SENT; |
David Gibson | d07fee7 | 2012-03-07 15:12:21 +0000 | [diff] [blame] | 572 | } |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 573 | } |
| 574 | |
Cédric Le Goater | 8362941 | 2019-05-13 10:42:44 +0200 | [diff] [blame] | 575 | static void ics_reset_irq(ICSIRQState *irq) |
| 576 | { |
| 577 | irq->priority = 0xff; |
| 578 | irq->saved_priority = 0xff; |
| 579 | } |
| 580 | |
Peter Maydell | ad80e36 | 2024-04-12 17:08:07 +0100 | [diff] [blame] | 581 | static void ics_reset_hold(Object *obj, ResetType type) |
Cédric Le Goater | eeefd43 | 2018-06-25 11:17:16 +0200 | [diff] [blame] | 582 | { |
Peter Maydell | a359da4 | 2022-11-25 11:52:39 +0000 | [diff] [blame] | 583 | ICSState *ics = ICS(obj); |
Philippe Mathieu-Daudé | 7650c8f | 2022-08-19 16:39:27 +0100 | [diff] [blame] | 584 | g_autofree uint8_t *flags = g_malloc(ics->nr_irqs); |
Cédric Le Goater | eeefd43 | 2018-06-25 11:17:16 +0200 | [diff] [blame] | 585 | int i; |
Cédric Le Goater | eeefd43 | 2018-06-25 11:17:16 +0200 | [diff] [blame] | 586 | |
| 587 | for (i = 0; i < ics->nr_irqs; i++) { |
| 588 | flags[i] = ics->irqs[i].flags; |
| 589 | } |
| 590 | |
| 591 | memset(ics->irqs, 0, sizeof(ICSIRQState) * ics->nr_irqs); |
| 592 | |
| 593 | for (i = 0; i < ics->nr_irqs; i++) { |
Cédric Le Goater | 8362941 | 2019-05-13 10:42:44 +0200 | [diff] [blame] | 594 | ics_reset_irq(ics->irqs + i); |
Cédric Le Goater | eeefd43 | 2018-06-25 11:17:16 +0200 | [diff] [blame] | 595 | ics->irqs[i].flags = flags[i]; |
| 596 | } |
David Gibson | da2ef5b | 2019-09-24 14:19:22 +1000 | [diff] [blame] | 597 | |
| 598 | if (kvm_irqchip_in_kernel()) { |
| 599 | Error *local_err = NULL; |
| 600 | |
Peter Maydell | a359da4 | 2022-11-25 11:52:39 +0000 | [diff] [blame] | 601 | ics_set_kvm_state(ics, &local_err); |
David Gibson | da2ef5b | 2019-09-24 14:19:22 +1000 | [diff] [blame] | 602 | if (local_err) { |
| 603 | error_report_err(local_err); |
| 604 | } |
| 605 | } |
Cédric Le Goater | eeefd43 | 2018-06-25 11:17:16 +0200 | [diff] [blame] | 606 | } |
| 607 | |
David Gibson | da2ef5b | 2019-09-24 14:19:22 +1000 | [diff] [blame] | 608 | static void ics_reset_handler(void *dev) |
| 609 | { |
Peter Maydell | 36cdc8b | 2022-11-25 11:52:38 +0000 | [diff] [blame] | 610 | device_cold_reset(dev); |
David Gibson | da2ef5b | 2019-09-24 14:19:22 +1000 | [diff] [blame] | 611 | } |
| 612 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 613 | static void ics_realize(DeviceState *dev, Error **errp) |
David Gibson | da2ef5b | 2019-09-24 14:19:22 +1000 | [diff] [blame] | 614 | { |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 615 | ICSState *ics = ICS(dev); |
Cédric Le Goater | 4e4169f | 2017-02-27 15:29:10 +0100 | [diff] [blame] | 616 | |
Greg Kurz | b015a98 | 2019-11-18 00:20:36 +0100 | [diff] [blame] | 617 | assert(ics->xics); |
Cédric Le Goater | 4e4169f | 2017-02-27 15:29:10 +0100 | [diff] [blame] | 618 | |
Cédric Le Goater | 0a647b7 | 2018-06-25 11:17:14 +0200 | [diff] [blame] | 619 | if (!ics->nr_irqs) { |
| 620 | error_setg(errp, "Number of interrupts needs to be greater 0"); |
| 621 | return; |
Cédric Le Goater | 4e4169f | 2017-02-27 15:29:10 +0100 | [diff] [blame] | 622 | } |
Markus Armbruster | b21e238 | 2022-03-15 15:41:56 +0100 | [diff] [blame] | 623 | ics->irqs = g_new0(ICSIRQState, ics->nr_irqs); |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 624 | |
| 625 | qemu_register_reset(ics_reset_handler, ics); |
Cédric Le Goater | 4e4169f | 2017-02-27 15:29:10 +0100 | [diff] [blame] | 626 | } |
| 627 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 628 | static void ics_instance_init(Object *obj) |
Cédric Le Goater | 815049a | 2018-06-25 11:17:15 +0200 | [diff] [blame] | 629 | { |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 630 | ICSState *ics = ICS(obj); |
Cédric Le Goater | 815049a | 2018-06-25 11:17:15 +0200 | [diff] [blame] | 631 | |
| 632 | ics->offset = XICS_IRQ_BASE; |
| 633 | } |
| 634 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 635 | static int ics_pre_save(void *opaque) |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 636 | { |
| 637 | ICSState *ics = opaque; |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 638 | |
Greg Kurz | d80b2cc | 2019-02-15 12:40:18 +0100 | [diff] [blame] | 639 | if (kvm_irqchip_in_kernel()) { |
| 640 | ics_get_kvm_state(ics); |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 641 | } |
| 642 | |
| 643 | return 0; |
| 644 | } |
| 645 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 646 | static int ics_post_load(void *opaque, int version_id) |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 647 | { |
| 648 | ICSState *ics = opaque; |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 649 | |
Greg Kurz | d80b2cc | 2019-02-15 12:40:18 +0100 | [diff] [blame] | 650 | if (kvm_irqchip_in_kernel()) { |
Greg Kurz | 330a21e | 2019-06-17 15:46:57 +0200 | [diff] [blame] | 651 | Error *local_err = NULL; |
| 652 | int ret; |
| 653 | |
| 654 | ret = ics_set_kvm_state(ics, &local_err); |
| 655 | if (ret < 0) { |
| 656 | error_report_err(local_err); |
| 657 | return ret; |
| 658 | } |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 659 | } |
| 660 | |
| 661 | return 0; |
| 662 | } |
| 663 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 664 | static const VMStateDescription vmstate_ics_irq = { |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 665 | .name = "ics/irq", |
| 666 | .version_id = 2, |
| 667 | .minimum_version_id = 1, |
Richard Henderson | 45b1f81 | 2023-12-21 14:16:15 +1100 | [diff] [blame] | 668 | .fields = (const VMStateField[]) { |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 669 | VMSTATE_UINT32(server, ICSIRQState), |
| 670 | VMSTATE_UINT8(priority, ICSIRQState), |
| 671 | VMSTATE_UINT8(saved_priority, ICSIRQState), |
| 672 | VMSTATE_UINT8(status, ICSIRQState), |
| 673 | VMSTATE_UINT8(flags, ICSIRQState), |
| 674 | VMSTATE_END_OF_LIST() |
| 675 | }, |
| 676 | }; |
| 677 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 678 | static const VMStateDescription vmstate_ics = { |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 679 | .name = "ics", |
| 680 | .version_id = 1, |
| 681 | .minimum_version_id = 1, |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 682 | .pre_save = ics_pre_save, |
| 683 | .post_load = ics_post_load, |
Richard Henderson | 45b1f81 | 2023-12-21 14:16:15 +1100 | [diff] [blame] | 684 | .fields = (const VMStateField[]) { |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 685 | /* Sanity check */ |
| 686 | VMSTATE_UINT32_EQUAL(nr_irqs, ICSState, NULL), |
| 687 | |
| 688 | VMSTATE_STRUCT_VARRAY_POINTER_UINT32(irqs, ICSState, nr_irqs, |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 689 | vmstate_ics_irq, |
Cédric Le Goater | c8b1846 | 2018-06-25 11:17:17 +0200 | [diff] [blame] | 690 | ICSIRQState), |
| 691 | VMSTATE_END_OF_LIST() |
| 692 | }, |
| 693 | }; |
| 694 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 695 | static Property ics_properties[] = { |
Cédric Le Goater | 0a647b7 | 2018-06-25 11:17:14 +0200 | [diff] [blame] | 696 | DEFINE_PROP_UINT32("nr-irqs", ICSState, nr_irqs, 0), |
Greg Kurz | b015a98 | 2019-11-18 00:20:36 +0100 | [diff] [blame] | 697 | DEFINE_PROP_LINK(ICS_PROP_XICS, ICSState, xics, TYPE_XICS_FABRIC, |
| 698 | XICSFabric *), |
Cédric Le Goater | 0a647b7 | 2018-06-25 11:17:14 +0200 | [diff] [blame] | 699 | DEFINE_PROP_END_OF_LIST(), |
| 700 | }; |
| 701 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 702 | static void ics_class_init(ObjectClass *klass, void *data) |
Cédric Le Goater | 4e4169f | 2017-02-27 15:29:10 +0100 | [diff] [blame] | 703 | { |
| 704 | DeviceClass *dc = DEVICE_CLASS(klass); |
Peter Maydell | a359da4 | 2022-11-25 11:52:39 +0000 | [diff] [blame] | 705 | ResettableClass *rc = RESETTABLE_CLASS(klass); |
Cédric Le Goater | 4e4169f | 2017-02-27 15:29:10 +0100 | [diff] [blame] | 706 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 707 | dc->realize = ics_realize; |
Marc-André Lureau | 4f67d30 | 2020-01-10 19:30:32 +0400 | [diff] [blame] | 708 | device_class_set_props(dc, ics_properties); |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 709 | dc->vmsd = &vmstate_ics; |
Greg Kurz | e6144bf | 2019-10-04 10:37:47 +0200 | [diff] [blame] | 710 | /* |
| 711 | * Reason: part of XICS interrupt controller, needs to be wired up, |
| 712 | * e.g. by spapr_irq_init(). |
| 713 | */ |
| 714 | dc->user_creatable = false; |
Peter Maydell | a359da4 | 2022-11-25 11:52:39 +0000 | [diff] [blame] | 715 | rc->phases.hold = ics_reset_hold; |
Cédric Le Goater | 4e4169f | 2017-02-27 15:29:10 +0100 | [diff] [blame] | 716 | } |
| 717 | |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 718 | static const TypeInfo ics_info = { |
| 719 | .name = TYPE_ICS, |
Benjamin Herrenschmidt | d4d7a59 | 2016-10-03 09:24:47 +0200 | [diff] [blame] | 720 | .parent = TYPE_DEVICE, |
Benjamin Herrenschmidt | d4d7a59 | 2016-10-03 09:24:47 +0200 | [diff] [blame] | 721 | .instance_size = sizeof(ICSState), |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 722 | .instance_init = ics_instance_init, |
| 723 | .class_init = ics_class_init, |
Benjamin Herrenschmidt | d4d7a59 | 2016-10-03 09:24:47 +0200 | [diff] [blame] | 724 | .class_size = sizeof(ICSStateClass), |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 725 | }; |
| 726 | |
Cédric Le Goater | 51b1800 | 2017-02-27 15:29:14 +0100 | [diff] [blame] | 727 | static const TypeInfo xics_fabric_info = { |
| 728 | .name = TYPE_XICS_FABRIC, |
| 729 | .parent = TYPE_INTERFACE, |
| 730 | .class_size = sizeof(XICSFabricClass), |
| 731 | }; |
| 732 | |
David Gibson | b5cec4c | 2011-04-01 15:15:25 +1100 | [diff] [blame] | 733 | /* |
| 734 | * Exported functions |
| 735 | */ |
Cédric Le Goater | b4f27d7 | 2017-02-27 15:29:25 +0100 | [diff] [blame] | 736 | ICPState *xics_icp_get(XICSFabric *xi, int server) |
| 737 | { |
| 738 | XICSFabricClass *xic = XICS_FABRIC_GET_CLASS(xi); |
| 739 | |
| 740 | return xic->icp_get(xi, server); |
| 741 | } |
| 742 | |
Benjamin Herrenschmidt | 9c7027b | 2016-06-29 00:35:13 +0530 | [diff] [blame] | 743 | void ics_set_irq_type(ICSState *ics, int srcno, bool lsi) |
Alexey Kardashevskiy | 4af8894 | 2014-05-30 19:34:12 +1000 | [diff] [blame] | 744 | { |
| 745 | assert(!(ics->irqs[srcno].flags & XICS_FLAGS_IRQ_MASK)); |
| 746 | |
| 747 | ics->irqs[srcno].flags |= |
| 748 | lsi ? XICS_FLAGS_IRQ_LSI : XICS_FLAGS_IRQ_MSI; |
Greg Kurz | 6cead90 | 2019-02-19 18:18:03 +0100 | [diff] [blame] | 749 | |
| 750 | if (kvm_irqchip_in_kernel()) { |
Greg Kurz | 330a21e | 2019-06-17 15:46:57 +0200 | [diff] [blame] | 751 | Error *local_err = NULL; |
| 752 | |
Cédric Le Goater | 8362941 | 2019-05-13 10:42:44 +0200 | [diff] [blame] | 753 | ics_reset_irq(ics->irqs + srcno); |
Greg Kurz | 330a21e | 2019-06-17 15:46:57 +0200 | [diff] [blame] | 754 | ics_set_kvm_state_one(ics, srcno, &local_err); |
| 755 | if (local_err) { |
| 756 | error_report_err(local_err); |
| 757 | } |
Greg Kurz | 6cead90 | 2019-02-19 18:18:03 +0100 | [diff] [blame] | 758 | } |
Alexey Kardashevskiy | 4af8894 | 2014-05-30 19:34:12 +1000 | [diff] [blame] | 759 | } |
| 760 | |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 761 | static void xics_register_types(void) |
| 762 | { |
David Gibson | 642e927 | 2019-09-24 15:29:25 +1000 | [diff] [blame] | 763 | type_register_static(&ics_info); |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 764 | type_register_static(&icp_info); |
Cédric Le Goater | 51b1800 | 2017-02-27 15:29:14 +0100 | [diff] [blame] | 765 | type_register_static(&xics_fabric_info); |
Anthony Liguori | c04d6cf | 2013-07-18 14:33:04 -0500 | [diff] [blame] | 766 | } |
| 767 | |
| 768 | type_init(xics_register_types) |