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 32c694cb4cSShashi Mallela /* 33c694cb4cSShashi Mallela * This is an internal enum used to distinguish between LPI triggered 34c694cb4cSShashi Mallela * via command queue and LPI triggered via gits_translater write. 35c694cb4cSShashi Mallela */ 36c694cb4cSShashi Mallela typedef enum ItsCmdType { 37c694cb4cSShashi Mallela NONE = 0, /* internal indication for GITS_TRANSLATER write */ 38c694cb4cSShashi Mallela CLEAR = 1, 39c694cb4cSShashi Mallela DISCARD = 2, 40c694cb4cSShashi Mallela INTERRUPT = 3, 41c694cb4cSShashi Mallela } ItsCmdType; 42c694cb4cSShashi Mallela 43c694cb4cSShashi Mallela typedef struct { 44c694cb4cSShashi Mallela uint32_t iteh; 45c694cb4cSShashi Mallela uint64_t itel; 46c694cb4cSShashi Mallela } IteEntry; 47c694cb4cSShashi 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 69c694cb4cSShashi Mallela static bool get_cte(GICv3ITSState *s, uint16_t icid, uint64_t *cte, 70c694cb4cSShashi Mallela MemTxResult *res) 71c694cb4cSShashi Mallela { 72c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 73c694cb4cSShashi Mallela uint64_t l2t_addr; 74c694cb4cSShashi Mallela uint64_t value; 75c694cb4cSShashi Mallela bool valid_l2t; 76c694cb4cSShashi Mallela uint32_t l2t_id; 77c694cb4cSShashi Mallela uint32_t max_l2_entries; 78c694cb4cSShashi Mallela 79c694cb4cSShashi Mallela if (s->ct.indirect) { 80c694cb4cSShashi Mallela l2t_id = icid / (s->ct.page_sz / L1TABLE_ENTRY_SIZE); 81c694cb4cSShashi Mallela 82c694cb4cSShashi Mallela value = address_space_ldq_le(as, 83c694cb4cSShashi Mallela s->ct.base_addr + 84c694cb4cSShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 85c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 86c694cb4cSShashi Mallela 87c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 88c694cb4cSShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 89c694cb4cSShashi Mallela 90c694cb4cSShashi Mallela if (valid_l2t) { 91c694cb4cSShashi Mallela max_l2_entries = s->ct.page_sz / s->ct.entry_sz; 92c694cb4cSShashi Mallela 93c694cb4cSShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 94c694cb4cSShashi Mallela 95c694cb4cSShashi Mallela *cte = address_space_ldq_le(as, l2t_addr + 96c694cb4cSShashi Mallela ((icid % max_l2_entries) * GITS_CTE_SIZE), 97c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 98c694cb4cSShashi Mallela } 99c694cb4cSShashi Mallela } 100c694cb4cSShashi Mallela } else { 101c694cb4cSShashi Mallela /* Flat level table */ 102c694cb4cSShashi Mallela *cte = address_space_ldq_le(as, s->ct.base_addr + 103c694cb4cSShashi Mallela (icid * GITS_CTE_SIZE), 104c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 105c694cb4cSShashi Mallela } 106c694cb4cSShashi Mallela 107*437dc0eaSPeter Maydell return FIELD_EX64(*cte, CTE, VALID); 108c694cb4cSShashi Mallela } 109c694cb4cSShashi Mallela 110c694cb4cSShashi Mallela static bool update_ite(GICv3ITSState *s, uint32_t eventid, uint64_t dte, 111c694cb4cSShashi Mallela IteEntry ite) 112c694cb4cSShashi Mallela { 113c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 114c694cb4cSShashi Mallela uint64_t itt_addr; 115c694cb4cSShashi Mallela MemTxResult res = MEMTX_OK; 116c694cb4cSShashi Mallela 117e07f8445SPeter Maydell itt_addr = FIELD_EX64(dte, DTE, ITTADDR); 118c694cb4cSShashi Mallela itt_addr <<= ITTADDR_SHIFT; /* 256 byte aligned */ 119c694cb4cSShashi Mallela 120c694cb4cSShashi Mallela address_space_stq_le(as, itt_addr + (eventid * (sizeof(uint64_t) + 121c694cb4cSShashi Mallela sizeof(uint32_t))), ite.itel, MEMTXATTRS_UNSPECIFIED, 122c694cb4cSShashi Mallela &res); 123c694cb4cSShashi Mallela 124c694cb4cSShashi Mallela if (res == MEMTX_OK) { 125c694cb4cSShashi Mallela address_space_stl_le(as, itt_addr + (eventid * (sizeof(uint64_t) + 126c694cb4cSShashi Mallela sizeof(uint32_t))) + sizeof(uint32_t), ite.iteh, 127c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 128c694cb4cSShashi Mallela } 129c694cb4cSShashi Mallela if (res != MEMTX_OK) { 130c694cb4cSShashi Mallela return false; 131c694cb4cSShashi Mallela } else { 132c694cb4cSShashi Mallela return true; 133c694cb4cSShashi Mallela } 134c694cb4cSShashi Mallela } 135c694cb4cSShashi Mallela 136c694cb4cSShashi Mallela static bool get_ite(GICv3ITSState *s, uint32_t eventid, uint64_t dte, 137c694cb4cSShashi Mallela uint16_t *icid, uint32_t *pIntid, MemTxResult *res) 138c694cb4cSShashi Mallela { 139c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 140c694cb4cSShashi Mallela uint64_t itt_addr; 141c694cb4cSShashi Mallela bool status = false; 142c694cb4cSShashi Mallela IteEntry ite = {}; 143c694cb4cSShashi Mallela 144e07f8445SPeter Maydell itt_addr = FIELD_EX64(dte, DTE, ITTADDR); 145c694cb4cSShashi Mallela itt_addr <<= ITTADDR_SHIFT; /* 256 byte aligned */ 146c694cb4cSShashi Mallela 147c694cb4cSShashi Mallela ite.itel = address_space_ldq_le(as, itt_addr + 148c694cb4cSShashi Mallela (eventid * (sizeof(uint64_t) + 149c694cb4cSShashi Mallela sizeof(uint32_t))), MEMTXATTRS_UNSPECIFIED, 150c694cb4cSShashi Mallela res); 151c694cb4cSShashi Mallela 152c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 153c694cb4cSShashi Mallela ite.iteh = address_space_ldl_le(as, itt_addr + 154c694cb4cSShashi Mallela (eventid * (sizeof(uint64_t) + 155c694cb4cSShashi Mallela sizeof(uint32_t))) + sizeof(uint32_t), 156c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 157c694cb4cSShashi Mallela 158c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 159764d6ba1SPeter Maydell if (FIELD_EX64(ite.itel, ITE_L, VALID)) { 160764d6ba1SPeter Maydell int inttype = FIELD_EX64(ite.itel, ITE_L, INTTYPE); 161764d6ba1SPeter Maydell if (inttype == ITE_INTTYPE_PHYSICAL) { 162764d6ba1SPeter Maydell *pIntid = FIELD_EX64(ite.itel, ITE_L, INTID); 163764d6ba1SPeter Maydell *icid = FIELD_EX32(ite.iteh, ITE_H, ICID); 164c694cb4cSShashi Mallela status = true; 165c694cb4cSShashi Mallela } 166c694cb4cSShashi Mallela } 167c694cb4cSShashi Mallela } 168c694cb4cSShashi Mallela } 169c694cb4cSShashi Mallela return status; 170c694cb4cSShashi Mallela } 171c694cb4cSShashi Mallela 172c694cb4cSShashi Mallela static uint64_t get_dte(GICv3ITSState *s, uint32_t devid, MemTxResult *res) 173c694cb4cSShashi Mallela { 174c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 175c694cb4cSShashi Mallela uint64_t l2t_addr; 176c694cb4cSShashi Mallela uint64_t value; 177c694cb4cSShashi Mallela bool valid_l2t; 178c694cb4cSShashi Mallela uint32_t l2t_id; 179c694cb4cSShashi Mallela uint32_t max_l2_entries; 180c694cb4cSShashi Mallela 181c694cb4cSShashi Mallela if (s->dt.indirect) { 182c694cb4cSShashi Mallela l2t_id = devid / (s->dt.page_sz / L1TABLE_ENTRY_SIZE); 183c694cb4cSShashi Mallela 184c694cb4cSShashi Mallela value = address_space_ldq_le(as, 185c694cb4cSShashi Mallela s->dt.base_addr + 186c694cb4cSShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 187c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 188c694cb4cSShashi Mallela 189c694cb4cSShashi Mallela if (*res == MEMTX_OK) { 190c694cb4cSShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 191c694cb4cSShashi Mallela 192c694cb4cSShashi Mallela if (valid_l2t) { 193c694cb4cSShashi Mallela max_l2_entries = s->dt.page_sz / s->dt.entry_sz; 194c694cb4cSShashi Mallela 195c694cb4cSShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 196c694cb4cSShashi Mallela 197c694cb4cSShashi Mallela value = address_space_ldq_le(as, l2t_addr + 198c694cb4cSShashi Mallela ((devid % max_l2_entries) * GITS_DTE_SIZE), 199c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 200c694cb4cSShashi Mallela } 201c694cb4cSShashi Mallela } 202c694cb4cSShashi Mallela } else { 203c694cb4cSShashi Mallela /* Flat level table */ 204c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->dt.base_addr + 205c694cb4cSShashi Mallela (devid * GITS_DTE_SIZE), 206c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, res); 207c694cb4cSShashi Mallela } 208c694cb4cSShashi Mallela 209c694cb4cSShashi Mallela return value; 210c694cb4cSShashi Mallela } 211c694cb4cSShashi Mallela 212c694cb4cSShashi Mallela /* 213c694cb4cSShashi Mallela * This function handles the processing of following commands based on 214c694cb4cSShashi Mallela * the ItsCmdType parameter passed:- 215c694cb4cSShashi Mallela * 1. triggering of lpi interrupt translation via ITS INT command 216c694cb4cSShashi Mallela * 2. triggering of lpi interrupt translation via gits_translater register 217c694cb4cSShashi Mallela * 3. handling of ITS CLEAR command 218c694cb4cSShashi Mallela * 4. handling of ITS DISCARD command 219c694cb4cSShashi Mallela */ 220c694cb4cSShashi Mallela static bool process_its_cmd(GICv3ITSState *s, uint64_t value, uint32_t offset, 221c694cb4cSShashi Mallela ItsCmdType cmd) 222c694cb4cSShashi Mallela { 223c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 224c694cb4cSShashi Mallela uint32_t devid, eventid; 225c694cb4cSShashi Mallela MemTxResult res = MEMTX_OK; 226c694cb4cSShashi Mallela bool dte_valid; 227c694cb4cSShashi Mallela uint64_t dte = 0; 228c694cb4cSShashi Mallela uint32_t max_eventid; 229c694cb4cSShashi Mallela uint16_t icid = 0; 230c694cb4cSShashi Mallela uint32_t pIntid = 0; 231c694cb4cSShashi Mallela bool ite_valid = false; 232c694cb4cSShashi Mallela uint64_t cte = 0; 233c694cb4cSShashi Mallela bool cte_valid = false; 234c694cb4cSShashi Mallela bool result = false; 23517fb5e36SShashi Mallela uint64_t rdbase; 236c694cb4cSShashi Mallela 237c694cb4cSShashi Mallela if (cmd == NONE) { 238c694cb4cSShashi Mallela devid = offset; 239c694cb4cSShashi Mallela } else { 240c694cb4cSShashi Mallela devid = ((value & DEVID_MASK) >> DEVID_SHIFT); 241c694cb4cSShashi Mallela 242c694cb4cSShashi Mallela offset += NUM_BYTES_IN_DW; 243c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 244c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 245c694cb4cSShashi Mallela } 246c694cb4cSShashi Mallela 247c694cb4cSShashi Mallela if (res != MEMTX_OK) { 248c694cb4cSShashi Mallela return result; 249c694cb4cSShashi Mallela } 250c694cb4cSShashi Mallela 251c694cb4cSShashi Mallela eventid = (value & EVENTID_MASK); 252c694cb4cSShashi Mallela 253c694cb4cSShashi Mallela dte = get_dte(s, devid, &res); 254c694cb4cSShashi Mallela 255c694cb4cSShashi Mallela if (res != MEMTX_OK) { 256c694cb4cSShashi Mallela return result; 257c694cb4cSShashi Mallela } 258e07f8445SPeter Maydell dte_valid = FIELD_EX64(dte, DTE, VALID); 259c694cb4cSShashi Mallela 260c694cb4cSShashi Mallela if (dte_valid) { 261e07f8445SPeter Maydell max_eventid = 1UL << (FIELD_EX64(dte, DTE, SIZE) + 1); 262c694cb4cSShashi Mallela 263c694cb4cSShashi Mallela ite_valid = get_ite(s, eventid, dte, &icid, &pIntid, &res); 264c694cb4cSShashi Mallela 265c694cb4cSShashi Mallela if (res != MEMTX_OK) { 266c694cb4cSShashi Mallela return result; 267c694cb4cSShashi Mallela } 268c694cb4cSShashi Mallela 269c694cb4cSShashi Mallela if (ite_valid) { 270c694cb4cSShashi Mallela cte_valid = get_cte(s, icid, &cte, &res); 271c694cb4cSShashi Mallela } 272c694cb4cSShashi Mallela 273c694cb4cSShashi Mallela if (res != MEMTX_OK) { 274c694cb4cSShashi Mallela return result; 275c694cb4cSShashi Mallela } 276229c57b1SAlex Bennée } else { 277229c57b1SAlex Bennée qemu_log_mask(LOG_GUEST_ERROR, 278229c57b1SAlex Bennée "%s: invalid command attributes: " 279229c57b1SAlex Bennée "invalid dte: %"PRIx64" for %d (MEM_TX: %d)\n", 280229c57b1SAlex Bennée __func__, dte, devid, res); 281229c57b1SAlex Bennée return result; 282c694cb4cSShashi Mallela } 283c694cb4cSShashi Mallela 284229c57b1SAlex Bennée 285c694cb4cSShashi Mallela /* 286229c57b1SAlex Bennée * In this implementation, in case of guest errors we ignore the 287229c57b1SAlex Bennée * command and move onto the next command in the queue. 288c694cb4cSShashi Mallela */ 2896c1db43dSPeter Maydell if (devid > s->dt.max_ids) { 290229c57b1SAlex Bennée qemu_log_mask(LOG_GUEST_ERROR, 291229c57b1SAlex Bennée "%s: invalid command attributes: devid %d>%d", 2926c1db43dSPeter Maydell __func__, devid, s->dt.max_ids); 293229c57b1SAlex Bennée 294229c57b1SAlex Bennée } else if (!dte_valid || !ite_valid || !cte_valid) { 295229c57b1SAlex Bennée qemu_log_mask(LOG_GUEST_ERROR, 296229c57b1SAlex Bennée "%s: invalid command attributes: " 297229c57b1SAlex Bennée "dte: %s, ite: %s, cte: %s\n", 298229c57b1SAlex Bennée __func__, 299229c57b1SAlex Bennée dte_valid ? "valid" : "invalid", 300229c57b1SAlex Bennée ite_valid ? "valid" : "invalid", 301229c57b1SAlex Bennée cte_valid ? "valid" : "invalid"); 302229c57b1SAlex Bennée } else if (eventid > max_eventid) { 303229c57b1SAlex Bennée qemu_log_mask(LOG_GUEST_ERROR, 304229c57b1SAlex Bennée "%s: invalid command attributes: eventid %d > %d\n", 305229c57b1SAlex Bennée __func__, eventid, max_eventid); 306c694cb4cSShashi Mallela } else { 307c694cb4cSShashi Mallela /* 308c694cb4cSShashi Mallela * Current implementation only supports rdbase == procnum 309c694cb4cSShashi Mallela * Hence rdbase physical address is ignored 310c694cb4cSShashi Mallela */ 311*437dc0eaSPeter Maydell rdbase = FIELD_EX64(cte, CTE, RDBASE); 31217fb5e36SShashi Mallela 313a120157bSPeter Maydell if (rdbase >= s->gicv3->num_cpu) { 31417fb5e36SShashi Mallela return result; 31517fb5e36SShashi Mallela } 31617fb5e36SShashi Mallela 31717fb5e36SShashi Mallela if ((cmd == CLEAR) || (cmd == DISCARD)) { 31817fb5e36SShashi Mallela gicv3_redist_process_lpi(&s->gicv3->cpu[rdbase], pIntid, 0); 31917fb5e36SShashi Mallela } else { 32017fb5e36SShashi Mallela gicv3_redist_process_lpi(&s->gicv3->cpu[rdbase], pIntid, 1); 32117fb5e36SShashi Mallela } 32217fb5e36SShashi Mallela 323c694cb4cSShashi Mallela if (cmd == DISCARD) { 324c694cb4cSShashi Mallela IteEntry ite = {}; 325c694cb4cSShashi Mallela /* remove mapping from interrupt translation table */ 326c694cb4cSShashi Mallela result = update_ite(s, eventid, dte, ite); 327c694cb4cSShashi Mallela } 328c694cb4cSShashi Mallela } 329c694cb4cSShashi Mallela 330c694cb4cSShashi Mallela return result; 331c694cb4cSShashi Mallela } 332c694cb4cSShashi Mallela 333c694cb4cSShashi Mallela static bool process_mapti(GICv3ITSState *s, uint64_t value, uint32_t offset, 334c694cb4cSShashi Mallela bool ignore_pInt) 335c694cb4cSShashi Mallela { 336c694cb4cSShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 337c694cb4cSShashi Mallela uint32_t devid, eventid; 338c694cb4cSShashi Mallela uint32_t pIntid = 0; 339c694cb4cSShashi Mallela uint32_t max_eventid, max_Intid; 340c694cb4cSShashi Mallela bool dte_valid; 341c694cb4cSShashi Mallela MemTxResult res = MEMTX_OK; 342c694cb4cSShashi Mallela uint16_t icid = 0; 343c694cb4cSShashi Mallela uint64_t dte = 0; 344c694cb4cSShashi Mallela bool result = false; 345c694cb4cSShashi Mallela 346c694cb4cSShashi Mallela devid = ((value & DEVID_MASK) >> DEVID_SHIFT); 347c694cb4cSShashi Mallela offset += NUM_BYTES_IN_DW; 348c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 349c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 350c694cb4cSShashi Mallela 351c694cb4cSShashi Mallela if (res != MEMTX_OK) { 352c694cb4cSShashi Mallela return result; 353c694cb4cSShashi Mallela } 354c694cb4cSShashi Mallela 355c694cb4cSShashi Mallela eventid = (value & EVENTID_MASK); 356c694cb4cSShashi Mallela 357b87fab1cSPeter Maydell if (ignore_pInt) { 358b87fab1cSPeter Maydell pIntid = eventid; 359b87fab1cSPeter Maydell } else { 360c694cb4cSShashi Mallela pIntid = ((value & pINTID_MASK) >> pINTID_SHIFT); 361c694cb4cSShashi Mallela } 362c694cb4cSShashi Mallela 363c694cb4cSShashi Mallela offset += NUM_BYTES_IN_DW; 364c694cb4cSShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 365c694cb4cSShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 366c694cb4cSShashi Mallela 367c694cb4cSShashi Mallela if (res != MEMTX_OK) { 368c694cb4cSShashi Mallela return result; 369c694cb4cSShashi Mallela } 370c694cb4cSShashi Mallela 371c694cb4cSShashi Mallela icid = value & ICID_MASK; 372c694cb4cSShashi Mallela 373c694cb4cSShashi Mallela dte = get_dte(s, devid, &res); 374c694cb4cSShashi Mallela 375c694cb4cSShashi Mallela if (res != MEMTX_OK) { 376c694cb4cSShashi Mallela return result; 377c694cb4cSShashi Mallela } 378e07f8445SPeter Maydell dte_valid = FIELD_EX64(dte, DTE, VALID); 379e07f8445SPeter Maydell max_eventid = 1UL << (FIELD_EX64(dte, DTE, SIZE) + 1); 380c694cb4cSShashi Mallela max_Intid = (1ULL << (GICD_TYPER_IDBITS + 1)) - 1; 381c694cb4cSShashi Mallela 3826c1db43dSPeter Maydell if ((devid > s->dt.max_ids) || (icid > s->ct.max_ids) 383c694cb4cSShashi Mallela || !dte_valid || (eventid > max_eventid) || 384b87fab1cSPeter Maydell (((pIntid < GICV3_LPI_INTID_START) || (pIntid > max_Intid)) && 385b87fab1cSPeter Maydell (pIntid != INTID_SPURIOUS))) { 386c694cb4cSShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 387c694cb4cSShashi Mallela "%s: invalid command attributes " 388c694cb4cSShashi Mallela "devid %d or icid %d or eventid %d or pIntid %d or" 389c694cb4cSShashi Mallela "unmapped dte %d\n", __func__, devid, icid, eventid, 390c694cb4cSShashi Mallela pIntid, dte_valid); 391c694cb4cSShashi Mallela /* 392c694cb4cSShashi Mallela * in this implementation, in case of error 393c694cb4cSShashi Mallela * we ignore this command and move onto the next 394c694cb4cSShashi Mallela * command in the queue 395c694cb4cSShashi Mallela */ 396c694cb4cSShashi Mallela } else { 397c694cb4cSShashi Mallela /* add ite entry to interrupt translation table */ 398764d6ba1SPeter Maydell IteEntry ite = {}; 399764d6ba1SPeter Maydell ite.itel = FIELD_DP64(ite.itel, ITE_L, VALID, dte_valid); 400764d6ba1SPeter Maydell ite.itel = FIELD_DP64(ite.itel, ITE_L, INTTYPE, ITE_INTTYPE_PHYSICAL); 401764d6ba1SPeter Maydell ite.itel = FIELD_DP64(ite.itel, ITE_L, INTID, pIntid); 402764d6ba1SPeter Maydell ite.itel = FIELD_DP64(ite.itel, ITE_L, DOORBELL, INTID_SPURIOUS); 403764d6ba1SPeter Maydell ite.iteh = FIELD_DP32(ite.iteh, ITE_H, ICID, icid); 404c694cb4cSShashi Mallela 405c694cb4cSShashi Mallela result = update_ite(s, eventid, dte, ite); 406c694cb4cSShashi Mallela } 407c694cb4cSShashi Mallela 408c694cb4cSShashi Mallela return result; 409c694cb4cSShashi Mallela } 410c694cb4cSShashi Mallela 4117eca39e0SShashi Mallela static bool update_cte(GICv3ITSState *s, uint16_t icid, bool valid, 4127eca39e0SShashi Mallela uint64_t rdbase) 4137eca39e0SShashi Mallela { 4147eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 4157eca39e0SShashi Mallela uint64_t value; 4167eca39e0SShashi Mallela uint64_t l2t_addr; 4177eca39e0SShashi Mallela bool valid_l2t; 4187eca39e0SShashi Mallela uint32_t l2t_id; 4197eca39e0SShashi Mallela uint32_t max_l2_entries; 4207eca39e0SShashi Mallela uint64_t cte = 0; 4217eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 4227eca39e0SShashi Mallela 4237eca39e0SShashi Mallela if (!s->ct.valid) { 4247eca39e0SShashi Mallela return true; 4257eca39e0SShashi Mallela } 4267eca39e0SShashi Mallela 4277eca39e0SShashi Mallela if (valid) { 4287eca39e0SShashi Mallela /* add mapping entry to collection table */ 429*437dc0eaSPeter Maydell cte = FIELD_DP64(cte, CTE, VALID, 1); 430*437dc0eaSPeter Maydell cte = FIELD_DP64(cte, CTE, RDBASE, rdbase); 4317eca39e0SShashi Mallela } 4327eca39e0SShashi Mallela 4337eca39e0SShashi Mallela /* 4347eca39e0SShashi Mallela * The specification defines the format of level 1 entries of a 4357eca39e0SShashi Mallela * 2-level table, but the format of level 2 entries and the format 4367eca39e0SShashi Mallela * of flat-mapped tables is IMPDEF. 4377eca39e0SShashi Mallela */ 4387eca39e0SShashi Mallela if (s->ct.indirect) { 4397eca39e0SShashi Mallela l2t_id = icid / (s->ct.page_sz / L1TABLE_ENTRY_SIZE); 4407eca39e0SShashi Mallela 4417eca39e0SShashi Mallela value = address_space_ldq_le(as, 4427eca39e0SShashi Mallela s->ct.base_addr + 4437eca39e0SShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 4447eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 4457eca39e0SShashi Mallela 4467eca39e0SShashi Mallela if (res != MEMTX_OK) { 4477eca39e0SShashi Mallela return false; 4487eca39e0SShashi Mallela } 4497eca39e0SShashi Mallela 4507eca39e0SShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 4517eca39e0SShashi Mallela 4527eca39e0SShashi Mallela if (valid_l2t) { 4537eca39e0SShashi Mallela max_l2_entries = s->ct.page_sz / s->ct.entry_sz; 4547eca39e0SShashi Mallela 4557eca39e0SShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 4567eca39e0SShashi Mallela 4577eca39e0SShashi Mallela address_space_stq_le(as, l2t_addr + 4587eca39e0SShashi Mallela ((icid % max_l2_entries) * GITS_CTE_SIZE), 4597eca39e0SShashi Mallela cte, MEMTXATTRS_UNSPECIFIED, &res); 4607eca39e0SShashi Mallela } 4617eca39e0SShashi Mallela } else { 4627eca39e0SShashi Mallela /* Flat level table */ 4637eca39e0SShashi Mallela address_space_stq_le(as, s->ct.base_addr + (icid * GITS_CTE_SIZE), 4647eca39e0SShashi Mallela cte, MEMTXATTRS_UNSPECIFIED, &res); 4657eca39e0SShashi Mallela } 4667eca39e0SShashi Mallela if (res != MEMTX_OK) { 4677eca39e0SShashi Mallela return false; 4687eca39e0SShashi Mallela } else { 4697eca39e0SShashi Mallela return true; 4707eca39e0SShashi Mallela } 4717eca39e0SShashi Mallela } 4727eca39e0SShashi Mallela 4737eca39e0SShashi Mallela static bool process_mapc(GICv3ITSState *s, uint32_t offset) 4747eca39e0SShashi Mallela { 4757eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 4767eca39e0SShashi Mallela uint16_t icid; 4777eca39e0SShashi Mallela uint64_t rdbase; 4787eca39e0SShashi Mallela bool valid; 4797eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 4807eca39e0SShashi Mallela bool result = false; 4817eca39e0SShashi Mallela uint64_t value; 4827eca39e0SShashi Mallela 4837eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 4847eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 4857eca39e0SShashi Mallela 4867eca39e0SShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 4877eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 4887eca39e0SShashi Mallela 4897eca39e0SShashi Mallela if (res != MEMTX_OK) { 4907eca39e0SShashi Mallela return result; 4917eca39e0SShashi Mallela } 4927eca39e0SShashi Mallela 4937eca39e0SShashi Mallela icid = value & ICID_MASK; 4947eca39e0SShashi Mallela 4957eca39e0SShashi Mallela rdbase = (value & R_MAPC_RDBASE_MASK) >> R_MAPC_RDBASE_SHIFT; 4967eca39e0SShashi Mallela rdbase &= RDBASE_PROCNUM_MASK; 4977eca39e0SShashi Mallela 4987eca39e0SShashi Mallela valid = (value & CMD_FIELD_VALID_MASK); 4997eca39e0SShashi Mallela 5006c1db43dSPeter Maydell if ((icid > s->ct.max_ids) || (rdbase >= s->gicv3->num_cpu)) { 5017eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 5027eca39e0SShashi Mallela "ITS MAPC: invalid collection table attributes " 5037eca39e0SShashi Mallela "icid %d rdbase %" PRIu64 "\n", icid, rdbase); 5047eca39e0SShashi Mallela /* 5057eca39e0SShashi Mallela * in this implementation, in case of error 5067eca39e0SShashi Mallela * we ignore this command and move onto the next 5077eca39e0SShashi Mallela * command in the queue 5087eca39e0SShashi Mallela */ 5097eca39e0SShashi Mallela } else { 5107eca39e0SShashi Mallela result = update_cte(s, icid, valid, rdbase); 5117eca39e0SShashi Mallela } 5127eca39e0SShashi Mallela 5137eca39e0SShashi Mallela return result; 5147eca39e0SShashi Mallela } 5157eca39e0SShashi Mallela 5167eca39e0SShashi Mallela static bool update_dte(GICv3ITSState *s, uint32_t devid, bool valid, 5177eca39e0SShashi Mallela uint8_t size, uint64_t itt_addr) 5187eca39e0SShashi Mallela { 5197eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 5207eca39e0SShashi Mallela uint64_t value; 5217eca39e0SShashi Mallela uint64_t l2t_addr; 5227eca39e0SShashi Mallela bool valid_l2t; 5237eca39e0SShashi Mallela uint32_t l2t_id; 5247eca39e0SShashi Mallela uint32_t max_l2_entries; 5257eca39e0SShashi Mallela uint64_t dte = 0; 5267eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 5277eca39e0SShashi Mallela 5287eca39e0SShashi Mallela if (s->dt.valid) { 5297eca39e0SShashi Mallela if (valid) { 5307eca39e0SShashi Mallela /* add mapping entry to device table */ 531e07f8445SPeter Maydell dte = FIELD_DP64(dte, DTE, VALID, 1); 532e07f8445SPeter Maydell dte = FIELD_DP64(dte, DTE, SIZE, size); 533e07f8445SPeter Maydell dte = FIELD_DP64(dte, DTE, ITTADDR, itt_addr); 5347eca39e0SShashi Mallela } 5357eca39e0SShashi Mallela } else { 5367eca39e0SShashi Mallela return true; 5377eca39e0SShashi Mallela } 5387eca39e0SShashi Mallela 5397eca39e0SShashi Mallela /* 5407eca39e0SShashi Mallela * The specification defines the format of level 1 entries of a 5417eca39e0SShashi Mallela * 2-level table, but the format of level 2 entries and the format 5427eca39e0SShashi Mallela * of flat-mapped tables is IMPDEF. 5437eca39e0SShashi Mallela */ 5447eca39e0SShashi Mallela if (s->dt.indirect) { 5457eca39e0SShashi Mallela l2t_id = devid / (s->dt.page_sz / L1TABLE_ENTRY_SIZE); 5467eca39e0SShashi Mallela 5477eca39e0SShashi Mallela value = address_space_ldq_le(as, 5487eca39e0SShashi Mallela s->dt.base_addr + 5497eca39e0SShashi Mallela (l2t_id * L1TABLE_ENTRY_SIZE), 5507eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 5517eca39e0SShashi Mallela 5527eca39e0SShashi Mallela if (res != MEMTX_OK) { 5537eca39e0SShashi Mallela return false; 5547eca39e0SShashi Mallela } 5557eca39e0SShashi Mallela 5567eca39e0SShashi Mallela valid_l2t = (value & L2_TABLE_VALID_MASK) != 0; 5577eca39e0SShashi Mallela 5587eca39e0SShashi Mallela if (valid_l2t) { 5597eca39e0SShashi Mallela max_l2_entries = s->dt.page_sz / s->dt.entry_sz; 5607eca39e0SShashi Mallela 5617eca39e0SShashi Mallela l2t_addr = value & ((1ULL << 51) - 1); 5627eca39e0SShashi Mallela 5637eca39e0SShashi Mallela address_space_stq_le(as, l2t_addr + 5647eca39e0SShashi Mallela ((devid % max_l2_entries) * GITS_DTE_SIZE), 5657eca39e0SShashi Mallela dte, MEMTXATTRS_UNSPECIFIED, &res); 5667eca39e0SShashi Mallela } 5677eca39e0SShashi Mallela } else { 5687eca39e0SShashi Mallela /* Flat level table */ 5697eca39e0SShashi Mallela address_space_stq_le(as, s->dt.base_addr + (devid * GITS_DTE_SIZE), 5707eca39e0SShashi Mallela dte, MEMTXATTRS_UNSPECIFIED, &res); 5717eca39e0SShashi Mallela } 5727eca39e0SShashi Mallela if (res != MEMTX_OK) { 5737eca39e0SShashi Mallela return false; 5747eca39e0SShashi Mallela } else { 5757eca39e0SShashi Mallela return true; 5767eca39e0SShashi Mallela } 5777eca39e0SShashi Mallela } 5787eca39e0SShashi Mallela 5797eca39e0SShashi Mallela static bool process_mapd(GICv3ITSState *s, uint64_t value, uint32_t offset) 5807eca39e0SShashi Mallela { 5817eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 5827eca39e0SShashi Mallela uint32_t devid; 5837eca39e0SShashi Mallela uint8_t size; 5847eca39e0SShashi Mallela uint64_t itt_addr; 5857eca39e0SShashi Mallela bool valid; 5867eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 5877eca39e0SShashi Mallela bool result = false; 5887eca39e0SShashi Mallela 5897eca39e0SShashi Mallela devid = ((value & DEVID_MASK) >> DEVID_SHIFT); 5907eca39e0SShashi Mallela 5917eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 5927eca39e0SShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 5937eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 5947eca39e0SShashi Mallela 5957eca39e0SShashi Mallela if (res != MEMTX_OK) { 5967eca39e0SShashi Mallela return result; 5977eca39e0SShashi Mallela } 5987eca39e0SShashi Mallela 5997eca39e0SShashi Mallela size = (value & SIZE_MASK); 6007eca39e0SShashi Mallela 6017eca39e0SShashi Mallela offset += NUM_BYTES_IN_DW; 6027eca39e0SShashi Mallela value = address_space_ldq_le(as, s->cq.base_addr + offset, 6037eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 6047eca39e0SShashi Mallela 6057eca39e0SShashi Mallela if (res != MEMTX_OK) { 6067eca39e0SShashi Mallela return result; 6077eca39e0SShashi Mallela } 6087eca39e0SShashi Mallela 6097eca39e0SShashi Mallela itt_addr = (value & ITTADDR_MASK) >> ITTADDR_SHIFT; 6107eca39e0SShashi Mallela 6117eca39e0SShashi Mallela valid = (value & CMD_FIELD_VALID_MASK); 6127eca39e0SShashi Mallela 6136c1db43dSPeter Maydell if ((devid > s->dt.max_ids) || 6147eca39e0SShashi Mallela (size > FIELD_EX64(s->typer, GITS_TYPER, IDBITS))) { 6157eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 6167eca39e0SShashi Mallela "ITS MAPD: invalid device table attributes " 6177eca39e0SShashi Mallela "devid %d or size %d\n", devid, size); 6187eca39e0SShashi Mallela /* 6197eca39e0SShashi Mallela * in this implementation, in case of error 6207eca39e0SShashi Mallela * we ignore this command and move onto the next 6217eca39e0SShashi Mallela * command in the queue 6227eca39e0SShashi Mallela */ 6237eca39e0SShashi Mallela } else { 6247eca39e0SShashi Mallela result = update_dte(s, devid, valid, size, itt_addr); 6257eca39e0SShashi Mallela } 6267eca39e0SShashi Mallela 6277eca39e0SShashi Mallela return result; 6287eca39e0SShashi Mallela } 6297eca39e0SShashi Mallela 6307eca39e0SShashi Mallela /* 6317eca39e0SShashi Mallela * Current implementation blocks until all 6327eca39e0SShashi Mallela * commands are processed 6337eca39e0SShashi Mallela */ 6347eca39e0SShashi Mallela static void process_cmdq(GICv3ITSState *s) 6357eca39e0SShashi Mallela { 6367eca39e0SShashi Mallela uint32_t wr_offset = 0; 6377eca39e0SShashi Mallela uint32_t rd_offset = 0; 6387eca39e0SShashi Mallela uint32_t cq_offset = 0; 6397eca39e0SShashi Mallela uint64_t data; 6407eca39e0SShashi Mallela AddressSpace *as = &s->gicv3->dma_as; 6417eca39e0SShashi Mallela MemTxResult res = MEMTX_OK; 6427eca39e0SShashi Mallela bool result = true; 6437eca39e0SShashi Mallela uint8_t cmd; 64417fb5e36SShashi Mallela int i; 6457eca39e0SShashi Mallela 6468d2d6dd9SPeter Maydell if (!(s->ctlr & R_GITS_CTLR_ENABLED_MASK)) { 6477eca39e0SShashi Mallela return; 6487eca39e0SShashi Mallela } 6497eca39e0SShashi Mallela 6507eca39e0SShashi Mallela wr_offset = FIELD_EX64(s->cwriter, GITS_CWRITER, OFFSET); 6517eca39e0SShashi Mallela 6527eca39e0SShashi Mallela if (wr_offset > s->cq.max_entries) { 6537eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 6547eca39e0SShashi Mallela "%s: invalid write offset " 6557eca39e0SShashi Mallela "%d\n", __func__, wr_offset); 6567eca39e0SShashi Mallela return; 6577eca39e0SShashi Mallela } 6587eca39e0SShashi Mallela 6597eca39e0SShashi Mallela rd_offset = FIELD_EX64(s->creadr, GITS_CREADR, OFFSET); 6607eca39e0SShashi Mallela 6617eca39e0SShashi Mallela if (rd_offset > s->cq.max_entries) { 6627eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 6637eca39e0SShashi Mallela "%s: invalid read offset " 6647eca39e0SShashi Mallela "%d\n", __func__, rd_offset); 6657eca39e0SShashi Mallela return; 6667eca39e0SShashi Mallela } 6677eca39e0SShashi Mallela 6687eca39e0SShashi Mallela while (wr_offset != rd_offset) { 6697eca39e0SShashi Mallela cq_offset = (rd_offset * GITS_CMDQ_ENTRY_SIZE); 6707eca39e0SShashi Mallela data = address_space_ldq_le(as, s->cq.base_addr + cq_offset, 6717eca39e0SShashi Mallela MEMTXATTRS_UNSPECIFIED, &res); 6727eca39e0SShashi Mallela if (res != MEMTX_OK) { 6737eca39e0SShashi Mallela result = false; 6747eca39e0SShashi Mallela } 6757eca39e0SShashi Mallela cmd = (data & CMD_MASK); 6767eca39e0SShashi Mallela 6777eca39e0SShashi Mallela switch (cmd) { 6787eca39e0SShashi Mallela case GITS_CMD_INT: 679c694cb4cSShashi Mallela res = process_its_cmd(s, data, cq_offset, INTERRUPT); 6807eca39e0SShashi Mallela break; 6817eca39e0SShashi Mallela case GITS_CMD_CLEAR: 682c694cb4cSShashi Mallela res = process_its_cmd(s, data, cq_offset, CLEAR); 6837eca39e0SShashi Mallela break; 6847eca39e0SShashi Mallela case GITS_CMD_SYNC: 6857eca39e0SShashi Mallela /* 6867eca39e0SShashi Mallela * Current implementation makes a blocking synchronous call 6877eca39e0SShashi Mallela * for every command issued earlier, hence the internal state 6887eca39e0SShashi Mallela * is already consistent by the time SYNC command is executed. 6897eca39e0SShashi Mallela * Hence no further processing is required for SYNC command. 6907eca39e0SShashi Mallela */ 6917eca39e0SShashi Mallela break; 6927eca39e0SShashi Mallela case GITS_CMD_MAPD: 6937eca39e0SShashi Mallela result = process_mapd(s, data, cq_offset); 6947eca39e0SShashi Mallela break; 6957eca39e0SShashi Mallela case GITS_CMD_MAPC: 6967eca39e0SShashi Mallela result = process_mapc(s, cq_offset); 6977eca39e0SShashi Mallela break; 6987eca39e0SShashi Mallela case GITS_CMD_MAPTI: 699c694cb4cSShashi Mallela result = process_mapti(s, data, cq_offset, false); 7007eca39e0SShashi Mallela break; 7017eca39e0SShashi Mallela case GITS_CMD_MAPI: 702c694cb4cSShashi Mallela result = process_mapti(s, data, cq_offset, true); 7037eca39e0SShashi Mallela break; 7047eca39e0SShashi Mallela case GITS_CMD_DISCARD: 705c694cb4cSShashi Mallela result = process_its_cmd(s, data, cq_offset, DISCARD); 7067eca39e0SShashi Mallela break; 7077eca39e0SShashi Mallela case GITS_CMD_INV: 7087eca39e0SShashi Mallela case GITS_CMD_INVALL: 70917fb5e36SShashi Mallela /* 71017fb5e36SShashi Mallela * Current implementation doesn't cache any ITS tables, 71117fb5e36SShashi Mallela * but the calculated lpi priority information. We only 71217fb5e36SShashi Mallela * need to trigger lpi priority re-calculation to be in 71317fb5e36SShashi Mallela * sync with LPI config table or pending table changes. 71417fb5e36SShashi Mallela */ 71517fb5e36SShashi Mallela for (i = 0; i < s->gicv3->num_cpu; i++) { 71617fb5e36SShashi Mallela gicv3_redist_update_lpi(&s->gicv3->cpu[i]); 71717fb5e36SShashi Mallela } 7187eca39e0SShashi Mallela break; 7197eca39e0SShashi Mallela default: 7207eca39e0SShashi Mallela break; 7217eca39e0SShashi Mallela } 7227eca39e0SShashi Mallela if (result) { 7237eca39e0SShashi Mallela rd_offset++; 7247eca39e0SShashi Mallela rd_offset %= s->cq.max_entries; 7257eca39e0SShashi Mallela s->creadr = FIELD_DP64(s->creadr, GITS_CREADR, OFFSET, rd_offset); 7267eca39e0SShashi Mallela } else { 7277eca39e0SShashi Mallela /* 7287eca39e0SShashi Mallela * in this implementation, in case of dma read/write error 7297eca39e0SShashi Mallela * we stall the command processing 7307eca39e0SShashi Mallela */ 7317eca39e0SShashi Mallela s->creadr = FIELD_DP64(s->creadr, GITS_CREADR, STALLED, 1); 7327eca39e0SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 7337eca39e0SShashi Mallela "%s: %x cmd processing failed\n", __func__, cmd); 7347eca39e0SShashi Mallela break; 7357eca39e0SShashi Mallela } 7367eca39e0SShashi Mallela } 7377eca39e0SShashi Mallela } 7387eca39e0SShashi Mallela 7391b08e436SShashi Mallela /* 7401b08e436SShashi Mallela * This function extracts the ITS Device and Collection table specific 7411b08e436SShashi Mallela * parameters (like base_addr, size etc) from GITS_BASER register. 7421b08e436SShashi Mallela * It is called during ITS enable and also during post_load migration 7431b08e436SShashi Mallela */ 7441b08e436SShashi Mallela static void extract_table_params(GICv3ITSState *s) 7451b08e436SShashi Mallela { 7461b08e436SShashi Mallela uint16_t num_pages = 0; 7471b08e436SShashi Mallela uint8_t page_sz_type; 7481b08e436SShashi Mallela uint8_t type; 7491b08e436SShashi Mallela uint32_t page_sz = 0; 7501b08e436SShashi Mallela uint64_t value; 7511b08e436SShashi Mallela 7521b08e436SShashi Mallela for (int i = 0; i < 8; i++) { 753e5487a41SPeter Maydell TableDesc *td; 754e5487a41SPeter Maydell int idbits; 755e5487a41SPeter Maydell 7561b08e436SShashi Mallela value = s->baser[i]; 7571b08e436SShashi Mallela 7581b08e436SShashi Mallela if (!value) { 7591b08e436SShashi Mallela continue; 7601b08e436SShashi Mallela } 7611b08e436SShashi Mallela 7621b08e436SShashi Mallela page_sz_type = FIELD_EX64(value, GITS_BASER, PAGESIZE); 7631b08e436SShashi Mallela 7641b08e436SShashi Mallela switch (page_sz_type) { 7651b08e436SShashi Mallela case 0: 7661b08e436SShashi Mallela page_sz = GITS_PAGE_SIZE_4K; 7671b08e436SShashi Mallela break; 7681b08e436SShashi Mallela 7691b08e436SShashi Mallela case 1: 7701b08e436SShashi Mallela page_sz = GITS_PAGE_SIZE_16K; 7711b08e436SShashi Mallela break; 7721b08e436SShashi Mallela 7731b08e436SShashi Mallela case 2: 7741b08e436SShashi Mallela case 3: 7751b08e436SShashi Mallela page_sz = GITS_PAGE_SIZE_64K; 7761b08e436SShashi Mallela break; 7771b08e436SShashi Mallela 7781b08e436SShashi Mallela default: 7791b08e436SShashi Mallela g_assert_not_reached(); 7801b08e436SShashi Mallela } 7811b08e436SShashi Mallela 7821b08e436SShashi Mallela num_pages = FIELD_EX64(value, GITS_BASER, SIZE) + 1; 7831b08e436SShashi Mallela 7841b08e436SShashi Mallela type = FIELD_EX64(value, GITS_BASER, TYPE); 7851b08e436SShashi Mallela 7861b08e436SShashi Mallela switch (type) { 7871b08e436SShashi Mallela case GITS_BASER_TYPE_DEVICE: 788e5487a41SPeter Maydell td = &s->dt; 789e5487a41SPeter Maydell idbits = FIELD_EX64(s->typer, GITS_TYPER, DEVBITS) + 1; 79062df780eSPeter Maydell break; 7911b08e436SShashi Mallela case GITS_BASER_TYPE_COLLECTION: 792e5487a41SPeter Maydell td = &s->ct; 7931b08e436SShashi Mallela if (FIELD_EX64(s->typer, GITS_TYPER, CIL)) { 794e5487a41SPeter Maydell idbits = FIELD_EX64(s->typer, GITS_TYPER, CIDBITS) + 1; 7951b08e436SShashi Mallela } else { 7961b08e436SShashi Mallela /* 16-bit CollectionId supported when CIL == 0 */ 797e5487a41SPeter Maydell idbits = 16; 7981b08e436SShashi Mallela } 7991b08e436SShashi Mallela break; 8001b08e436SShashi Mallela default: 801e5487a41SPeter Maydell /* 802e5487a41SPeter Maydell * GITS_BASER<n>.TYPE is read-only, so GITS_BASER_RO_MASK 803e5487a41SPeter Maydell * ensures we will only see type values corresponding to 804e5487a41SPeter Maydell * the values set up in gicv3_its_reset(). 805e5487a41SPeter Maydell */ 806e5487a41SPeter Maydell g_assert_not_reached(); 8071b08e436SShashi Mallela } 808e5487a41SPeter Maydell 809e5487a41SPeter Maydell memset(td, 0, sizeof(*td)); 810e5487a41SPeter Maydell td->valid = FIELD_EX64(value, GITS_BASER, VALID); 811e5487a41SPeter Maydell /* 812e5487a41SPeter Maydell * If GITS_BASER<n>.Valid is 0 for any <n> then we will not process 813e5487a41SPeter Maydell * interrupts. (GITS_TYPER.HCC is 0 for this implementation, so we 814e5487a41SPeter Maydell * do not have a special case where the GITS_BASER<n>.Valid bit is 0 815e5487a41SPeter Maydell * for the register corresponding to the Collection table but we 816e5487a41SPeter Maydell * still have to process interrupts using non-memory-backed 817e5487a41SPeter Maydell * Collection table entries.) 818e5487a41SPeter Maydell */ 819e5487a41SPeter Maydell if (!td->valid) { 820e5487a41SPeter Maydell continue; 821e5487a41SPeter Maydell } 822e5487a41SPeter Maydell td->page_sz = page_sz; 823e5487a41SPeter Maydell td->indirect = FIELD_EX64(value, GITS_BASER, INDIRECT); 8249ae85431SPeter Maydell td->entry_sz = FIELD_EX64(value, GITS_BASER, ENTRYSIZE) + 1; 825e5487a41SPeter Maydell td->base_addr = baser_base_addr(value, page_sz); 826e5487a41SPeter Maydell if (!td->indirect) { 827e5487a41SPeter Maydell td->max_entries = (num_pages * page_sz) / td->entry_sz; 828e5487a41SPeter Maydell } else { 829e5487a41SPeter Maydell td->max_entries = (((num_pages * page_sz) / 830e5487a41SPeter Maydell L1TABLE_ENTRY_SIZE) * 831e5487a41SPeter Maydell (page_sz / td->entry_sz)); 832e5487a41SPeter Maydell } 833e5487a41SPeter Maydell td->max_ids = 1ULL << idbits; 8341b08e436SShashi Mallela } 8351b08e436SShashi Mallela } 8361b08e436SShashi Mallela 8371b08e436SShashi Mallela static void extract_cmdq_params(GICv3ITSState *s) 8381b08e436SShashi Mallela { 8391b08e436SShashi Mallela uint16_t num_pages = 0; 8401b08e436SShashi Mallela uint64_t value = s->cbaser; 8411b08e436SShashi Mallela 8421b08e436SShashi Mallela num_pages = FIELD_EX64(value, GITS_CBASER, SIZE) + 1; 8431b08e436SShashi Mallela 8441b08e436SShashi Mallela memset(&s->cq, 0 , sizeof(s->cq)); 8451b08e436SShashi Mallela s->cq.valid = FIELD_EX64(value, GITS_CBASER, VALID); 8461b08e436SShashi Mallela 8471b08e436SShashi Mallela if (s->cq.valid) { 8481b08e436SShashi Mallela s->cq.max_entries = (num_pages * GITS_PAGE_SIZE_4K) / 8491b08e436SShashi Mallela GITS_CMDQ_ENTRY_SIZE; 8501b08e436SShashi Mallela s->cq.base_addr = FIELD_EX64(value, GITS_CBASER, PHYADDR); 8511b08e436SShashi Mallela s->cq.base_addr <<= R_GITS_CBASER_PHYADDR_SHIFT; 8521b08e436SShashi Mallela } 8531b08e436SShashi Mallela } 8541b08e436SShashi Mallela 85518f6290aSShashi Mallela static MemTxResult gicv3_its_translation_write(void *opaque, hwaddr offset, 85618f6290aSShashi Mallela uint64_t data, unsigned size, 85718f6290aSShashi Mallela MemTxAttrs attrs) 85818f6290aSShashi Mallela { 859c694cb4cSShashi Mallela GICv3ITSState *s = (GICv3ITSState *)opaque; 860c694cb4cSShashi Mallela bool result = true; 861c694cb4cSShashi Mallela uint32_t devid = 0; 862c694cb4cSShashi Mallela 863c694cb4cSShashi Mallela switch (offset) { 864c694cb4cSShashi Mallela case GITS_TRANSLATER: 8658d2d6dd9SPeter Maydell if (s->ctlr & R_GITS_CTLR_ENABLED_MASK) { 866c694cb4cSShashi Mallela devid = attrs.requester_id; 867c694cb4cSShashi Mallela result = process_its_cmd(s, data, devid, NONE); 868c694cb4cSShashi Mallela } 869c694cb4cSShashi Mallela break; 870c694cb4cSShashi Mallela default: 871c694cb4cSShashi Mallela break; 872c694cb4cSShashi Mallela } 873c694cb4cSShashi Mallela 874c694cb4cSShashi Mallela if (result) { 87518f6290aSShashi Mallela return MEMTX_OK; 876c694cb4cSShashi Mallela } else { 877c694cb4cSShashi Mallela return MEMTX_ERROR; 878c694cb4cSShashi Mallela } 87918f6290aSShashi Mallela } 88018f6290aSShashi Mallela 88118f6290aSShashi Mallela static bool its_writel(GICv3ITSState *s, hwaddr offset, 88218f6290aSShashi Mallela uint64_t value, MemTxAttrs attrs) 88318f6290aSShashi Mallela { 88418f6290aSShashi Mallela bool result = true; 8851b08e436SShashi Mallela int index; 88618f6290aSShashi Mallela 8871b08e436SShashi Mallela switch (offset) { 8881b08e436SShashi Mallela case GITS_CTLR: 8892f459cd1SShashi Mallela if (value & R_GITS_CTLR_ENABLED_MASK) { 8908d2d6dd9SPeter Maydell s->ctlr |= R_GITS_CTLR_ENABLED_MASK; 8911b08e436SShashi Mallela extract_table_params(s); 8921b08e436SShashi Mallela extract_cmdq_params(s); 8931b08e436SShashi Mallela s->creadr = 0; 8947eca39e0SShashi Mallela process_cmdq(s); 8952f459cd1SShashi Mallela } else { 8968d2d6dd9SPeter Maydell s->ctlr &= ~R_GITS_CTLR_ENABLED_MASK; 8971b08e436SShashi Mallela } 8981b08e436SShashi Mallela break; 8991b08e436SShashi Mallela case GITS_CBASER: 9001b08e436SShashi Mallela /* 9011b08e436SShashi Mallela * IMPDEF choice:- GITS_CBASER register becomes RO if ITS is 9021b08e436SShashi Mallela * already enabled 9031b08e436SShashi Mallela */ 9048d2d6dd9SPeter Maydell if (!(s->ctlr & R_GITS_CTLR_ENABLED_MASK)) { 9051b08e436SShashi Mallela s->cbaser = deposit64(s->cbaser, 0, 32, value); 9061b08e436SShashi Mallela s->creadr = 0; 9071b08e436SShashi Mallela s->cwriter = s->creadr; 9081b08e436SShashi Mallela } 9091b08e436SShashi Mallela break; 9101b08e436SShashi Mallela case GITS_CBASER + 4: 9111b08e436SShashi Mallela /* 9121b08e436SShashi Mallela * IMPDEF choice:- GITS_CBASER register becomes RO if ITS is 9131b08e436SShashi Mallela * already enabled 9141b08e436SShashi Mallela */ 9158d2d6dd9SPeter Maydell if (!(s->ctlr & R_GITS_CTLR_ENABLED_MASK)) { 9161b08e436SShashi Mallela s->cbaser = deposit64(s->cbaser, 32, 32, value); 9171b08e436SShashi Mallela s->creadr = 0; 9181b08e436SShashi Mallela s->cwriter = s->creadr; 9191b08e436SShashi Mallela } 9201b08e436SShashi Mallela break; 9211b08e436SShashi Mallela case GITS_CWRITER: 9221b08e436SShashi Mallela s->cwriter = deposit64(s->cwriter, 0, 32, 9231b08e436SShashi Mallela (value & ~R_GITS_CWRITER_RETRY_MASK)); 9247eca39e0SShashi Mallela if (s->cwriter != s->creadr) { 9257eca39e0SShashi Mallela process_cmdq(s); 9267eca39e0SShashi Mallela } 9271b08e436SShashi Mallela break; 9281b08e436SShashi Mallela case GITS_CWRITER + 4: 9291b08e436SShashi Mallela s->cwriter = deposit64(s->cwriter, 32, 32, value); 9301b08e436SShashi Mallela break; 9311b08e436SShashi Mallela case GITS_CREADR: 9321b08e436SShashi Mallela if (s->gicv3->gicd_ctlr & GICD_CTLR_DS) { 9331b08e436SShashi Mallela s->creadr = deposit64(s->creadr, 0, 32, 9341b08e436SShashi Mallela (value & ~R_GITS_CREADR_STALLED_MASK)); 9351b08e436SShashi Mallela } else { 9361b08e436SShashi Mallela /* RO register, ignore the write */ 9371b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 9381b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 9391b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 9401b08e436SShashi Mallela } 9411b08e436SShashi Mallela break; 9421b08e436SShashi Mallela case GITS_CREADR + 4: 9431b08e436SShashi Mallela if (s->gicv3->gicd_ctlr & GICD_CTLR_DS) { 9441b08e436SShashi Mallela s->creadr = deposit64(s->creadr, 32, 32, value); 9451b08e436SShashi Mallela } else { 9461b08e436SShashi Mallela /* RO register, ignore the write */ 9471b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 9481b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 9491b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 9501b08e436SShashi Mallela } 9511b08e436SShashi Mallela break; 9521b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 9531b08e436SShashi Mallela /* 9541b08e436SShashi Mallela * IMPDEF choice:- GITS_BASERn register becomes RO if ITS is 9551b08e436SShashi Mallela * already enabled 9561b08e436SShashi Mallela */ 9578d2d6dd9SPeter Maydell if (!(s->ctlr & R_GITS_CTLR_ENABLED_MASK)) { 9581b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 9591b08e436SShashi Mallela 9601b08e436SShashi Mallela if (offset & 7) { 9611b08e436SShashi Mallela value <<= 32; 9621b08e436SShashi Mallela value &= ~GITS_BASER_RO_MASK; 9631b08e436SShashi Mallela s->baser[index] &= GITS_BASER_RO_MASK | MAKE_64BIT_MASK(0, 32); 9641b08e436SShashi Mallela s->baser[index] |= value; 9651b08e436SShashi Mallela } else { 9661b08e436SShashi Mallela value &= ~GITS_BASER_RO_MASK; 9671b08e436SShashi Mallela s->baser[index] &= GITS_BASER_RO_MASK | MAKE_64BIT_MASK(32, 32); 9681b08e436SShashi Mallela s->baser[index] |= value; 9691b08e436SShashi Mallela } 9701b08e436SShashi Mallela } 9711b08e436SShashi Mallela break; 9721b08e436SShashi Mallela case GITS_IIDR: 9731b08e436SShashi Mallela case GITS_IDREGS ... GITS_IDREGS + 0x2f: 9741b08e436SShashi Mallela /* RO registers, ignore the write */ 9751b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 9761b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 9771b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 9781b08e436SShashi Mallela break; 9791b08e436SShashi Mallela default: 9801b08e436SShashi Mallela result = false; 9811b08e436SShashi Mallela break; 9821b08e436SShashi Mallela } 98318f6290aSShashi Mallela return result; 98418f6290aSShashi Mallela } 98518f6290aSShashi Mallela 98618f6290aSShashi Mallela static bool its_readl(GICv3ITSState *s, hwaddr offset, 98718f6290aSShashi Mallela uint64_t *data, MemTxAttrs attrs) 98818f6290aSShashi Mallela { 98918f6290aSShashi Mallela bool result = true; 9901b08e436SShashi Mallela int index; 99118f6290aSShashi Mallela 9921b08e436SShashi Mallela switch (offset) { 9931b08e436SShashi Mallela case GITS_CTLR: 9941b08e436SShashi Mallela *data = s->ctlr; 9951b08e436SShashi Mallela break; 9961b08e436SShashi Mallela case GITS_IIDR: 9971b08e436SShashi Mallela *data = gicv3_iidr(); 9981b08e436SShashi Mallela break; 9991b08e436SShashi Mallela case GITS_IDREGS ... GITS_IDREGS + 0x2f: 10001b08e436SShashi Mallela /* ID registers */ 10011b08e436SShashi Mallela *data = gicv3_idreg(offset - GITS_IDREGS); 10021b08e436SShashi Mallela break; 10031b08e436SShashi Mallela case GITS_TYPER: 10041b08e436SShashi Mallela *data = extract64(s->typer, 0, 32); 10051b08e436SShashi Mallela break; 10061b08e436SShashi Mallela case GITS_TYPER + 4: 10071b08e436SShashi Mallela *data = extract64(s->typer, 32, 32); 10081b08e436SShashi Mallela break; 10091b08e436SShashi Mallela case GITS_CBASER: 10101b08e436SShashi Mallela *data = extract64(s->cbaser, 0, 32); 10111b08e436SShashi Mallela break; 10121b08e436SShashi Mallela case GITS_CBASER + 4: 10131b08e436SShashi Mallela *data = extract64(s->cbaser, 32, 32); 10141b08e436SShashi Mallela break; 10151b08e436SShashi Mallela case GITS_CREADR: 10161b08e436SShashi Mallela *data = extract64(s->creadr, 0, 32); 10171b08e436SShashi Mallela break; 10181b08e436SShashi Mallela case GITS_CREADR + 4: 10191b08e436SShashi Mallela *data = extract64(s->creadr, 32, 32); 10201b08e436SShashi Mallela break; 10211b08e436SShashi Mallela case GITS_CWRITER: 10221b08e436SShashi Mallela *data = extract64(s->cwriter, 0, 32); 10231b08e436SShashi Mallela break; 10241b08e436SShashi Mallela case GITS_CWRITER + 4: 10251b08e436SShashi Mallela *data = extract64(s->cwriter, 32, 32); 10261b08e436SShashi Mallela break; 10271b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 10281b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 10291b08e436SShashi Mallela if (offset & 7) { 10301b08e436SShashi Mallela *data = extract64(s->baser[index], 32, 32); 10311b08e436SShashi Mallela } else { 10321b08e436SShashi Mallela *data = extract64(s->baser[index], 0, 32); 10331b08e436SShashi Mallela } 10341b08e436SShashi Mallela break; 10351b08e436SShashi Mallela default: 10361b08e436SShashi Mallela result = false; 10371b08e436SShashi Mallela break; 10381b08e436SShashi Mallela } 103918f6290aSShashi Mallela return result; 104018f6290aSShashi Mallela } 104118f6290aSShashi Mallela 104218f6290aSShashi Mallela static bool its_writell(GICv3ITSState *s, hwaddr offset, 104318f6290aSShashi Mallela uint64_t value, MemTxAttrs attrs) 104418f6290aSShashi Mallela { 104518f6290aSShashi Mallela bool result = true; 10461b08e436SShashi Mallela int index; 104718f6290aSShashi Mallela 10481b08e436SShashi Mallela switch (offset) { 10491b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 10501b08e436SShashi Mallela /* 10511b08e436SShashi Mallela * IMPDEF choice:- GITS_BASERn register becomes RO if ITS is 10521b08e436SShashi Mallela * already enabled 10531b08e436SShashi Mallela */ 10548d2d6dd9SPeter Maydell if (!(s->ctlr & R_GITS_CTLR_ENABLED_MASK)) { 10551b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 10561b08e436SShashi Mallela s->baser[index] &= GITS_BASER_RO_MASK; 10571b08e436SShashi Mallela s->baser[index] |= (value & ~GITS_BASER_RO_MASK); 10581b08e436SShashi Mallela } 10591b08e436SShashi Mallela break; 10601b08e436SShashi Mallela case GITS_CBASER: 10611b08e436SShashi Mallela /* 10621b08e436SShashi Mallela * IMPDEF choice:- GITS_CBASER register becomes RO if ITS is 10631b08e436SShashi Mallela * already enabled 10641b08e436SShashi Mallela */ 10658d2d6dd9SPeter Maydell if (!(s->ctlr & R_GITS_CTLR_ENABLED_MASK)) { 10661b08e436SShashi Mallela s->cbaser = value; 10671b08e436SShashi Mallela s->creadr = 0; 10681b08e436SShashi Mallela s->cwriter = s->creadr; 10691b08e436SShashi Mallela } 10701b08e436SShashi Mallela break; 10711b08e436SShashi Mallela case GITS_CWRITER: 10721b08e436SShashi Mallela s->cwriter = value & ~R_GITS_CWRITER_RETRY_MASK; 10737eca39e0SShashi Mallela if (s->cwriter != s->creadr) { 10747eca39e0SShashi Mallela process_cmdq(s); 10757eca39e0SShashi Mallela } 10761b08e436SShashi Mallela break; 10771b08e436SShashi Mallela case GITS_CREADR: 10781b08e436SShashi Mallela if (s->gicv3->gicd_ctlr & GICD_CTLR_DS) { 10791b08e436SShashi Mallela s->creadr = value & ~R_GITS_CREADR_STALLED_MASK; 10801b08e436SShashi Mallela } else { 10811b08e436SShashi Mallela /* RO register, ignore the write */ 10821b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 10831b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 10841b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 10851b08e436SShashi Mallela } 10861b08e436SShashi Mallela break; 10871b08e436SShashi Mallela case GITS_TYPER: 10881b08e436SShashi Mallela /* RO registers, ignore the write */ 10891b08e436SShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 10901b08e436SShashi Mallela "%s: invalid guest write to RO register at offset " 10911b08e436SShashi Mallela TARGET_FMT_plx "\n", __func__, offset); 10921b08e436SShashi Mallela break; 10931b08e436SShashi Mallela default: 10941b08e436SShashi Mallela result = false; 10951b08e436SShashi Mallela break; 10961b08e436SShashi Mallela } 109718f6290aSShashi Mallela return result; 109818f6290aSShashi Mallela } 109918f6290aSShashi Mallela 110018f6290aSShashi Mallela static bool its_readll(GICv3ITSState *s, hwaddr offset, 110118f6290aSShashi Mallela uint64_t *data, MemTxAttrs attrs) 110218f6290aSShashi Mallela { 110318f6290aSShashi Mallela bool result = true; 11041b08e436SShashi Mallela int index; 110518f6290aSShashi Mallela 11061b08e436SShashi Mallela switch (offset) { 11071b08e436SShashi Mallela case GITS_TYPER: 11081b08e436SShashi Mallela *data = s->typer; 11091b08e436SShashi Mallela break; 11101b08e436SShashi Mallela case GITS_BASER ... GITS_BASER + 0x3f: 11111b08e436SShashi Mallela index = (offset - GITS_BASER) / 8; 11121b08e436SShashi Mallela *data = s->baser[index]; 11131b08e436SShashi Mallela break; 11141b08e436SShashi Mallela case GITS_CBASER: 11151b08e436SShashi Mallela *data = s->cbaser; 11161b08e436SShashi Mallela break; 11171b08e436SShashi Mallela case GITS_CREADR: 11181b08e436SShashi Mallela *data = s->creadr; 11191b08e436SShashi Mallela break; 11201b08e436SShashi Mallela case GITS_CWRITER: 11211b08e436SShashi Mallela *data = s->cwriter; 11221b08e436SShashi Mallela break; 11231b08e436SShashi Mallela default: 11241b08e436SShashi Mallela result = false; 11251b08e436SShashi Mallela break; 11261b08e436SShashi Mallela } 112718f6290aSShashi Mallela return result; 112818f6290aSShashi Mallela } 112918f6290aSShashi Mallela 113018f6290aSShashi Mallela static MemTxResult gicv3_its_read(void *opaque, hwaddr offset, uint64_t *data, 113118f6290aSShashi Mallela unsigned size, MemTxAttrs attrs) 113218f6290aSShashi Mallela { 113318f6290aSShashi Mallela GICv3ITSState *s = (GICv3ITSState *)opaque; 113418f6290aSShashi Mallela bool result; 113518f6290aSShashi Mallela 113618f6290aSShashi Mallela switch (size) { 113718f6290aSShashi Mallela case 4: 113818f6290aSShashi Mallela result = its_readl(s, offset, data, attrs); 113918f6290aSShashi Mallela break; 114018f6290aSShashi Mallela case 8: 114118f6290aSShashi Mallela result = its_readll(s, offset, data, attrs); 114218f6290aSShashi Mallela break; 114318f6290aSShashi Mallela default: 114418f6290aSShashi Mallela result = false; 114518f6290aSShashi Mallela break; 114618f6290aSShashi Mallela } 114718f6290aSShashi Mallela 114818f6290aSShashi Mallela if (!result) { 114918f6290aSShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 115018f6290aSShashi Mallela "%s: invalid guest read at offset " TARGET_FMT_plx 115118f6290aSShashi Mallela "size %u\n", __func__, offset, size); 115218f6290aSShashi Mallela /* 115318f6290aSShashi Mallela * The spec requires that reserved registers are RAZ/WI; 115418f6290aSShashi Mallela * so use false returns from leaf functions as a way to 115518f6290aSShashi Mallela * trigger the guest-error logging but don't return it to 115618f6290aSShashi Mallela * the caller, or we'll cause a spurious guest data abort. 115718f6290aSShashi Mallela */ 115818f6290aSShashi Mallela *data = 0; 115918f6290aSShashi Mallela } 116018f6290aSShashi Mallela return MEMTX_OK; 116118f6290aSShashi Mallela } 116218f6290aSShashi Mallela 116318f6290aSShashi Mallela static MemTxResult gicv3_its_write(void *opaque, hwaddr offset, uint64_t data, 116418f6290aSShashi Mallela unsigned size, MemTxAttrs attrs) 116518f6290aSShashi Mallela { 116618f6290aSShashi Mallela GICv3ITSState *s = (GICv3ITSState *)opaque; 116718f6290aSShashi Mallela bool result; 116818f6290aSShashi Mallela 116918f6290aSShashi Mallela switch (size) { 117018f6290aSShashi Mallela case 4: 117118f6290aSShashi Mallela result = its_writel(s, offset, data, attrs); 117218f6290aSShashi Mallela break; 117318f6290aSShashi Mallela case 8: 117418f6290aSShashi Mallela result = its_writell(s, offset, data, attrs); 117518f6290aSShashi Mallela break; 117618f6290aSShashi Mallela default: 117718f6290aSShashi Mallela result = false; 117818f6290aSShashi Mallela break; 117918f6290aSShashi Mallela } 118018f6290aSShashi Mallela 118118f6290aSShashi Mallela if (!result) { 118218f6290aSShashi Mallela qemu_log_mask(LOG_GUEST_ERROR, 118318f6290aSShashi Mallela "%s: invalid guest write at offset " TARGET_FMT_plx 118418f6290aSShashi Mallela "size %u\n", __func__, offset, size); 118518f6290aSShashi Mallela /* 118618f6290aSShashi Mallela * The spec requires that reserved registers are RAZ/WI; 118718f6290aSShashi Mallela * so use false returns from leaf functions as a way to 118818f6290aSShashi Mallela * trigger the guest-error logging but don't return it to 118918f6290aSShashi Mallela * the caller, or we'll cause a spurious guest data abort. 119018f6290aSShashi Mallela */ 119118f6290aSShashi Mallela } 119218f6290aSShashi Mallela return MEMTX_OK; 119318f6290aSShashi Mallela } 119418f6290aSShashi Mallela 119518f6290aSShashi Mallela static const MemoryRegionOps gicv3_its_control_ops = { 119618f6290aSShashi Mallela .read_with_attrs = gicv3_its_read, 119718f6290aSShashi Mallela .write_with_attrs = gicv3_its_write, 119818f6290aSShashi Mallela .valid.min_access_size = 4, 119918f6290aSShashi Mallela .valid.max_access_size = 8, 120018f6290aSShashi Mallela .impl.min_access_size = 4, 120118f6290aSShashi Mallela .impl.max_access_size = 8, 120218f6290aSShashi Mallela .endianness = DEVICE_NATIVE_ENDIAN, 120318f6290aSShashi Mallela }; 120418f6290aSShashi Mallela 120518f6290aSShashi Mallela static const MemoryRegionOps gicv3_its_translation_ops = { 120618f6290aSShashi Mallela .write_with_attrs = gicv3_its_translation_write, 120718f6290aSShashi Mallela .valid.min_access_size = 2, 120818f6290aSShashi Mallela .valid.max_access_size = 4, 120918f6290aSShashi Mallela .impl.min_access_size = 2, 121018f6290aSShashi Mallela .impl.max_access_size = 4, 121118f6290aSShashi Mallela .endianness = DEVICE_NATIVE_ENDIAN, 121218f6290aSShashi Mallela }; 121318f6290aSShashi Mallela 121418f6290aSShashi Mallela static void gicv3_arm_its_realize(DeviceState *dev, Error **errp) 121518f6290aSShashi Mallela { 121618f6290aSShashi Mallela GICv3ITSState *s = ARM_GICV3_ITS_COMMON(dev); 121718f6290aSShashi Mallela int i; 121818f6290aSShashi Mallela 121918f6290aSShashi Mallela for (i = 0; i < s->gicv3->num_cpu; i++) { 122018f6290aSShashi Mallela if (!(s->gicv3->cpu[i].gicr_typer & GICR_TYPER_PLPIS)) { 122118f6290aSShashi Mallela error_setg(errp, "Physical LPI not supported by CPU %d", i); 122218f6290aSShashi Mallela return; 122318f6290aSShashi Mallela } 122418f6290aSShashi Mallela } 122518f6290aSShashi Mallela 122618f6290aSShashi Mallela gicv3_its_init_mmio(s, &gicv3_its_control_ops, &gicv3_its_translation_ops); 122718f6290aSShashi Mallela 12281b08e436SShashi Mallela address_space_init(&s->gicv3->dma_as, s->gicv3->dma, 12291b08e436SShashi Mallela "gicv3-its-sysmem"); 12301b08e436SShashi Mallela 123118f6290aSShashi Mallela /* set the ITS default features supported */ 1232764d6ba1SPeter Maydell s->typer = FIELD_DP64(s->typer, GITS_TYPER, PHYSICAL, 1); 123318f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, ITT_ENTRY_SIZE, 123418f6290aSShashi Mallela ITS_ITT_ENTRY_SIZE - 1); 123518f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, IDBITS, ITS_IDBITS); 123618f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, DEVBITS, ITS_DEVBITS); 123718f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, CIL, 1); 123818f6290aSShashi Mallela s->typer = FIELD_DP64(s->typer, GITS_TYPER, CIDBITS, ITS_CIDBITS); 123918f6290aSShashi Mallela } 124018f6290aSShashi Mallela 124118f6290aSShashi Mallela static void gicv3_its_reset(DeviceState *dev) 124218f6290aSShashi Mallela { 124318f6290aSShashi Mallela GICv3ITSState *s = ARM_GICV3_ITS_COMMON(dev); 124418f6290aSShashi Mallela GICv3ITSClass *c = ARM_GICV3_ITS_GET_CLASS(s); 124518f6290aSShashi Mallela 124618f6290aSShashi Mallela c->parent_reset(dev); 124718f6290aSShashi Mallela 124818f6290aSShashi Mallela /* Quiescent bit reset to 1 */ 124918f6290aSShashi Mallela s->ctlr = FIELD_DP32(s->ctlr, GITS_CTLR, QUIESCENT, 1); 125018f6290aSShashi Mallela 125118f6290aSShashi Mallela /* 125218f6290aSShashi Mallela * setting GITS_BASER0.Type = 0b001 (Device) 125318f6290aSShashi Mallela * GITS_BASER1.Type = 0b100 (Collection Table) 125418f6290aSShashi Mallela * GITS_BASER<n>.Type,where n = 3 to 7 are 0b00 (Unimplemented) 125518f6290aSShashi Mallela * GITS_BASER<0,1>.Page_Size = 64KB 125618f6290aSShashi Mallela * and default translation table entry size to 16 bytes 125718f6290aSShashi Mallela */ 125818f6290aSShashi Mallela s->baser[0] = FIELD_DP64(s->baser[0], GITS_BASER, TYPE, 125918f6290aSShashi Mallela GITS_BASER_TYPE_DEVICE); 126018f6290aSShashi Mallela s->baser[0] = FIELD_DP64(s->baser[0], GITS_BASER, PAGESIZE, 126118f6290aSShashi Mallela GITS_BASER_PAGESIZE_64K); 126218f6290aSShashi Mallela s->baser[0] = FIELD_DP64(s->baser[0], GITS_BASER, ENTRYSIZE, 126318f6290aSShashi Mallela GITS_DTE_SIZE - 1); 126418f6290aSShashi Mallela 126518f6290aSShashi Mallela s->baser[1] = FIELD_DP64(s->baser[1], GITS_BASER, TYPE, 126618f6290aSShashi Mallela GITS_BASER_TYPE_COLLECTION); 126718f6290aSShashi Mallela s->baser[1] = FIELD_DP64(s->baser[1], GITS_BASER, PAGESIZE, 126818f6290aSShashi Mallela GITS_BASER_PAGESIZE_64K); 126918f6290aSShashi Mallela s->baser[1] = FIELD_DP64(s->baser[1], GITS_BASER, ENTRYSIZE, 127018f6290aSShashi Mallela GITS_CTE_SIZE - 1); 127118f6290aSShashi Mallela } 127218f6290aSShashi Mallela 12731b08e436SShashi Mallela static void gicv3_its_post_load(GICv3ITSState *s) 12741b08e436SShashi Mallela { 12758d2d6dd9SPeter Maydell if (s->ctlr & R_GITS_CTLR_ENABLED_MASK) { 12761b08e436SShashi Mallela extract_table_params(s); 12771b08e436SShashi Mallela extract_cmdq_params(s); 12781b08e436SShashi Mallela } 12791b08e436SShashi Mallela } 12801b08e436SShashi Mallela 128118f6290aSShashi Mallela static Property gicv3_its_props[] = { 128218f6290aSShashi Mallela DEFINE_PROP_LINK("parent-gicv3", GICv3ITSState, gicv3, "arm-gicv3", 128318f6290aSShashi Mallela GICv3State *), 128418f6290aSShashi Mallela DEFINE_PROP_END_OF_LIST(), 128518f6290aSShashi Mallela }; 128618f6290aSShashi Mallela 128718f6290aSShashi Mallela static void gicv3_its_class_init(ObjectClass *klass, void *data) 128818f6290aSShashi Mallela { 128918f6290aSShashi Mallela DeviceClass *dc = DEVICE_CLASS(klass); 129018f6290aSShashi Mallela GICv3ITSClass *ic = ARM_GICV3_ITS_CLASS(klass); 12911b08e436SShashi Mallela GICv3ITSCommonClass *icc = ARM_GICV3_ITS_COMMON_CLASS(klass); 129218f6290aSShashi Mallela 129318f6290aSShashi Mallela dc->realize = gicv3_arm_its_realize; 129418f6290aSShashi Mallela device_class_set_props(dc, gicv3_its_props); 129518f6290aSShashi Mallela device_class_set_parent_reset(dc, gicv3_its_reset, &ic->parent_reset); 12961b08e436SShashi Mallela icc->post_load = gicv3_its_post_load; 129718f6290aSShashi Mallela } 129818f6290aSShashi Mallela 129918f6290aSShashi Mallela static const TypeInfo gicv3_its_info = { 130018f6290aSShashi Mallela .name = TYPE_ARM_GICV3_ITS, 130118f6290aSShashi Mallela .parent = TYPE_ARM_GICV3_ITS_COMMON, 130218f6290aSShashi Mallela .instance_size = sizeof(GICv3ITSState), 130318f6290aSShashi Mallela .class_init = gicv3_its_class_init, 130418f6290aSShashi Mallela .class_size = sizeof(GICv3ITSClass), 130518f6290aSShashi Mallela }; 130618f6290aSShashi Mallela 130718f6290aSShashi Mallela static void gicv3_its_register_types(void) 130818f6290aSShashi Mallela { 130918f6290aSShashi Mallela type_register_static(&gicv3_its_info); 131018f6290aSShashi Mallela } 131118f6290aSShashi Mallela 131218f6290aSShashi Mallela type_init(gicv3_its_register_types) 1313