118f6290aSShashi Mallela /* 218f6290aSShashi Mallela * ITS emulation for a GICv3-based system 318f6290aSShashi Mallela * 418f6290aSShashi Mallela * Copyright Linaro.org 2021 518f6290aSShashi Mallela * 618f6290aSShashi Mallela * Authors: 718f6290aSShashi Mallela * Shashi Mallela <shashi.mallela@linaro.org> 818f6290aSShashi Mallela * 918f6290aSShashi Mallela * This work is licensed under the terms of the GNU GPL, version 2 or (at your 1018f6290aSShashi Mallela * option) any later version. See the COPYING file in the top-level directory. 1118f6290aSShashi Mallela * 1218f6290aSShashi Mallela */ 1318f6290aSShashi Mallela 1418f6290aSShashi Mallela #include "qemu/osdep.h" 1518f6290aSShashi Mallela #include "qemu/log.h" 1618f6290aSShashi Mallela #include "hw/qdev-properties.h" 1718f6290aSShashi Mallela #include "hw/intc/arm_gicv3_its_common.h" 1818f6290aSShashi Mallela #include "gicv3_internal.h" 1918f6290aSShashi Mallela #include "qom/object.h" 2018f6290aSShashi Mallela #include "qapi/error.h" 2118f6290aSShashi Mallela 2218f6290aSShashi Mallela typedef struct GICv3ITSClass GICv3ITSClass; 2318f6290aSShashi Mallela /* This is reusing the GICv3ITSState typedef from ARM_GICV3_ITS_COMMON */ 2418f6290aSShashi Mallela DECLARE_OBJ_CHECKERS(GICv3ITSState, GICv3ITSClass, 2518f6290aSShashi Mallela ARM_GICV3_ITS, TYPE_ARM_GICV3_ITS) 2618f6290aSShashi Mallela 2718f6290aSShashi Mallela struct GICv3ITSClass { 2818f6290aSShashi Mallela GICv3ITSCommonClass parent_class; 2918f6290aSShashi Mallela void (*parent_reset)(DeviceState *dev); 3018f6290aSShashi Mallela }; 3118f6290aSShashi Mallela 32*c694cb4cSShashi Mallela /* 33*c694cb4cSShashi Mallela * This is an internal enum used to distinguish between LPI triggered 34*c694cb4cSShashi Mallela * via command queue and LPI triggered via gits_translater write. 35*c694cb4cSShashi Mallela */ 36*c694cb4cSShashi Mallela typedef enum ItsCmdType { 37*c694cb4cSShashi Mallela NONE = 0, /* internal indication for GITS_TRANSLATER write */ 38*c694cb4cSShashi Mallela CLEAR = 1, 39*c694cb4cSShashi Mallela DISCARD = 2, 40*c694cb4cSShashi Mallela INTERRUPT = 3, 41*c694cb4cSShashi Mallela } ItsCmdType; 42*c694cb4cSShashi Mallela 43*c694cb4cSShashi Mallela typedef struct { 44*c694cb4cSShashi Mallela uint32_t iteh; 45*c694cb4cSShashi Mallela uint64_t itel; 46*c694cb4cSShashi Mallela } IteEntry; 47*c694cb4cSShashi Mallela 481b08e436SShashi Mallela static uint64_t baser_base_addr(uint64_t value, uint32_t page_sz) 491b08e436SShashi Mallela { 501b08e436SShashi Mallela uint64_t result = 0; 511b08e436SShashi Mallela 521b08e436SShashi Mallela switch (page_sz) { 531b08e436SShashi Mallela case GITS_PAGE_SIZE_4K: 541b08e436SShashi Mallela case GITS_PAGE_SIZE_16K: 551b08e436SShashi Mallela result = FIELD_EX64(value, GITS_BASER, PHYADDR) << 12; 561b08e436SShashi Mallela break; 571b08e436SShashi Mallela 581b08e436SShashi Mallela case GITS_PAGE_SIZE_64K: 591b08e436SShashi Mallela result = FIELD_EX64(value, GITS_BASER, PHYADDRL_64K) << 16; 601b08e436SShashi Mallela result |= FIELD_EX64(value, GITS_BASER, PHYADDRH_64K) << 48; 611b08e436SShashi Mallela break; 621b08e436SShashi Mallela 631b08e436SShashi Mallela default: 641b08e436SShashi Mallela break; 651b08e436SShashi Mallela } 661b08e436SShashi Mallela return result; 671b08e436SShashi Mallela } 681b08e436SShashi Mallela 69*c694cb4cSShashi Mallela static bool get_cte(GICv3ITSState *s, uint16_t icid, uint64_t *cte, 70*c694cb4cSShashi Mallela MemTxResult *res) 71*c694cb4cSShashi Mallela { 72*c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 73*c694cb4cSShashi Mallela uint64_t l2t_addr; 74*c694cb4cSShashi Mallela uint64_t value; 75*c694cb4cSShashi Mallela bool valid_l2t; 76*c694cb4cSShashi Mallela uint32_t l2t_id; 77*c694cb4cSShashi Mallela uint32_t max_l2_entries; 78*c694cb4cSShashi Mallela 79*c694cb4cSShashi Mallela if (s->ct.indirect) { 80*c694cb4cSShashi Mallela l2t_id = icid / (s->ct.page_sz / L1TABLE_ENTRY_SIZE); 81*c694cb4cSShashi Mallela 82*c694cb4cSShashi Mallela value = address_space_ldq_le(as, 83*c694cb4cSShashi Mallela s->ct.base_addr + 84*c694cb4cSShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 85*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 86*c694cb4cSShashi Mallela 87*c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 88*c694cb4cSShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 89*c694cb4cSShashi Mallela 90*c694cb4cSShashi Mallela if (valid_l2t) { 91*c694cb4cSShashi Mallela max_l2_entries = s->ct.page_sz / s->ct.entry_sz; 92*c694cb4cSShashi Mallela 93*c694cb4cSShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 94*c694cb4cSShashi Mallela 95*c694cb4cSShashi Mallela *cte = address_space_ldq_le(as, l2t_addr + 96*c694cb4cSShashi Mallela ((icid % max_l2_entries) * GITS_CTE_SIZE), 97*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 98*c694cb4cSShashi Mallela } 99*c694cb4cSShashi Mallela } 100*c694cb4cSShashi Mallela } else { 101*c694cb4cSShashi Mallela /* Flat level table */ 102*c694cb4cSShashi Mallela *cte = address_space_ldq_le(as, s->ct.base_addr + 103*c694cb4cSShashi Mallela (icid * GITS_CTE_SIZE), 104*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 105*c694cb4cSShashi Mallela } 106*c694cb4cSShashi Mallela 107*c694cb4cSShashi Mallela return (*cte & TABLE_ENTRY_VALID_MASK) != 0; 108*c694cb4cSShashi Mallela } 109*c694cb4cSShashi Mallela 110*c694cb4cSShashi Mallela static bool update_ite(GICv3ITSState *s, uint32_t eventid, uint64_t dte, 111*c694cb4cSShashi Mallela IteEntry ite) 112*c694cb4cSShashi Mallela { 113*c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 114*c694cb4cSShashi Mallela uint64_t itt_addr; 115*c694cb4cSShashi Mallela MemTxResult res = MEMTX_OK; 116*c694cb4cSShashi Mallela 117*c694cb4cSShashi Mallela itt_addr = (dte & GITS_DTE_ITTADDR_MASK) >> GITS_DTE_ITTADDR_SHIFT; 118*c694cb4cSShashi Mallela itt_addr <<= ITTADDR_SHIFT; /* 256 byte aligned */ 119*c694cb4cSShashi Mallela 120*c694cb4cSShashi Mallela address_space_stq_le(as, itt_addr + (eventid * (sizeof(uint64_t) + 121*c694cb4cSShashi Mallela sizeof(uint32_t))), ite.itel, MEMTXATTRS_UNSPECIFIED, 122*c694cb4cSShashi Mallela &res); 123*c694cb4cSShashi Mallela 124*c694cb4cSShashi Mallela if (res == MEMTX_OK) { 125*c694cb4cSShashi Mallela address_space_stl_le(as, itt_addr + (eventid * (sizeof(uint64_t) + 126*c694cb4cSShashi Mallela sizeof(uint32_t))) + sizeof(uint32_t), ite.iteh, 127*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 128*c694cb4cSShashi Mallela } 129*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 130*c694cb4cSShashi Mallela return false; 131*c694cb4cSShashi Mallela } else { 132*c694cb4cSShashi Mallela return true; 133*c694cb4cSShashi Mallela } 134*c694cb4cSShashi Mallela } 135*c694cb4cSShashi Mallela 136*c694cb4cSShashi Mallela static bool get_ite(GICv3ITSState *s, uint32_t eventid, uint64_t dte, 137*c694cb4cSShashi Mallela uint16_t *icid, uint32_t *pIntid, MemTxResult *res) 138*c694cb4cSShashi Mallela { 139*c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 140*c694cb4cSShashi Mallela uint64_t itt_addr; 141*c694cb4cSShashi Mallela bool status = false; 142*c694cb4cSShashi Mallela IteEntry ite = {}; 143*c694cb4cSShashi Mallela 144*c694cb4cSShashi Mallela itt_addr = (dte & GITS_DTE_ITTADDR_MASK) >> GITS_DTE_ITTADDR_SHIFT; 145*c694cb4cSShashi Mallela itt_addr <<= ITTADDR_SHIFT; /* 256 byte aligned */ 146*c694cb4cSShashi Mallela 147*c694cb4cSShashi Mallela ite.itel = address_space_ldq_le(as, itt_addr + 148*c694cb4cSShashi Mallela (eventid * (sizeof(uint64_t) + 149*c694cb4cSShashi Mallela sizeof(uint32_t))), MEMTXATTRS_UNSPECIFIED, 150*c694cb4cSShashi Mallela res); 151*c694cb4cSShashi Mallela 152*c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 153*c694cb4cSShashi Mallela ite.iteh = address_space_ldl_le(as, itt_addr + 154*c694cb4cSShashi Mallela (eventid * (sizeof(uint64_t) + 155*c694cb4cSShashi Mallela sizeof(uint32_t))) + sizeof(uint32_t), 156*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 157*c694cb4cSShashi Mallela 158*c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 159*c694cb4cSShashi Mallela if (ite.itel & TABLE_ENTRY_VALID_MASK) { 160*c694cb4cSShashi Mallela if ((ite.itel >> ITE_ENTRY_INTTYPE_SHIFT) & 161*c694cb4cSShashi Mallela GITS_TYPE_PHYSICAL) { 162*c694cb4cSShashi Mallela *pIntid = (ite.itel & ITE_ENTRY_INTID_MASK) >> 163*c694cb4cSShashi Mallela ITE_ENTRY_INTID_SHIFT; 164*c694cb4cSShashi Mallela *icid = ite.iteh & ITE_ENTRY_ICID_MASK; 165*c694cb4cSShashi Mallela status = true; 166*c694cb4cSShashi Mallela } 167*c694cb4cSShashi Mallela } 168*c694cb4cSShashi Mallela } 169*c694cb4cSShashi Mallela } 170*c694cb4cSShashi Mallela return status; 171*c694cb4cSShashi Mallela } 172*c694cb4cSShashi Mallela 173*c694cb4cSShashi Mallela static uint64_t get_dte(GICv3ITSState *s, uint32_t devid, MemTxResult *res) 174*c694cb4cSShashi Mallela { 175*c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 176*c694cb4cSShashi Mallela uint64_t l2t_addr; 177*c694cb4cSShashi Mallela uint64_t value; 178*c694cb4cSShashi Mallela bool valid_l2t; 179*c694cb4cSShashi Mallela uint32_t l2t_id; 180*c694cb4cSShashi Mallela uint32_t max_l2_entries; 181*c694cb4cSShashi Mallela 182*c694cb4cSShashi Mallela if (s->dt.indirect) { 183*c694cb4cSShashi Mallela l2t_id = devid / (s->dt.page_sz / L1TABLE_ENTRY_SIZE); 184*c694cb4cSShashi Mallela 185*c694cb4cSShashi Mallela value = address_space_ldq_le(as, 186*c694cb4cSShashi Mallela s->dt.base_addr + 187*c694cb4cSShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 188*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 189*c694cb4cSShashi Mallela 190*c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 191*c694cb4cSShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 192*c694cb4cSShashi Mallela 193*c694cb4cSShashi Mallela if (valid_l2t) { 194*c694cb4cSShashi Mallela max_l2_entries = s->dt.page_sz / s->dt.entry_sz; 195*c694cb4cSShashi Mallela 196*c694cb4cSShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 197*c694cb4cSShashi Mallela 198*c694cb4cSShashi Mallela value = address_space_ldq_le(as, l2t_addr + 199*c694cb4cSShashi Mallela ((devid % max_l2_entries) * GITS_DTE_SIZE), 200*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 201*c694cb4cSShashi Mallela } 202*c694cb4cSShashi Mallela } 203*c694cb4cSShashi Mallela } else { 204*c694cb4cSShashi Mallela /* Flat level table */ 205*c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->dt.base_addr + 206*c694cb4cSShashi Mallela (devid * GITS_DTE_SIZE), 207*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 208*c694cb4cSShashi Mallela } 209*c694cb4cSShashi Mallela 210*c694cb4cSShashi Mallela return value; 211*c694cb4cSShashi Mallela } 212*c694cb4cSShashi Mallela 213*c694cb4cSShashi Mallela /* 214*c694cb4cSShashi Mallela * This function handles the processing of following commands based on 215*c694cb4cSShashi Mallela * the ItsCmdType parameter passed:- 216*c694cb4cSShashi Mallela * 1. triggering of lpi interrupt translation via ITS INT command 217*c694cb4cSShashi Mallela * 2. triggering of lpi interrupt translation via gits_translater register 218*c694cb4cSShashi Mallela * 3. handling of ITS CLEAR command 219*c694cb4cSShashi Mallela * 4. handling of ITS DISCARD command 220*c694cb4cSShashi Mallela */ 221*c694cb4cSShashi Mallela static bool process_its_cmd(GICv3ITSState *s, uint64_t value, uint32_t offset, 222*c694cb4cSShashi Mallela ItsCmdType cmd) 223*c694cb4cSShashi Mallela { 224*c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 225*c694cb4cSShashi Mallela uint32_t devid, eventid; 226*c694cb4cSShashi Mallela MemTxResult res = MEMTX_OK; 227*c694cb4cSShashi Mallela bool dte_valid; 228*c694cb4cSShashi Mallela uint64_t dte = 0; 229*c694cb4cSShashi Mallela uint32_t max_eventid; 230*c694cb4cSShashi Mallela uint16_t icid = 0; 231*c694cb4cSShashi Mallela uint32_t pIntid = 0; 232*c694cb4cSShashi Mallela bool ite_valid = false; 233*c694cb4cSShashi Mallela uint64_t cte = 0; 234*c694cb4cSShashi Mallela bool cte_valid = false; 235*c694cb4cSShashi Mallela bool result = false; 236*c694cb4cSShashi Mallela 237*c694cb4cSShashi Mallela if (cmd == NONE) { 238*c694cb4cSShashi Mallela devid = offset; 239*c694cb4cSShashi Mallela } else { 240*c694cb4cSShashi Mallela devid = ((value & DEVID_MASK) >> DEVID_SHIFT); 241*c694cb4cSShashi Mallela 242*c694cb4cSShashi Mallela offset += NUM_BYTES_IN_DW; 243*c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 244*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 245*c694cb4cSShashi Mallela } 246*c694cb4cSShashi Mallela 247*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 248*c694cb4cSShashi Mallela return result; 249*c694cb4cSShashi Mallela } 250*c694cb4cSShashi Mallela 251*c694cb4cSShashi Mallela eventid = (value & EVENTID_MASK); 252*c694cb4cSShashi Mallela 253*c694cb4cSShashi Mallela dte = get_dte(s, devid, &res); 254*c694cb4cSShashi Mallela 255*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 256*c694cb4cSShashi Mallela return result; 257*c694cb4cSShashi Mallela } 258*c694cb4cSShashi Mallela dte_valid = dte & TABLE_ENTRY_VALID_MASK; 259*c694cb4cSShashi Mallela 260*c694cb4cSShashi Mallela if (dte_valid) { 261*c694cb4cSShashi Mallela max_eventid = (1UL << (((dte >> 1U) & SIZE_MASK) + 1)); 262*c694cb4cSShashi Mallela 263*c694cb4cSShashi Mallela ite_valid = get_ite(s, eventid, dte, &icid, &pIntid, &res); 264*c694cb4cSShashi Mallela 265*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 266*c694cb4cSShashi Mallela return result; 267*c694cb4cSShashi Mallela } 268*c694cb4cSShashi Mallela 269*c694cb4cSShashi Mallela if (ite_valid) { 270*c694cb4cSShashi Mallela cte_valid = get_cte(s, icid, &cte, &res); 271*c694cb4cSShashi Mallela } 272*c694cb4cSShashi Mallela 273*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 274*c694cb4cSShashi Mallela return result; 275*c694cb4cSShashi Mallela } 276*c694cb4cSShashi Mallela } 277*c694cb4cSShashi Mallela 278*c694cb4cSShashi Mallela if ((devid > s->dt.maxids.max_devids) || !dte_valid || !ite_valid || 279*c694cb4cSShashi Mallela !cte_valid || (eventid > max_eventid)) { 280*c694cb4cSShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 281*c694cb4cSShashi Mallela "%s: invalid command attributes " 282*c694cb4cSShashi Mallela "devid %d or eventid %d or invalid dte %d or" 283*c694cb4cSShashi Mallela "invalid cte %d or invalid ite %d\n", 284*c694cb4cSShashi Mallela __func__, devid, eventid, dte_valid, cte_valid, 285*c694cb4cSShashi Mallela ite_valid); 286*c694cb4cSShashi Mallela /* 287*c694cb4cSShashi Mallela * in this implementation, in case of error 288*c694cb4cSShashi Mallela * we ignore this command and move onto the next 289*c694cb4cSShashi Mallela * command in the queue 290*c694cb4cSShashi Mallela */ 291*c694cb4cSShashi Mallela } else { 292*c694cb4cSShashi Mallela /* 293*c694cb4cSShashi Mallela * Current implementation only supports rdbase == procnum 294*c694cb4cSShashi Mallela * Hence rdbase physical address is ignored 295*c694cb4cSShashi Mallela */ 296*c694cb4cSShashi Mallela if (cmd == DISCARD) { 297*c694cb4cSShashi Mallela IteEntry ite = {}; 298*c694cb4cSShashi Mallela /* remove mapping from interrupt translation table */ 299*c694cb4cSShashi Mallela result = update_ite(s, eventid, dte, ite); 300*c694cb4cSShashi Mallela } 301*c694cb4cSShashi Mallela } 302*c694cb4cSShashi Mallela 303*c694cb4cSShashi Mallela return result; 304*c694cb4cSShashi Mallela } 305*c694cb4cSShashi Mallela 306*c694cb4cSShashi Mallela static bool process_mapti(GICv3ITSState *s, uint64_t value, uint32_t offset, 307*c694cb4cSShashi Mallela bool ignore_pInt) 308*c694cb4cSShashi Mallela { 309*c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 310*c694cb4cSShashi Mallela uint32_t devid, eventid; 311*c694cb4cSShashi Mallela uint32_t pIntid = 0; 312*c694cb4cSShashi Mallela uint32_t max_eventid, max_Intid; 313*c694cb4cSShashi Mallela bool dte_valid; 314*c694cb4cSShashi Mallela MemTxResult res = MEMTX_OK; 315*c694cb4cSShashi Mallela uint16_t icid = 0; 316*c694cb4cSShashi Mallela uint64_t dte = 0; 317*c694cb4cSShashi Mallela IteEntry ite; 318*c694cb4cSShashi Mallela uint32_t int_spurious = INTID_SPURIOUS; 319*c694cb4cSShashi Mallela bool result = false; 320*c694cb4cSShashi Mallela 321*c694cb4cSShashi Mallela devid = ((value & DEVID_MASK) >> DEVID_SHIFT); 322*c694cb4cSShashi Mallela offset += NUM_BYTES_IN_DW; 323*c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 324*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 325*c694cb4cSShashi Mallela 326*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 327*c694cb4cSShashi Mallela return result; 328*c694cb4cSShashi Mallela } 329*c694cb4cSShashi Mallela 330*c694cb4cSShashi Mallela eventid = (value & EVENTID_MASK); 331*c694cb4cSShashi Mallela 332*c694cb4cSShashi Mallela if (!ignore_pInt) { 333*c694cb4cSShashi Mallela pIntid = ((value & pINTID_MASK) >> pINTID_SHIFT); 334*c694cb4cSShashi Mallela } 335*c694cb4cSShashi Mallela 336*c694cb4cSShashi Mallela offset += NUM_BYTES_IN_DW; 337*c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 338*c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 339*c694cb4cSShashi Mallela 340*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 341*c694cb4cSShashi Mallela return result; 342*c694cb4cSShashi Mallela } 343*c694cb4cSShashi Mallela 344*c694cb4cSShashi Mallela icid = value & ICID_MASK; 345*c694cb4cSShashi Mallela 346*c694cb4cSShashi Mallela dte = get_dte(s, devid, &res); 347*c694cb4cSShashi Mallela 348*c694cb4cSShashi Mallela if (res != MEMTX_OK) { 349*c694cb4cSShashi Mallela return result; 350*c694cb4cSShashi Mallela } 351*c694cb4cSShashi Mallela dte_valid = dte & TABLE_ENTRY_VALID_MASK; 352*c694cb4cSShashi Mallela 353*c694cb4cSShashi Mallela max_eventid = (1UL << (((dte >> 1U) & SIZE_MASK) + 1)); 354*c694cb4cSShashi Mallela 355*c694cb4cSShashi Mallela if (!ignore_pInt) { 356*c694cb4cSShashi Mallela max_Intid = (1ULL << (GICD_TYPER_IDBITS + 1)) - 1; 357*c694cb4cSShashi Mallela } 358*c694cb4cSShashi Mallela 359*c694cb4cSShashi Mallela if ((devid > s->dt.maxids.max_devids) || (icid > s->ct.maxids.max_collids) 360*c694cb4cSShashi Mallela || !dte_valid || (eventid > max_eventid) || 361*c694cb4cSShashi Mallela (!ignore_pInt && (((pIntid < GICV3_LPI_INTID_START) || 362*c694cb4cSShashi Mallela (pIntid > max_Intid)) && (pIntid != INTID_SPURIOUS)))) { 363*c694cb4cSShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 364*c694cb4cSShashi Mallela "%s: invalid command attributes " 365*c694cb4cSShashi Mallela "devid %d or icid %d or eventid %d or pIntid %d or" 366*c694cb4cSShashi Mallela "unmapped dte %d\n", __func__, devid, icid, eventid, 367*c694cb4cSShashi Mallela pIntid, dte_valid); 368*c694cb4cSShashi Mallela /* 369*c694cb4cSShashi Mallela * in this implementation, in case of error 370*c694cb4cSShashi Mallela * we ignore this command and move onto the next 371*c694cb4cSShashi Mallela * command in the queue 372*c694cb4cSShashi Mallela */ 373*c694cb4cSShashi Mallela } else { 374*c694cb4cSShashi Mallela /* add ite entry to interrupt translation table */ 375*c694cb4cSShashi Mallela ite.itel = (dte_valid & TABLE_ENTRY_VALID_MASK) | 376*c694cb4cSShashi Mallela (GITS_TYPE_PHYSICAL << ITE_ENTRY_INTTYPE_SHIFT); 377*c694cb4cSShashi Mallela 378*c694cb4cSShashi Mallela if (ignore_pInt) { 379*c694cb4cSShashi Mallela ite.itel |= (eventid << ITE_ENTRY_INTID_SHIFT); 380*c694cb4cSShashi Mallela } else { 381*c694cb4cSShashi Mallela ite.itel |= (pIntid << ITE_ENTRY_INTID_SHIFT); 382*c694cb4cSShashi Mallela } 383*c694cb4cSShashi Mallela ite.itel |= (int_spurious << ITE_ENTRY_INTSP_SHIFT); 384*c694cb4cSShashi Mallela ite.iteh = icid; 385*c694cb4cSShashi Mallela 386*c694cb4cSShashi Mallela result = update_ite(s, eventid, dte, ite); 387*c694cb4cSShashi Mallela } 388*c694cb4cSShashi Mallela 389*c694cb4cSShashi Mallela return result; 390*c694cb4cSShashi Mallela } 391*c694cb4cSShashi Mallela 3927eca39e0SShashi Mallela static bool update_cte(GICv3ITSState *s, uint16_t icid, bool valid, 3937eca39e0SShashi Mallela uint64_t rdbase) 3947eca39e0SShashi Mallela { 3957eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 3967eca39e0SShashi Mallela uint64_t value; 3977eca39e0SShashi Mallela uint64_t l2t_addr; 3987eca39e0SShashi Mallela bool valid_l2t; 3997eca39e0SShashi Mallela uint32_t l2t_id; 4007eca39e0SShashi Mallela uint32_t max_l2_entries; 4017eca39e0SShashi Mallela uint64_t cte = 0; 4027eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 4037eca39e0SShashi Mallela 4047eca39e0SShashi Mallela if (!s->ct.valid) { 4057eca39e0SShashi Mallela return true; 4067eca39e0SShashi Mallela } 4077eca39e0SShashi Mallela 4087eca39e0SShashi Mallela if (valid) { 4097eca39e0SShashi Mallela /* add mapping entry to collection table */ 4107eca39e0SShashi Mallela cte = (valid & TABLE_ENTRY_VALID_MASK) | (rdbase << 1ULL); 4117eca39e0SShashi Mallela } 4127eca39e0SShashi Mallela 4137eca39e0SShashi Mallela /* 4147eca39e0SShashi Mallela * The specification defines the format of level 1 entries of a 4157eca39e0SShashi Mallela * 2-level table, but the format of level 2 entries and the format 4167eca39e0SShashi Mallela * of flat-mapped tables is IMPDEF. 4177eca39e0SShashi Mallela */ 4187eca39e0SShashi Mallela if (s->ct.indirect) { 4197eca39e0SShashi Mallela l2t_id = icid / (s->ct.page_sz / L1TABLE_ENTRY_SIZE); 4207eca39e0SShashi Mallela 4217eca39e0SShashi Mallela value = address_space_ldq_le(as, 4227eca39e0SShashi Mallela s->ct.base_addr + 4237eca39e0SShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 4247eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 4257eca39e0SShashi Mallela 4267eca39e0SShashi Mallela if (res != MEMTX_OK) { 4277eca39e0SShashi Mallela return false; 4287eca39e0SShashi Mallela } 4297eca39e0SShashi Mallela 4307eca39e0SShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 4317eca39e0SShashi Mallela 4327eca39e0SShashi Mallela if (valid_l2t) { 4337eca39e0SShashi Mallela max_l2_entries = s->ct.page_sz / s->ct.entry_sz; 4347eca39e0SShashi Mallela 4357eca39e0SShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 4367eca39e0SShashi Mallela 4377eca39e0SShashi Mallela address_space_stq_le(as, l2t_addr + 4387eca39e0SShashi Mallela ((icid % max_l2_entries) * GITS_CTE_SIZE), 4397eca39e0SShashi Mallela cte, MEMTXATTRS_UNSPECIFIED, &res); 4407eca39e0SShashi Mallela } 4417eca39e0SShashi Mallela } else { 4427eca39e0SShashi Mallela /* Flat level table */ 4437eca39e0SShashi Mallela address_space_stq_le(as, s->ct.base_addr + (icid * GITS_CTE_SIZE), 4447eca39e0SShashi Mallela cte, MEMTXATTRS_UNSPECIFIED, &res); 4457eca39e0SShashi Mallela } 4467eca39e0SShashi Mallela if (res != MEMTX_OK) { 4477eca39e0SShashi Mallela return false; 4487eca39e0SShashi Mallela } else { 4497eca39e0SShashi Mallela return true; 4507eca39e0SShashi Mallela } 4517eca39e0SShashi Mallela } 4527eca39e0SShashi Mallela 4537eca39e0SShashi Mallela static bool process_mapc(GICv3ITSState *s, uint32_t offset) 4547eca39e0SShashi Mallela { 4557eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 4567eca39e0SShashi Mallela uint16_t icid; 4577eca39e0SShashi Mallela uint64_t rdbase; 4587eca39e0SShashi Mallela bool valid; 4597eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 4607eca39e0SShashi Mallela bool result = false; 4617eca39e0SShashi Mallela uint64_t value; 4627eca39e0SShashi Mallela 4637eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 4647eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 4657eca39e0SShashi Mallela 4667eca39e0SShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 4677eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 4687eca39e0SShashi Mallela 4697eca39e0SShashi Mallela if (res != MEMTX_OK) { 4707eca39e0SShashi Mallela return result; 4717eca39e0SShashi Mallela } 4727eca39e0SShashi Mallela 4737eca39e0SShashi Mallela icid = value & ICID_MASK; 4747eca39e0SShashi Mallela 4757eca39e0SShashi Mallela rdbase = (value & R_MAPC_RDBASE_MASK) >> R_MAPC_RDBASE_SHIFT; 4767eca39e0SShashi Mallela rdbase &= RDBASE_PROCNUM_MASK; 4777eca39e0SShashi Mallela 4787eca39e0SShashi Mallela valid = (value & CMD_FIELD_VALID_MASK); 4797eca39e0SShashi Mallela 4807eca39e0SShashi Mallela if ((icid > s->ct.maxids.max_collids) || (rdbase > s->gicv3->num_cpu)) { 4817eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 4827eca39e0SShashi Mallela "ITS MAPC: invalid collection table attributes " 4837eca39e0SShashi Mallela "icid %d rdbase %" PRIu64 "\n", icid, rdbase); 4847eca39e0SShashi Mallela /* 4857eca39e0SShashi Mallela * in this implementation, in case of error 4867eca39e0SShashi Mallela * we ignore this command and move onto the next 4877eca39e0SShashi Mallela * command in the queue 4887eca39e0SShashi Mallela */ 4897eca39e0SShashi Mallela } else { 4907eca39e0SShashi Mallela result = update_cte(s, icid, valid, rdbase); 4917eca39e0SShashi Mallela } 4927eca39e0SShashi Mallela 4937eca39e0SShashi Mallela return result; 4947eca39e0SShashi Mallela } 4957eca39e0SShashi Mallela 4967eca39e0SShashi Mallela static bool update_dte(GICv3ITSState *s, uint32_t devid, bool valid, 4977eca39e0SShashi Mallela uint8_t size, uint64_t itt_addr) 4987eca39e0SShashi Mallela { 4997eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 5007eca39e0SShashi Mallela uint64_t value; 5017eca39e0SShashi Mallela uint64_t l2t_addr; 5027eca39e0SShashi Mallela bool valid_l2t; 5037eca39e0SShashi Mallela uint32_t l2t_id; 5047eca39e0SShashi Mallela uint32_t max_l2_entries; 5057eca39e0SShashi Mallela uint64_t dte = 0; 5067eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 5077eca39e0SShashi Mallela 5087eca39e0SShashi Mallela if (s->dt.valid) { 5097eca39e0SShashi Mallela if (valid) { 5107eca39e0SShashi Mallela /* add mapping entry to device table */ 5117eca39e0SShashi Mallela dte = (valid & TABLE_ENTRY_VALID_MASK) | 5127eca39e0SShashi Mallela ((size & SIZE_MASK) << 1U) | 5137eca39e0SShashi Mallela (itt_addr << GITS_DTE_ITTADDR_SHIFT); 5147eca39e0SShashi Mallela } 5157eca39e0SShashi Mallela } else { 5167eca39e0SShashi Mallela return true; 5177eca39e0SShashi Mallela } 5187eca39e0SShashi Mallela 5197eca39e0SShashi Mallela /* 5207eca39e0SShashi Mallela * The specification defines the format of level 1 entries of a 5217eca39e0SShashi Mallela * 2-level table, but the format of level 2 entries and the format 5227eca39e0SShashi Mallela * of flat-mapped tables is IMPDEF. 5237eca39e0SShashi Mallela */ 5247eca39e0SShashi Mallela if (s->dt.indirect) { 5257eca39e0SShashi Mallela l2t_id = devid / (s->dt.page_sz / L1TABLE_ENTRY_SIZE); 5267eca39e0SShashi Mallela 5277eca39e0SShashi Mallela value = address_space_ldq_le(as, 5287eca39e0SShashi Mallela s->dt.base_addr + 5297eca39e0SShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 5307eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 5317eca39e0SShashi Mallela 5327eca39e0SShashi Mallela if (res != MEMTX_OK) { 5337eca39e0SShashi Mallela return false; 5347eca39e0SShashi Mallela } 5357eca39e0SShashi Mallela 5367eca39e0SShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 5377eca39e0SShashi Mallela 5387eca39e0SShashi Mallela if (valid_l2t) { 5397eca39e0SShashi Mallela max_l2_entries = s->dt.page_sz / s->dt.entry_sz; 5407eca39e0SShashi Mallela 5417eca39e0SShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 5427eca39e0SShashi Mallela 5437eca39e0SShashi Mallela address_space_stq_le(as, l2t_addr + 5447eca39e0SShashi Mallela ((devid % max_l2_entries) * GITS_DTE_SIZE), 5457eca39e0SShashi Mallela dte, MEMTXATTRS_UNSPECIFIED, &res); 5467eca39e0SShashi Mallela } 5477eca39e0SShashi Mallela } else { 5487eca39e0SShashi Mallela /* Flat level table */ 5497eca39e0SShashi Mallela address_space_stq_le(as, s->dt.base_addr + (devid * GITS_DTE_SIZE), 5507eca39e0SShashi Mallela dte, MEMTXATTRS_UNSPECIFIED, &res); 5517eca39e0SShashi Mallela } 5527eca39e0SShashi Mallela if (res != MEMTX_OK) { 5537eca39e0SShashi Mallela return false; 5547eca39e0SShashi Mallela } else { 5557eca39e0SShashi Mallela return true; 5567eca39e0SShashi Mallela } 5577eca39e0SShashi Mallela } 5587eca39e0SShashi Mallela 5597eca39e0SShashi Mallela static bool process_mapd(GICv3ITSState *s, uint64_t value, uint32_t offset) 5607eca39e0SShashi Mallela { 5617eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 5627eca39e0SShashi Mallela uint32_t devid; 5637eca39e0SShashi Mallela uint8_t size; 5647eca39e0SShashi Mallela uint64_t itt_addr; 5657eca39e0SShashi Mallela bool valid; 5667eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 5677eca39e0SShashi Mallela bool result = false; 5687eca39e0SShashi Mallela 5697eca39e0SShashi Mallela devid = ((value & DEVID_MASK) >> DEVID_SHIFT); 5707eca39e0SShashi Mallela 5717eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 5727eca39e0SShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 5737eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 5747eca39e0SShashi Mallela 5757eca39e0SShashi Mallela if (res != MEMTX_OK) { 5767eca39e0SShashi Mallela return result; 5777eca39e0SShashi Mallela } 5787eca39e0SShashi Mallela 5797eca39e0SShashi Mallela size = (value & SIZE_MASK); 5807eca39e0SShashi Mallela 5817eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 5827eca39e0SShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 5837eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 5847eca39e0SShashi Mallela 5857eca39e0SShashi Mallela if (res != MEMTX_OK) { 5867eca39e0SShashi Mallela return result; 5877eca39e0SShashi Mallela } 5887eca39e0SShashi Mallela 5897eca39e0SShashi Mallela itt_addr = (value & ITTADDR_MASK) >> ITTADDR_SHIFT; 5907eca39e0SShashi Mallela 5917eca39e0SShashi Mallela valid = (value & CMD_FIELD_VALID_MASK); 5927eca39e0SShashi Mallela 5937eca39e0SShashi Mallela if ((devid > s->dt.maxids.max_devids) || 5947eca39e0SShashi Mallela (size > FIELD_EX64(s->typer, GITS_TYPER, IDBITS))) { 5957eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 5967eca39e0SShashi Mallela "ITS MAPD: invalid device table attributes " 5977eca39e0SShashi Mallela "devid %d or size %d\n", devid, size); 5987eca39e0SShashi Mallela /* 5997eca39e0SShashi Mallela * in this implementation, in case of error 6007eca39e0SShashi Mallela * we ignore this command and move onto the next 6017eca39e0SShashi Mallela * command in the queue 6027eca39e0SShashi Mallela */ 6037eca39e0SShashi Mallela } else { 6047eca39e0SShashi Mallela result = update_dte(s, devid, valid, size, itt_addr); 6057eca39e0SShashi Mallela } 6067eca39e0SShashi Mallela 6077eca39e0SShashi Mallela return result; 6087eca39e0SShashi Mallela } 6097eca39e0SShashi Mallela 6107eca39e0SShashi Mallela /* 6117eca39e0SShashi Mallela * Current implementation blocks until all 6127eca39e0SShashi Mallela * commands are processed 6137eca39e0SShashi Mallela */ 6147eca39e0SShashi Mallela static void process_cmdq(GICv3ITSState *s) 6157eca39e0SShashi Mallela { 6167eca39e0SShashi Mallela uint32_t wr_offset = 0; 6177eca39e0SShashi Mallela uint32_t rd_offset = 0; 6187eca39e0SShashi Mallela uint32_t cq_offset = 0; 6197eca39e0SShashi Mallela uint64_t data; 6207eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 6217eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 6227eca39e0SShashi Mallela bool result = true; 6237eca39e0SShashi Mallela uint8_t cmd; 6247eca39e0SShashi Mallela 6257eca39e0SShashi Mallela if (!(s->ctlr & ITS_CTLR_ENABLED)) { 6267eca39e0SShashi Mallela return; 6277eca39e0SShashi Mallela } 6287eca39e0SShashi Mallela 6297eca39e0SShashi Mallela wr_offset = FIELD_EX64(s->cwriter, GITS_CWRITER, OFFSET); 6307eca39e0SShashi Mallela 6317eca39e0SShashi Mallela if (wr_offset > s->cq.max_entries) { 6327eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 6337eca39e0SShashi Mallela "%s: invalid write offset " 6347eca39e0SShashi Mallela "%d\n", __func__, wr_offset); 6357eca39e0SShashi Mallela return; 6367eca39e0SShashi Mallela } 6377eca39e0SShashi Mallela 6387eca39e0SShashi Mallela rd_offset = FIELD_EX64(s->creadr, GITS_CREADR, OFFSET); 6397eca39e0SShashi Mallela 6407eca39e0SShashi Mallela if (rd_offset > s->cq.max_entries) { 6417eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 6427eca39e0SShashi Mallela "%s: invalid read offset " 6437eca39e0SShashi Mallela "%d\n", __func__, rd_offset); 6447eca39e0SShashi Mallela return; 6457eca39e0SShashi Mallela } 6467eca39e0SShashi Mallela 6477eca39e0SShashi Mallela while (wr_offset != rd_offset) { 6487eca39e0SShashi Mallela cq_offset = (rd_offset * GITS_CMDQ_ENTRY_SIZE); 6497eca39e0SShashi Mallela data = address_space_ldq_le(as, s->cq.base_addr + cq_offset, 6507eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 6517eca39e0SShashi Mallela if (res != MEMTX_OK) { 6527eca39e0SShashi Mallela result = false; 6537eca39e0SShashi Mallela } 6547eca39e0SShashi Mallela cmd = (data & CMD_MASK); 6557eca39e0SShashi Mallela 6567eca39e0SShashi Mallela switch (cmd) { 6577eca39e0SShashi Mallela case GITS_CMD_INT: 658*c694cb4cSShashi Mallela res = process_its_cmd(s, data, cq_offset, INTERRUPT); 6597eca39e0SShashi Mallela break; 6607eca39e0SShashi Mallela case GITS_CMD_CLEAR: 661*c694cb4cSShashi Mallela res = process_its_cmd(s, data, cq_offset, CLEAR); 6627eca39e0SShashi Mallela break; 6637eca39e0SShashi Mallela case GITS_CMD_SYNC: 6647eca39e0SShashi Mallela /* 6657eca39e0SShashi Mallela * Current implementation makes a blocking synchronous call 6667eca39e0SShashi Mallela * for every command issued earlier, hence the internal state 6677eca39e0SShashi Mallela * is already consistent by the time SYNC command is executed. 6687eca39e0SShashi Mallela * Hence no further processing is required for SYNC command. 6697eca39e0SShashi Mallela */ 6707eca39e0SShashi Mallela break; 6717eca39e0SShashi Mallela case GITS_CMD_MAPD: 6727eca39e0SShashi Mallela result = process_mapd(s, data, cq_offset); 6737eca39e0SShashi Mallela break; 6747eca39e0SShashi Mallela case GITS_CMD_MAPC: 6757eca39e0SShashi Mallela result = process_mapc(s, cq_offset); 6767eca39e0SShashi Mallela break; 6777eca39e0SShashi Mallela case GITS_CMD_MAPTI: 678*c694cb4cSShashi Mallela result = process_mapti(s, data, cq_offset, false); 6797eca39e0SShashi Mallela break; 6807eca39e0SShashi Mallela case GITS_CMD_MAPI: 681*c694cb4cSShashi Mallela result = process_mapti(s, data, cq_offset, true); 6827eca39e0SShashi Mallela break; 6837eca39e0SShashi Mallela case GITS_CMD_DISCARD: 684*c694cb4cSShashi Mallela result = process_its_cmd(s, data, cq_offset, DISCARD); 6857eca39e0SShashi Mallela break; 6867eca39e0SShashi Mallela case GITS_CMD_INV: 6877eca39e0SShashi Mallela case GITS_CMD_INVALL: 6887eca39e0SShashi Mallela break; 6897eca39e0SShashi Mallela default: 6907eca39e0SShashi Mallela break; 6917eca39e0SShashi Mallela } 6927eca39e0SShashi Mallela if (result) { 6937eca39e0SShashi Mallela rd_offset++; 6947eca39e0SShashi Mallela rd_offset %= s->cq.max_entries; 6957eca39e0SShashi Mallela s->creadr = FIELD_DP64(s->creadr, GITS_CREADR, OFFSET, rd_offset); 6967eca39e0SShashi Mallela } else { 6977eca39e0SShashi Mallela /* 6987eca39e0SShashi Mallela * in this implementation, in case of dma read/write error 6997eca39e0SShashi Mallela * we stall the command processing 7007eca39e0SShashi Mallela */ 7017eca39e0SShashi Mallela s->creadr = FIELD_DP64(s->creadr, GITS_CREADR, STALLED, 1); 7027eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 7037eca39e0SShashi Mallela "%s: %x cmd processing failed\n", __func__, cmd); 7047eca39e0SShashi Mallela break; 7057eca39e0SShashi Mallela } 7067eca39e0SShashi Mallela } 7077eca39e0SShashi Mallela } 7087eca39e0SShashi Mallela 7091b08e436SShashi Mallela /* 7101b08e436SShashi Mallela * This function extracts the ITS Device and Collection table specific 7111b08e436SShashi Mallela * parameters (like base_addr, size etc) from GITS_BASER register. 7121b08e436SShashi Mallela * It is called during ITS enable and also during post_load migration 7131b08e436SShashi Mallela */ 7141b08e436SShashi Mallela static void extract_table_params(GICv3ITSState *s) 7151b08e436SShashi Mallela { 7161b08e436SShashi Mallela uint16_t num_pages = 0; 7171b08e436SShashi Mallela uint8_t page_sz_type; 7181b08e436SShashi Mallela uint8_t type; 7191b08e436SShashi Mallela uint32_t page_sz = 0; 7201b08e436SShashi Mallela uint64_t value; 7211b08e436SShashi Mallela 7221b08e436SShashi Mallela for (int i = 0; i < 8; i++) { 7231b08e436SShashi Mallela value = s->baser[i]; 7241b08e436SShashi Mallela 7251b08e436SShashi Mallela if (!value) { 7261b08e436SShashi Mallela continue; 7271b08e436SShashi Mallela } 7281b08e436SShashi Mallela 7291b08e436SShashi Mallela page_sz_type = FIELD_EX64(value, GITS_BASER, PAGESIZE); 7301b08e436SShashi Mallela 7311b08e436SShashi Mallela switch (page_sz_type) { 7321b08e436SShashi Mallela case 0: 7331b08e436SShashi Mallela page_sz = GITS_PAGE_SIZE_4K; 7341b08e436SShashi Mallela break; 7351b08e436SShashi Mallela 7361b08e436SShashi Mallela case 1: 7371b08e436SShashi Mallela page_sz = GITS_PAGE_SIZE_16K; 7381b08e436SShashi Mallela break; 7391b08e436SShashi Mallela 7401b08e436SShashi Mallela case 2: 7411b08e436SShashi Mallela case 3: 7421b08e436SShashi Mallela page_sz = GITS_PAGE_SIZE_64K; 7431b08e436SShashi Mallela break; 7441b08e436SShashi Mallela 7451b08e436SShashi Mallela default: 7461b08e436SShashi Mallela g_assert_not_reached(); 7471b08e436SShashi Mallela } 7481b08e436SShashi Mallela 7491b08e436SShashi Mallela num_pages = FIELD_EX64(value, GITS_BASER, SIZE) + 1; 7501b08e436SShashi Mallela 7511b08e436SShashi Mallela type = FIELD_EX64(value, GITS_BASER, TYPE); 7521b08e436SShashi Mallela 7531b08e436SShashi Mallela switch (type) { 7541b08e436SShashi Mallela 7551b08e436SShashi Mallela case GITS_BASER_TYPE_DEVICE: 7561b08e436SShashi Mallela memset(&s->dt, 0 , sizeof(s->dt)); 7571b08e436SShashi Mallela s->dt.valid = FIELD_EX64(value, GITS_BASER, VALID); 7581b08e436SShashi Mallela 7591b08e436SShashi Mallela if (!s->dt.valid) { 7601b08e436SShashi Mallela return; 7611b08e436SShashi Mallela } 7621b08e436SShashi Mallela 7631b08e436SShashi Mallela s->dt.page_sz = page_sz; 7641b08e436SShashi Mallela s->dt.indirect = FIELD_EX64(value, GITS_BASER, INDIRECT); 7651b08e436SShashi Mallela s->dt.entry_sz = FIELD_EX64(value, GITS_BASER, ENTRYSIZE); 7661b08e436SShashi Mallela 7671b08e436SShashi Mallela if (!s->dt.indirect) { 7681b08e436SShashi Mallela s->dt.max_entries = (num_pages * page_sz) / s->dt.entry_sz; 7691b08e436SShashi Mallela } else { 7701b08e436SShashi Mallela s->dt.max_entries = (((num_pages * page_sz) / 7711b08e436SShashi Mallela L1TABLE_ENTRY_SIZE) * 7721b08e436SShashi Mallela (page_sz / s->dt.entry_sz)); 7731b08e436SShashi Mallela } 7741b08e436SShashi Mallela 7751b08e436SShashi Mallela s->dt.maxids.max_devids = (1UL << (FIELD_EX64(s->typer, GITS_TYPER, 7761b08e436SShashi Mallela DEVBITS) + 1)); 7771b08e436SShashi Mallela 7781b08e436SShashi Mallela s->dt.base_addr = baser_base_addr(value, page_sz); 7791b08e436SShashi Mallela 7801b08e436SShashi Mallela break; 7811b08e436SShashi Mallela 7821b08e436SShashi Mallela case GITS_BASER_TYPE_COLLECTION: 7831b08e436SShashi Mallela memset(&s->ct, 0 , sizeof(s->ct)); 7841b08e436SShashi Mallela s->ct.valid = FIELD_EX64(value, GITS_BASER, VALID); 7851b08e436SShashi Mallela 7861b08e436SShashi Mallela /* 7871b08e436SShashi Mallela * GITS_TYPER.HCC is 0 for this implementation 7881b08e436SShashi Mallela * hence writes are discarded if ct.valid is 0 7891b08e436SShashi Mallela */ 7901b08e436SShashi Mallela if (!s->ct.valid) { 7911b08e436SShashi Mallela return; 7921b08e436SShashi Mallela } 7931b08e436SShashi Mallela 7941b08e436SShashi Mallela s->ct.page_sz = page_sz; 7951b08e436SShashi Mallela s->ct.indirect = FIELD_EX64(value, GITS_BASER, INDIRECT); 7961b08e436SShashi Mallela s->ct.entry_sz = FIELD_EX64(value, GITS_BASER, ENTRYSIZE); 7971b08e436SShashi Mallela 7981b08e436SShashi Mallela if (!s->ct.indirect) { 7991b08e436SShashi Mallela s->ct.max_entries = (num_pages * page_sz) / s->ct.entry_sz; 8001b08e436SShashi Mallela } else { 8011b08e436SShashi Mallela s->ct.max_entries = (((num_pages * page_sz) / 8021b08e436SShashi Mallela L1TABLE_ENTRY_SIZE) * 8031b08e436SShashi Mallela (page_sz / s->ct.entry_sz)); 8041b08e436SShashi Mallela } 8051b08e436SShashi Mallela 8061b08e436SShashi Mallela if (FIELD_EX64(s->typer, GITS_TYPER, CIL)) { 8071b08e436SShashi Mallela s->ct.maxids.max_collids = (1UL << (FIELD_EX64(s->typer, 8081b08e436SShashi Mallela GITS_TYPER, CIDBITS) + 1)); 8091b08e436SShashi Mallela } else { 8101b08e436SShashi Mallela /* 16-bit CollectionId supported when CIL == 0 */ 8111b08e436SShashi Mallela s->ct.maxids.max_collids = (1UL << 16); 8121b08e436SShashi Mallela } 8131b08e436SShashi Mallela 8141b08e436SShashi Mallela s->ct.base_addr = baser_base_addr(value, page_sz); 8151b08e436SShashi Mallela 8161b08e436SShashi Mallela break; 8171b08e436SShashi Mallela 8181b08e436SShashi Mallela default: 8191b08e436SShashi Mallela break; 8201b08e436SShashi Mallela } 8211b08e436SShashi Mallela } 8221b08e436SShashi Mallela } 8231b08e436SShashi Mallela 8241b08e436SShashi Mallela static void extract_cmdq_params(GICv3ITSState *s) 8251b08e436SShashi Mallela { 8261b08e436SShashi Mallela uint16_t num_pages = 0; 8271b08e436SShashi Mallela uint64_t value = s->cbaser; 8281b08e436SShashi Mallela 8291b08e436SShashi Mallela num_pages = FIELD_EX64(value, GITS_CBASER, SIZE) + 1; 8301b08e436SShashi Mallela 8311b08e436SShashi Mallela memset(&s->cq, 0 , sizeof(s->cq)); 8321b08e436SShashi Mallela s->cq.valid = FIELD_EX64(value, GITS_CBASER, VALID); 8331b08e436SShashi Mallela 8341b08e436SShashi Mallela if (s->cq.valid) { 8351b08e436SShashi Mallela s->cq.max_entries = (num_pages * GITS_PAGE_SIZE_4K) / 8361b08e436SShashi Mallela GITS_CMDQ_ENTRY_SIZE; 8371b08e436SShashi Mallela s->cq.base_addr = FIELD_EX64(value, GITS_CBASER, PHYADDR); 8381b08e436SShashi Mallela s->cq.base_addr <<= R_GITS_CBASER_PHYADDR_SHIFT; 8391b08e436SShashi Mallela } 8401b08e436SShashi Mallela } 8411b08e436SShashi Mallela 84218f6290aSShashi Mallela static MemTxResult gicv3_its_translation_write(void *opaque, hwaddr offset, 84318f6290aSShashi Mallela uint64_t data, unsigned size, 84418f6290aSShashi Mallela MemTxAttrs attrs) 84518f6290aSShashi Mallela { 846*c694cb4cSShashi Mallela GICv3ITSState *s = (GICv3ITSState *)opaque; 847*c694cb4cSShashi Mallela bool result = true; 848*c694cb4cSShashi Mallela uint32_t devid = 0; 849*c694cb4cSShashi Mallela 850*c694cb4cSShashi Mallela switch (offset) { 851*c694cb4cSShashi Mallela case GITS_TRANSLATER: 852*c694cb4cSShashi Mallela if (s->ctlr & ITS_CTLR_ENABLED) { 853*c694cb4cSShashi Mallela devid = attrs.requester_id; 854*c694cb4cSShashi Mallela result = process_its_cmd(s, data, devid, NONE); 855*c694cb4cSShashi Mallela } 856*c694cb4cSShashi Mallela break; 857*c694cb4cSShashi Mallela default: 858*c694cb4cSShashi Mallela break; 859*c694cb4cSShashi Mallela } 860*c694cb4cSShashi Mallela 861*c694cb4cSShashi Mallela if (result) { 86218f6290aSShashi Mallela return MEMTX_OK; 863*c694cb4cSShashi Mallela } else { 864*c694cb4cSShashi Mallela return MEMTX_ERROR; 865*c694cb4cSShashi Mallela } 86618f6290aSShashi Mallela } 86718f6290aSShashi Mallela 86818f6290aSShashi Mallela static bool its_writel(GICv3ITSState *s, hwaddr offset, 86918f6290aSShashi Mallela uint64_t value, MemTxAttrs attrs) 87018f6290aSShashi Mallela { 87118f6290aSShashi Mallela bool result = true; 8721b08e436SShashi Mallela int index; 87318f6290aSShashi Mallela 8741b08e436SShashi Mallela switch (offset) { 8751b08e436SShashi Mallela case GITS_CTLR: 8761b08e436SShashi Mallela s->ctlr |= (value & ~(s->ctlr)); 8771b08e436SShashi Mallela 8781b08e436SShashi Mallela if (s->ctlr & ITS_CTLR_ENABLED) { 8791b08e436SShashi Mallela extract_table_params(s); 8801b08e436SShashi Mallela extract_cmdq_params(s); 8811b08e436SShashi Mallela s->creadr = 0; 8827eca39e0SShashi Mallela process_cmdq(s); 8831b08e436SShashi Mallela } 8841b08e436SShashi Mallela break; 8851b08e436SShashi Mallela case GITS_CBASER: 8861b08e436SShashi Mallela /* 8871b08e436SShashi Mallela * IMPDEF choice:- GITS_CBASER register becomes RO if ITS is 8881b08e436SShashi Mallela * already enabled 8891b08e436SShashi Mallela */ 8901b08e436SShashi Mallela if (!(s->ctlr & ITS_CTLR_ENABLED)) { 8911b08e436SShashi Mallela s->cbaser = deposit64(s->cbaser, 0, 32, value); 8921b08e436SShashi Mallela s->creadr = 0; 8931b08e436SShashi Mallela s->cwriter = s->creadr; 8941b08e436SShashi Mallela } 8951b08e436SShashi Mallela break; 8961b08e436SShashi Mallela case GITS_CBASER + 4: 8971b08e436SShashi Mallela /* 8981b08e436SShashi Mallela * IMPDEF choice:- GITS_CBASER register becomes RO if ITS is 8991b08e436SShashi Mallela * already enabled 9001b08e436SShashi Mallela */ 9011b08e436SShashi Mallela if (!(s->ctlr & ITS_CTLR_ENABLED)) { 9021b08e436SShashi Mallela s->cbaser = deposit64(s->cbaser, 32, 32, value); 9031b08e436SShashi Mallela s->creadr = 0; 9041b08e436SShashi Mallela s->cwriter = s->creadr; 9051b08e436SShashi Mallela } 9061b08e436SShashi Mallela break; 9071b08e436SShashi Mallela case GITS_CWRITER: 9081b08e436SShashi Mallela s->cwriter = deposit64(s->cwriter, 0, 32, 9091b08e436SShashi Mallela (value & ~R_GITS_CWRITER_RETRY_MASK)); 9107eca39e0SShashi Mallela if (s->cwriter != s->creadr) { 9117eca39e0SShashi Mallela process_cmdq(s); 9127eca39e0SShashi Mallela } 9131b08e436SShashi Mallela break; 9141b08e436SShashi Mallela case GITS_CWRITER + 4: 9151b08e436SShashi Mallela s->cwriter = deposit64(s->cwriter, 32, 32, value); 9161b08e436SShashi Mallela break; 9171b08e436SShashi Mallela case GITS_CREADR: 9181b08e436SShashi Mallela if (s->gicv3->gicd_ctlr & GICD_CTLR_DS) { 9191b08e436SShashi Mallela s->creadr = deposit64(s->creadr, 0, 32, 9201b08e436SShashi Mallela (value & ~R_GITS_CREADR_STALLED_MASK)); 9211b08e436SShashi Mallela } else { 9221b08e436SShashi Mallela /* RO register, ignore the write */ 9231b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 9241b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 9251b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 9261b08e436SShashi Mallela } 9271b08e436SShashi Mallela break; 9281b08e436SShashi Mallela case GITS_CREADR + 4: 9291b08e436SShashi Mallela if (s->gicv3->gicd_ctlr & GICD_CTLR_DS) { 9301b08e436SShashi Mallela s->creadr = deposit64(s->creadr, 32, 32, value); 9311b08e436SShashi Mallela } else { 9321b08e436SShashi Mallela /* RO register, ignore the write */ 9331b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 9341b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 9351b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 9361b08e436SShashi Mallela } 9371b08e436SShashi Mallela break; 9381b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 9391b08e436SShashi Mallela /* 9401b08e436SShashi Mallela * IMPDEF choice:- GITS_BASERn register becomes RO if ITS is 9411b08e436SShashi Mallela * already enabled 9421b08e436SShashi Mallela */ 9431b08e436SShashi Mallela if (!(s->ctlr & ITS_CTLR_ENABLED)) { 9441b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 9451b08e436SShashi Mallela 9461b08e436SShashi Mallela if (offset & 7) { 9471b08e436SShashi Mallela value <<= 32; 9481b08e436SShashi Mallela value &= ~GITS_BASER_RO_MASK; 9491b08e436SShashi Mallela s->baser[index] &= GITS_BASER_RO_MASK | MAKE_64BIT_MASK(0, 32); 9501b08e436SShashi Mallela s->baser[index] |= value; 9511b08e436SShashi Mallela } else { 9521b08e436SShashi Mallela value &= ~GITS_BASER_RO_MASK; 9531b08e436SShashi Mallela s->baser[index] &= GITS_BASER_RO_MASK | MAKE_64BIT_MASK(32, 32); 9541b08e436SShashi Mallela s->baser[index] |= value; 9551b08e436SShashi Mallela } 9561b08e436SShashi Mallela } 9571b08e436SShashi Mallela break; 9581b08e436SShashi Mallela case GITS_IIDR: 9591b08e436SShashi Mallela case GITS_IDREGS ... GITS_IDREGS + 0x2f: 9601b08e436SShashi Mallela /* RO registers, ignore the write */ 9611b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 9621b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 9631b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 9641b08e436SShashi Mallela break; 9651b08e436SShashi Mallela default: 9661b08e436SShashi Mallela result = false; 9671b08e436SShashi Mallela break; 9681b08e436SShashi Mallela } 96918f6290aSShashi Mallela return result; 97018f6290aSShashi Mallela } 97118f6290aSShashi Mallela 97218f6290aSShashi Mallela static bool its_readl(GICv3ITSState *s, hwaddr offset, 97318f6290aSShashi Mallela uint64_t *data, MemTxAttrs attrs) 97418f6290aSShashi Mallela { 97518f6290aSShashi Mallela bool result = true; 9761b08e436SShashi Mallela int index; 97718f6290aSShashi Mallela 9781b08e436SShashi Mallela switch (offset) { 9791b08e436SShashi Mallela case GITS_CTLR: 9801b08e436SShashi Mallela *data = s->ctlr; 9811b08e436SShashi Mallela break; 9821b08e436SShashi Mallela case GITS_IIDR: 9831b08e436SShashi Mallela *data = gicv3_iidr(); 9841b08e436SShashi Mallela break; 9851b08e436SShashi Mallela case GITS_IDREGS ... GITS_IDREGS + 0x2f: 9861b08e436SShashi Mallela /* ID registers */ 9871b08e436SShashi Mallela *data = gicv3_idreg(offset - GITS_IDREGS); 9881b08e436SShashi Mallela break; 9891b08e436SShashi Mallela case GITS_TYPER: 9901b08e436SShashi Mallela *data = extract64(s->typer, 0, 32); 9911b08e436SShashi Mallela break; 9921b08e436SShashi Mallela case GITS_TYPER + 4: 9931b08e436SShashi Mallela *data = extract64(s->typer, 32, 32); 9941b08e436SShashi Mallela break; 9951b08e436SShashi Mallela case GITS_CBASER: 9961b08e436SShashi Mallela *data = extract64(s->cbaser, 0, 32); 9971b08e436SShashi Mallela break; 9981b08e436SShashi Mallela case GITS_CBASER + 4: 9991b08e436SShashi Mallela *data = extract64(s->cbaser, 32, 32); 10001b08e436SShashi Mallela break; 10011b08e436SShashi Mallela case GITS_CREADR: 10021b08e436SShashi Mallela *data = extract64(s->creadr, 0, 32); 10031b08e436SShashi Mallela break; 10041b08e436SShashi Mallela case GITS_CREADR + 4: 10051b08e436SShashi Mallela *data = extract64(s->creadr, 32, 32); 10061b08e436SShashi Mallela break; 10071b08e436SShashi Mallela case GITS_CWRITER: 10081b08e436SShashi Mallela *data = extract64(s->cwriter, 0, 32); 10091b08e436SShashi Mallela break; 10101b08e436SShashi Mallela case GITS_CWRITER + 4: 10111b08e436SShashi Mallela *data = extract64(s->cwriter, 32, 32); 10121b08e436SShashi Mallela break; 10131b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 10141b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 10151b08e436SShashi Mallela if (offset & 7) { 10161b08e436SShashi Mallela *data = extract64(s->baser[index], 32, 32); 10171b08e436SShashi Mallela } else { 10181b08e436SShashi Mallela *data = extract64(s->baser[index], 0, 32); 10191b08e436SShashi Mallela } 10201b08e436SShashi Mallela break; 10211b08e436SShashi Mallela default: 10221b08e436SShashi Mallela result = false; 10231b08e436SShashi Mallela break; 10241b08e436SShashi Mallela } 102518f6290aSShashi Mallela return result; 102618f6290aSShashi Mallela } 102718f6290aSShashi Mallela 102818f6290aSShashi Mallela static bool its_writell(GICv3ITSState *s, hwaddr offset, 102918f6290aSShashi Mallela uint64_t value, MemTxAttrs attrs) 103018f6290aSShashi Mallela { 103118f6290aSShashi Mallela bool result = true; 10321b08e436SShashi Mallela int index; 103318f6290aSShashi Mallela 10341b08e436SShashi Mallela switch (offset) { 10351b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 10361b08e436SShashi Mallela /* 10371b08e436SShashi Mallela * IMPDEF choice:- GITS_BASERn register becomes RO if ITS is 10381b08e436SShashi Mallela * already enabled 10391b08e436SShashi Mallela */ 10401b08e436SShashi Mallela if (!(s->ctlr & ITS_CTLR_ENABLED)) { 10411b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 10421b08e436SShashi Mallela s->baser[index] &= GITS_BASER_RO_MASK; 10431b08e436SShashi Mallela s->baser[index] |= (value & ~GITS_BASER_RO_MASK); 10441b08e436SShashi Mallela } 10451b08e436SShashi Mallela break; 10461b08e436SShashi Mallela case GITS_CBASER: 10471b08e436SShashi Mallela /* 10481b08e436SShashi Mallela * IMPDEF choice:- GITS_CBASER register becomes RO if ITS is 10491b08e436SShashi Mallela * already enabled 10501b08e436SShashi Mallela */ 10511b08e436SShashi Mallela if (!(s->ctlr & ITS_CTLR_ENABLED)) { 10521b08e436SShashi Mallela s->cbaser = value; 10531b08e436SShashi Mallela s->creadr = 0; 10541b08e436SShashi Mallela s->cwriter = s->creadr; 10551b08e436SShashi Mallela } 10561b08e436SShashi Mallela break; 10571b08e436SShashi Mallela case GITS_CWRITER: 10581b08e436SShashi Mallela s->cwriter = value & ~R_GITS_CWRITER_RETRY_MASK; 10597eca39e0SShashi Mallela if (s->cwriter != s->creadr) { 10607eca39e0SShashi Mallela process_cmdq(s); 10617eca39e0SShashi Mallela } 10621b08e436SShashi Mallela break; 10631b08e436SShashi Mallela case GITS_CREADR: 10641b08e436SShashi Mallela if (s->gicv3->gicd_ctlr & GICD_CTLR_DS) { 10651b08e436SShashi Mallela s->creadr = value & ~R_GITS_CREADR_STALLED_MASK; 10661b08e436SShashi Mallela } else { 10671b08e436SShashi Mallela /* RO register, ignore the write */ 10681b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 10691b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 10701b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 10711b08e436SShashi Mallela } 10721b08e436SShashi Mallela break; 10731b08e436SShashi Mallela case GITS_TYPER: 10741b08e436SShashi Mallela /* RO registers, ignore the write */ 10751b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 10761b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 10771b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 10781b08e436SShashi Mallela break; 10791b08e436SShashi Mallela default: 10801b08e436SShashi Mallela result = false; 10811b08e436SShashi Mallela break; 10821b08e436SShashi Mallela } 108318f6290aSShashi Mallela return result; 108418f6290aSShashi Mallela } 108518f6290aSShashi Mallela 108618f6290aSShashi Mallela static bool its_readll(GICv3ITSState *s, hwaddr offset, 108718f6290aSShashi Mallela uint64_t *data, MemTxAttrs attrs) 108818f6290aSShashi Mallela { 108918f6290aSShashi Mallela bool result = true; 10901b08e436SShashi Mallela int index; 109118f6290aSShashi Mallela 10921b08e436SShashi Mallela switch (offset) { 10931b08e436SShashi Mallela case GITS_TYPER: 10941b08e436SShashi Mallela *data = s->typer; 10951b08e436SShashi Mallela break; 10961b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 10971b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 10981b08e436SShashi Mallela *data = s->baser[index]; 10991b08e436SShashi Mallela break; 11001b08e436SShashi Mallela case GITS_CBASER: 11011b08e436SShashi Mallela *data = s->cbaser; 11021b08e436SShashi Mallela break; 11031b08e436SShashi Mallela case GITS_CREADR: 11041b08e436SShashi Mallela *data = s->creadr; 11051b08e436SShashi Mallela break; 11061b08e436SShashi Mallela case GITS_CWRITER: 11071b08e436SShashi Mallela *data = s->cwriter; 11081b08e436SShashi Mallela break; 11091b08e436SShashi Mallela default: 11101b08e436SShashi Mallela result = false; 11111b08e436SShashi Mallela break; 11121b08e436SShashi Mallela } 111318f6290aSShashi Mallela return result; 111418f6290aSShashi Mallela } 111518f6290aSShashi Mallela 111618f6290aSShashi Mallela static MemTxResult gicv3_its_read(void *opaque, hwaddr offset, uint64_t *data, 111718f6290aSShashi Mallela unsigned size, MemTxAttrs attrs) 111818f6290aSShashi Mallela { 111918f6290aSShashi Mallela GICv3ITSState *s = (GICv3ITSState *)opaque; 112018f6290aSShashi Mallela bool result; 112118f6290aSShashi Mallela 112218f6290aSShashi Mallela switch (size) { 112318f6290aSShashi Mallela case 4: 112418f6290aSShashi Mallela result = its_readl(s, offset, data, attrs); 112518f6290aSShashi Mallela break; 112618f6290aSShashi Mallela case 8: 112718f6290aSShashi Mallela result = its_readll(s, offset, data, attrs); 112818f6290aSShashi Mallela break; 112918f6290aSShashi Mallela default: 113018f6290aSShashi Mallela result = false; 113118f6290aSShashi Mallela break; 113218f6290aSShashi Mallela } 113318f6290aSShashi Mallela 113418f6290aSShashi Mallela if (!result) { 113518f6290aSShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 113618f6290aSShashi Mallela "%s: invalid guest read at offset " TARGET_FMT_plx 113718f6290aSShashi Mallela "size %u\n", __func__, offset, size); 113818f6290aSShashi Mallela /* 113918f6290aSShashi Mallela * The spec requires that reserved registers are RAZ/WI; 114018f6290aSShashi Mallela * so use false returns from leaf functions as a way to 114118f6290aSShashi Mallela * trigger the guest-error logging but don't return it to 114218f6290aSShashi Mallela * the caller, or we'll cause a spurious guest data abort. 114318f6290aSShashi Mallela */ 114418f6290aSShashi Mallela *data = 0; 114518f6290aSShashi Mallela } 114618f6290aSShashi Mallela return MEMTX_OK; 114718f6290aSShashi Mallela } 114818f6290aSShashi Mallela 114918f6290aSShashi Mallela static MemTxResult gicv3_its_write(void *opaque, hwaddr offset, uint64_t data, 115018f6290aSShashi Mallela unsigned size, MemTxAttrs attrs) 115118f6290aSShashi Mallela { 115218f6290aSShashi Mallela GICv3ITSState *s = (GICv3ITSState *)opaque; 115318f6290aSShashi Mallela bool result; 115418f6290aSShashi Mallela 115518f6290aSShashi Mallela switch (size) { 115618f6290aSShashi Mallela case 4: 115718f6290aSShashi Mallela result = its_writel(s, offset, data, attrs); 115818f6290aSShashi Mallela break; 115918f6290aSShashi Mallela case 8: 116018f6290aSShashi Mallela result = its_writell(s, offset, data, attrs); 116118f6290aSShashi Mallela break; 116218f6290aSShashi Mallela default: 116318f6290aSShashi Mallela result = false; 116418f6290aSShashi Mallela break; 116518f6290aSShashi Mallela } 116618f6290aSShashi Mallela 116718f6290aSShashi Mallela if (!result) { 116818f6290aSShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 116918f6290aSShashi Mallela "%s: invalid guest write at offset " TARGET_FMT_plx 117018f6290aSShashi Mallela "size %u\n", __func__, offset, size); 117118f6290aSShashi Mallela /* 117218f6290aSShashi Mallela * The spec requires that reserved registers are RAZ/WI; 117318f6290aSShashi Mallela * so use false returns from leaf functions as a way to 117418f6290aSShashi Mallela * trigger the guest-error logging but don't return it to 117518f6290aSShashi Mallela * the caller, or we'll cause a spurious guest data abort. 117618f6290aSShashi Mallela */ 117718f6290aSShashi Mallela } 117818f6290aSShashi Mallela return MEMTX_OK; 117918f6290aSShashi Mallela } 118018f6290aSShashi Mallela 118118f6290aSShashi Mallela static const MemoryRegionOps gicv3_its_control_ops = { 118218f6290aSShashi Mallela .read_with_attrs = gicv3_its_read, 118318f6290aSShashi Mallela .write_with_attrs = gicv3_its_write, 118418f6290aSShashi Mallela .valid.min_access_size = 4, 118518f6290aSShashi Mallela .valid.max_access_size = 8, 118618f6290aSShashi Mallela .impl.min_access_size = 4, 118718f6290aSShashi Mallela .impl.max_access_size = 8, 118818f6290aSShashi Mallela .endianness = DEVICE_NATIVE_ENDIAN, 118918f6290aSShashi Mallela }; 119018f6290aSShashi Mallela 119118f6290aSShashi Mallela static const MemoryRegionOps gicv3_its_translation_ops = { 119218f6290aSShashi Mallela .write_with_attrs = gicv3_its_translation_write, 119318f6290aSShashi Mallela .valid.min_access_size = 2, 119418f6290aSShashi Mallela .valid.max_access_size = 4, 119518f6290aSShashi Mallela .impl.min_access_size = 2, 119618f6290aSShashi Mallela .impl.max_access_size = 4, 119718f6290aSShashi Mallela .endianness = DEVICE_NATIVE_ENDIAN, 119818f6290aSShashi Mallela }; 119918f6290aSShashi Mallela 120018f6290aSShashi Mallela static void gicv3_arm_its_realize(DeviceState *dev, Error **errp) 120118f6290aSShashi Mallela { 120218f6290aSShashi Mallela GICv3ITSState *s = ARM_GICV3_ITS_COMMON(dev); 120318f6290aSShashi Mallela int i; 120418f6290aSShashi Mallela 120518f6290aSShashi Mallela for (i = 0; i < s->gicv3->num_cpu; i++) { 120618f6290aSShashi Mallela if (!(s->gicv3->cpu[i].gicr_typer & GICR_TYPER_PLPIS)) { 120718f6290aSShashi Mallela error_setg(errp, "Physical LPI not supported by CPU %d", i); 120818f6290aSShashi Mallela return; 120918f6290aSShashi Mallela } 121018f6290aSShashi Mallela } 121118f6290aSShashi Mallela 121218f6290aSShashi Mallela gicv3_its_init_mmio(s, &gicv3_its_control_ops, &gicv3_its_translation_ops); 121318f6290aSShashi Mallela 12141b08e436SShashi Mallela address_space_init(&s->gicv3->dma_as, s->gicv3->dma, 12151b08e436SShashi Mallela "gicv3-its-sysmem"); 12161b08e436SShashi Mallela 121718f6290aSShashi Mallela /* set the ITS default features supported */ 121818f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, PHYSICAL, 121918f6290aSShashi Mallela GITS_TYPE_PHYSICAL); 122018f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, ITT_ENTRY_SIZE, 122118f6290aSShashi Mallela ITS_ITT_ENTRY_SIZE - 1); 122218f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, IDBITS, ITS_IDBITS); 122318f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, DEVBITS, ITS_DEVBITS); 122418f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, CIL, 1); 122518f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, CIDBITS, ITS_CIDBITS); 122618f6290aSShashi Mallela } 122718f6290aSShashi Mallela 122818f6290aSShashi Mallela static void gicv3_its_reset(DeviceState *dev) 122918f6290aSShashi Mallela { 123018f6290aSShashi Mallela GICv3ITSState *s = ARM_GICV3_ITS_COMMON(dev); 123118f6290aSShashi Mallela GICv3ITSClass *c = ARM_GICV3_ITS_GET_CLASS(s); 123218f6290aSShashi Mallela 123318f6290aSShashi Mallela c->parent_reset(dev); 123418f6290aSShashi Mallela 123518f6290aSShashi Mallela /* Quiescent bit reset to 1 */ 123618f6290aSShashi Mallela s->ctlr = FIELD_DP32(s->ctlr, GITS_CTLR, QUIESCENT, 1); 123718f6290aSShashi Mallela 123818f6290aSShashi Mallela /* 123918f6290aSShashi Mallela * setting GITS_BASER0.Type = 0b001 (Device) 124018f6290aSShashi Mallela * GITS_BASER1.Type = 0b100 (Collection Table) 124118f6290aSShashi Mallela * GITS_BASER<n>.Type,where n = 3 to 7 are 0b00 (Unimplemented) 124218f6290aSShashi Mallela * GITS_BASER<0,1>.Page_Size = 64KB 124318f6290aSShashi Mallela * and default translation table entry size to 16 bytes 124418f6290aSShashi Mallela */ 124518f6290aSShashi Mallela s->baser[0] = FIELD_DP64(s->baser[0], GITS_BASER, TYPE, 124618f6290aSShashi Mallela GITS_BASER_TYPE_DEVICE); 124718f6290aSShashi Mallela s->baser[0] = FIELD_DP64(s->baser[0], GITS_BASER, PAGESIZE, 124818f6290aSShashi Mallela GITS_BASER_PAGESIZE_64K); 124918f6290aSShashi Mallela s->baser[0] = FIELD_DP64(s->baser[0], GITS_BASER, ENTRYSIZE, 125018f6290aSShashi Mallela GITS_DTE_SIZE - 1); 125118f6290aSShashi Mallela 125218f6290aSShashi Mallela s->baser[1] = FIELD_DP64(s->baser[1], GITS_BASER, TYPE, 125318f6290aSShashi Mallela GITS_BASER_TYPE_COLLECTION); 125418f6290aSShashi Mallela s->baser[1] = FIELD_DP64(s->baser[1], GITS_BASER, PAGESIZE, 125518f6290aSShashi Mallela GITS_BASER_PAGESIZE_64K); 125618f6290aSShashi Mallela s->baser[1] = FIELD_DP64(s->baser[1], GITS_BASER, ENTRYSIZE, 125718f6290aSShashi Mallela GITS_CTE_SIZE - 1); 125818f6290aSShashi Mallela } 125918f6290aSShashi Mallela 12601b08e436SShashi Mallela static void gicv3_its_post_load(GICv3ITSState *s) 12611b08e436SShashi Mallela { 12621b08e436SShashi Mallela if (s->ctlr & ITS_CTLR_ENABLED) { 12631b08e436SShashi Mallela extract_table_params(s); 12641b08e436SShashi Mallela extract_cmdq_params(s); 12651b08e436SShashi Mallela } 12661b08e436SShashi Mallela } 12671b08e436SShashi Mallela 126818f6290aSShashi Mallela static Property gicv3_its_props[] = { 126918f6290aSShashi Mallela DEFINE_PROP_LINK("parent-gicv3", GICv3ITSState, gicv3, "arm-gicv3", 127018f6290aSShashi Mallela GICv3State *), 127118f6290aSShashi Mallela DEFINE_PROP_END_OF_LIST(), 127218f6290aSShashi Mallela }; 127318f6290aSShashi Mallela 127418f6290aSShashi Mallela static void gicv3_its_class_init(ObjectClass *klass, void *data) 127518f6290aSShashi Mallela { 127618f6290aSShashi Mallela DeviceClass *dc = DEVICE_CLASS(klass); 127718f6290aSShashi Mallela GICv3ITSClass *ic = ARM_GICV3_ITS_CLASS(klass); 12781b08e436SShashi Mallela GICv3ITSCommonClass *icc = ARM_GICV3_ITS_COMMON_CLASS(klass); 127918f6290aSShashi Mallela 128018f6290aSShashi Mallela dc->realize = gicv3_arm_its_realize; 128118f6290aSShashi Mallela device_class_set_props(dc, gicv3_its_props); 128218f6290aSShashi Mallela device_class_set_parent_reset(dc, gicv3_its_reset, &ic->parent_reset); 12831b08e436SShashi Mallela icc->post_load = gicv3_its_post_load; 128418f6290aSShashi Mallela } 128518f6290aSShashi Mallela 128618f6290aSShashi Mallela static const TypeInfo gicv3_its_info = { 128718f6290aSShashi Mallela .name = TYPE_ARM_GICV3_ITS, 128818f6290aSShashi Mallela .parent = TYPE_ARM_GICV3_ITS_COMMON, 128918f6290aSShashi Mallela .instance_size = sizeof(GICv3ITSState), 129018f6290aSShashi Mallela .class_init = gicv3_its_class_init, 129118f6290aSShashi Mallela .class_size = sizeof(GICv3ITSClass), 129218f6290aSShashi Mallela }; 129318f6290aSShashi Mallela 129418f6290aSShashi Mallela static void gicv3_its_register_types(void) 129518f6290aSShashi Mallela { 129618f6290aSShashi Mallela type_register_static(&gicv3_its_info); 129718f6290aSShashi Mallela } 129818f6290aSShashi Mallela 129918f6290aSShashi Mallela type_init(gicv3_its_register_types) 1300