134e65944SIsaku Yamahata /* 234e65944SIsaku Yamahata * pcie_aer.c 334e65944SIsaku Yamahata * 434e65944SIsaku Yamahata * Copyright (c) 2010 Isaku Yamahata <yamahata at valinux co jp> 534e65944SIsaku Yamahata * VA Linux Systems Japan K.K. 634e65944SIsaku Yamahata * 734e65944SIsaku Yamahata * This program is free software; you can redistribute it and/or modify 834e65944SIsaku Yamahata * it under the terms of the GNU General Public License as published by 934e65944SIsaku Yamahata * the Free Software Foundation; either version 2 of the License, or 1034e65944SIsaku Yamahata * (at your option) any later version. 1134e65944SIsaku Yamahata * 1234e65944SIsaku Yamahata * This program is distributed in the hope that it will be useful, 1334e65944SIsaku Yamahata * but WITHOUT ANY WARRANTY; without even the implied warranty of 1434e65944SIsaku Yamahata * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 1534e65944SIsaku Yamahata * GNU General Public License for more details. 1634e65944SIsaku Yamahata * 1734e65944SIsaku Yamahata * You should have received a copy of the GNU General Public License along 1834e65944SIsaku Yamahata * with this program; if not, see <http://www.gnu.org/licenses/>. 1934e65944SIsaku Yamahata */ 2034e65944SIsaku Yamahata 2197d5408fSPeter Maydell #include "qemu/osdep.h" 229c17d615SPaolo Bonzini #include "sysemu/sysemu.h" 237b1b5d19SPaolo Bonzini #include "qapi/qmp/types.h" 24*c7eb39cbSEric Blake #include "qapi/qmp/qjson.h" 2583c9089eSPaolo Bonzini #include "monitor/monitor.h" 26c759b24fSMichael S. Tsirkin #include "hw/pci/pci_bridge.h" 27c759b24fSMichael S. Tsirkin #include "hw/pci/pcie.h" 28c759b24fSMichael S. Tsirkin #include "hw/pci/msix.h" 29c759b24fSMichael S. Tsirkin #include "hw/pci/msi.h" 3006aac7bdSMichael S. Tsirkin #include "hw/pci/pci_bus.h" 31c759b24fSMichael S. Tsirkin #include "hw/pci/pcie_regs.h" 3234e65944SIsaku Yamahata 3334e65944SIsaku Yamahata //#define DEBUG_PCIE 3434e65944SIsaku Yamahata #ifdef DEBUG_PCIE 3534e65944SIsaku Yamahata # define PCIE_DPRINTF(fmt, ...) \ 3634e65944SIsaku Yamahata fprintf(stderr, "%s:%d " fmt, __func__, __LINE__, ## __VA_ARGS__) 3734e65944SIsaku Yamahata #else 3834e65944SIsaku Yamahata # define PCIE_DPRINTF(fmt, ...) do {} while (0) 3934e65944SIsaku Yamahata #endif 4034e65944SIsaku Yamahata #define PCIE_DEV_PRINTF(dev, fmt, ...) \ 4134e65944SIsaku Yamahata PCIE_DPRINTF("%s:%x "fmt, (dev)->name, (dev)->devfn, ## __VA_ARGS__) 4234e65944SIsaku Yamahata 4381486b55SJan Kiszka #define PCI_ERR_SRC_COR_OFFS 0 4481486b55SJan Kiszka #define PCI_ERR_SRC_UNCOR_OFFS 2 4581486b55SJan Kiszka 4634e65944SIsaku Yamahata /* From 6.2.7 Error Listing and Rules. Table 6-2, 6-3 and 6-4 */ 4734e65944SIsaku Yamahata static uint32_t pcie_aer_uncor_default_severity(uint32_t status) 4834e65944SIsaku Yamahata { 4934e65944SIsaku Yamahata switch (status) { 5034e65944SIsaku Yamahata case PCI_ERR_UNC_INTN: 5134e65944SIsaku Yamahata case PCI_ERR_UNC_DLP: 5234e65944SIsaku Yamahata case PCI_ERR_UNC_SDN: 5334e65944SIsaku Yamahata case PCI_ERR_UNC_RX_OVER: 5434e65944SIsaku Yamahata case PCI_ERR_UNC_FCP: 5534e65944SIsaku Yamahata case PCI_ERR_UNC_MALF_TLP: 5634e65944SIsaku Yamahata return PCI_ERR_ROOT_CMD_FATAL_EN; 5734e65944SIsaku Yamahata case PCI_ERR_UNC_POISON_TLP: 5834e65944SIsaku Yamahata case PCI_ERR_UNC_ECRC: 5934e65944SIsaku Yamahata case PCI_ERR_UNC_UNSUP: 6034e65944SIsaku Yamahata case PCI_ERR_UNC_COMP_TIME: 6134e65944SIsaku Yamahata case PCI_ERR_UNC_COMP_ABORT: 6234e65944SIsaku Yamahata case PCI_ERR_UNC_UNX_COMP: 6334e65944SIsaku Yamahata case PCI_ERR_UNC_ACSV: 6434e65944SIsaku Yamahata case PCI_ERR_UNC_MCBTLP: 6534e65944SIsaku Yamahata case PCI_ERR_UNC_ATOP_EBLOCKED: 6634e65944SIsaku Yamahata case PCI_ERR_UNC_TLP_PRF_BLOCKED: 6734e65944SIsaku Yamahata return PCI_ERR_ROOT_CMD_NONFATAL_EN; 6834e65944SIsaku Yamahata default: 6934e65944SIsaku Yamahata abort(); 7034e65944SIsaku Yamahata break; 7134e65944SIsaku Yamahata } 7234e65944SIsaku Yamahata return PCI_ERR_ROOT_CMD_FATAL_EN; 7334e65944SIsaku Yamahata } 7434e65944SIsaku Yamahata 7534e65944SIsaku Yamahata static int aer_log_add_err(PCIEAERLog *aer_log, const PCIEAERErr *err) 7634e65944SIsaku Yamahata { 7734e65944SIsaku Yamahata if (aer_log->log_num == aer_log->log_max) { 7834e65944SIsaku Yamahata return -1; 7934e65944SIsaku Yamahata } 8034e65944SIsaku Yamahata memcpy(&aer_log->log[aer_log->log_num], err, sizeof *err); 8134e65944SIsaku Yamahata aer_log->log_num++; 8234e65944SIsaku Yamahata return 0; 8334e65944SIsaku Yamahata } 8434e65944SIsaku Yamahata 8534e65944SIsaku Yamahata static void aer_log_del_err(PCIEAERLog *aer_log, PCIEAERErr *err) 8634e65944SIsaku Yamahata { 8734e65944SIsaku Yamahata assert(aer_log->log_num); 8834e65944SIsaku Yamahata *err = aer_log->log[0]; 8934e65944SIsaku Yamahata aer_log->log_num--; 9034e65944SIsaku Yamahata memmove(&aer_log->log[0], &aer_log->log[1], 9134e65944SIsaku Yamahata aer_log->log_num * sizeof *err); 9234e65944SIsaku Yamahata } 9334e65944SIsaku Yamahata 9434e65944SIsaku Yamahata static void aer_log_clear_all_err(PCIEAERLog *aer_log) 9534e65944SIsaku Yamahata { 9634e65944SIsaku Yamahata aer_log->log_num = 0; 9734e65944SIsaku Yamahata } 9834e65944SIsaku Yamahata 998d86ada2SChen Fan int pcie_aer_init(PCIDevice *dev, uint16_t offset, uint16_t size) 10034e65944SIsaku Yamahata { 10134e65944SIsaku Yamahata PCIExpressDevice *exp; 10234e65944SIsaku Yamahata 10334e65944SIsaku Yamahata pcie_add_capability(dev, PCI_EXT_CAP_ID_ERR, PCI_ERR_VER, 1048d86ada2SChen Fan offset, size); 10534e65944SIsaku Yamahata exp = &dev->exp; 10634e65944SIsaku Yamahata exp->aer_cap = offset; 10734e65944SIsaku Yamahata 10834e65944SIsaku Yamahata /* log_max is property */ 10934e65944SIsaku Yamahata if (dev->exp.aer_log.log_max == PCIE_AER_LOG_MAX_UNSET) { 11034e65944SIsaku Yamahata dev->exp.aer_log.log_max = PCIE_AER_LOG_MAX_DEFAULT; 11134e65944SIsaku Yamahata } 11234e65944SIsaku Yamahata /* clip down the value to avoid unreasobale memory usage */ 11334e65944SIsaku Yamahata if (dev->exp.aer_log.log_max > PCIE_AER_LOG_MAX_LIMIT) { 11434e65944SIsaku Yamahata return -EINVAL; 11534e65944SIsaku Yamahata } 1167267c094SAnthony Liguori dev->exp.aer_log.log = g_malloc0(sizeof dev->exp.aer_log.log[0] * 11734e65944SIsaku Yamahata dev->exp.aer_log.log_max); 11834e65944SIsaku Yamahata 11934e65944SIsaku Yamahata pci_set_long(dev->w1cmask + offset + PCI_ERR_UNCOR_STATUS, 12034e65944SIsaku Yamahata PCI_ERR_UNC_SUPPORTED); 12134e65944SIsaku Yamahata 12234e65944SIsaku Yamahata pci_set_long(dev->config + offset + PCI_ERR_UNCOR_SEVER, 12334e65944SIsaku Yamahata PCI_ERR_UNC_SEVERITY_DEFAULT); 12434e65944SIsaku Yamahata pci_set_long(dev->wmask + offset + PCI_ERR_UNCOR_SEVER, 12534e65944SIsaku Yamahata PCI_ERR_UNC_SUPPORTED); 12634e65944SIsaku Yamahata 12734e65944SIsaku Yamahata pci_long_test_and_set_mask(dev->w1cmask + offset + PCI_ERR_COR_STATUS, 128310e91f7SChen Fan PCI_ERR_COR_SUPPORTED); 12934e65944SIsaku Yamahata 13034e65944SIsaku Yamahata pci_set_long(dev->config + offset + PCI_ERR_COR_MASK, 13134e65944SIsaku Yamahata PCI_ERR_COR_MASK_DEFAULT); 13234e65944SIsaku Yamahata pci_set_long(dev->wmask + offset + PCI_ERR_COR_MASK, 13334e65944SIsaku Yamahata PCI_ERR_COR_SUPPORTED); 13434e65944SIsaku Yamahata 13534e65944SIsaku Yamahata /* capabilities and control. multiple header logging is supported */ 13634e65944SIsaku Yamahata if (dev->exp.aer_log.log_max > 0) { 13734e65944SIsaku Yamahata pci_set_long(dev->config + offset + PCI_ERR_CAP, 13834e65944SIsaku Yamahata PCI_ERR_CAP_ECRC_GENC | PCI_ERR_CAP_ECRC_CHKC | 13934e65944SIsaku Yamahata PCI_ERR_CAP_MHRC); 14034e65944SIsaku Yamahata pci_set_long(dev->wmask + offset + PCI_ERR_CAP, 14134e65944SIsaku Yamahata PCI_ERR_CAP_ECRC_GENE | PCI_ERR_CAP_ECRC_CHKE | 14234e65944SIsaku Yamahata PCI_ERR_CAP_MHRE); 14334e65944SIsaku Yamahata } else { 14434e65944SIsaku Yamahata pci_set_long(dev->config + offset + PCI_ERR_CAP, 14534e65944SIsaku Yamahata PCI_ERR_CAP_ECRC_GENC | PCI_ERR_CAP_ECRC_CHKC); 14634e65944SIsaku Yamahata pci_set_long(dev->wmask + offset + PCI_ERR_CAP, 14734e65944SIsaku Yamahata PCI_ERR_CAP_ECRC_GENE | PCI_ERR_CAP_ECRC_CHKE); 14834e65944SIsaku Yamahata } 14934e65944SIsaku Yamahata 15034e65944SIsaku Yamahata switch (pcie_cap_get_type(dev)) { 15134e65944SIsaku Yamahata case PCI_EXP_TYPE_ROOT_PORT: 15234e65944SIsaku Yamahata /* this case will be set by pcie_aer_root_init() */ 15334e65944SIsaku Yamahata /* fallthrough */ 15434e65944SIsaku Yamahata case PCI_EXP_TYPE_DOWNSTREAM: 15534e65944SIsaku Yamahata case PCI_EXP_TYPE_UPSTREAM: 15634e65944SIsaku Yamahata pci_word_test_and_set_mask(dev->wmask + PCI_BRIDGE_CONTROL, 15734e65944SIsaku Yamahata PCI_BRIDGE_CTL_SERR); 15834e65944SIsaku Yamahata pci_long_test_and_set_mask(dev->w1cmask + PCI_STATUS, 15934e65944SIsaku Yamahata PCI_SEC_STATUS_RCV_SYSTEM_ERROR); 16034e65944SIsaku Yamahata break; 16134e65944SIsaku Yamahata default: 16234e65944SIsaku Yamahata /* nothing */ 16334e65944SIsaku Yamahata break; 16434e65944SIsaku Yamahata } 16534e65944SIsaku Yamahata return 0; 16634e65944SIsaku Yamahata } 16734e65944SIsaku Yamahata 16834e65944SIsaku Yamahata void pcie_aer_exit(PCIDevice *dev) 16934e65944SIsaku Yamahata { 1707267c094SAnthony Liguori g_free(dev->exp.aer_log.log); 17134e65944SIsaku Yamahata } 17234e65944SIsaku Yamahata 17334e65944SIsaku Yamahata static void pcie_aer_update_uncor_status(PCIDevice *dev) 17434e65944SIsaku Yamahata { 17534e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 17634e65944SIsaku Yamahata PCIEAERLog *aer_log = &dev->exp.aer_log; 17734e65944SIsaku Yamahata 17834e65944SIsaku Yamahata uint16_t i; 17934e65944SIsaku Yamahata for (i = 0; i < aer_log->log_num; i++) { 18034e65944SIsaku Yamahata pci_long_test_and_set_mask(aer_cap + PCI_ERR_UNCOR_STATUS, 18134e65944SIsaku Yamahata dev->exp.aer_log.log[i].status); 18234e65944SIsaku Yamahata } 18334e65944SIsaku Yamahata } 18434e65944SIsaku Yamahata 18534e65944SIsaku Yamahata /* 18634e65944SIsaku Yamahata * return value: 187247c97f3SMichael S. Tsirkin * true: error message needs to be sent up 18834e65944SIsaku Yamahata * false: error message is masked 18934e65944SIsaku Yamahata * 19034e65944SIsaku Yamahata * 6.2.6 Error Message Control 19134e65944SIsaku Yamahata * Figure 6-3 19234e65944SIsaku Yamahata * all pci express devices part 19334e65944SIsaku Yamahata */ 19434e65944SIsaku Yamahata static bool 19534e65944SIsaku Yamahata pcie_aer_msg_alldev(PCIDevice *dev, const PCIEAERMsg *msg) 19634e65944SIsaku Yamahata { 19734e65944SIsaku Yamahata if (!(pcie_aer_msg_is_uncor(msg) && 19834e65944SIsaku Yamahata (pci_get_word(dev->config + PCI_COMMAND) & PCI_COMMAND_SERR))) { 19934e65944SIsaku Yamahata return false; 20034e65944SIsaku Yamahata } 20134e65944SIsaku Yamahata 20234e65944SIsaku Yamahata /* Signaled System Error 20334e65944SIsaku Yamahata * 20434e65944SIsaku Yamahata * 7.5.1.1 Command register 20534e65944SIsaku Yamahata * Bit 8 SERR# Enable 20634e65944SIsaku Yamahata * 20734e65944SIsaku Yamahata * When Set, this bit enables reporting of Non-fatal and Fatal 20834e65944SIsaku Yamahata * errors detected by the Function to the Root Complex. Note that 20934e65944SIsaku Yamahata * errors are reported if enabled either through this bit or through 21034e65944SIsaku Yamahata * the PCI Express specific bits in the Device Control register (see 21134e65944SIsaku Yamahata * Section 7.8.4). 21234e65944SIsaku Yamahata */ 21334e65944SIsaku Yamahata pci_word_test_and_set_mask(dev->config + PCI_STATUS, 21434e65944SIsaku Yamahata PCI_STATUS_SIG_SYSTEM_ERROR); 21534e65944SIsaku Yamahata 21634e65944SIsaku Yamahata if (!(msg->severity & 21734e65944SIsaku Yamahata pci_get_word(dev->config + dev->exp.exp_cap + PCI_EXP_DEVCTL))) { 21834e65944SIsaku Yamahata return false; 21934e65944SIsaku Yamahata } 22034e65944SIsaku Yamahata 22134e65944SIsaku Yamahata /* send up error message */ 222247c97f3SMichael S. Tsirkin return true; 223247c97f3SMichael S. Tsirkin } 224247c97f3SMichael S. Tsirkin 22534e65944SIsaku Yamahata /* 22634e65944SIsaku Yamahata * return value: 22734e65944SIsaku Yamahata * true: error message is sent up 22834e65944SIsaku Yamahata * false: error message is masked 22934e65944SIsaku Yamahata * 23034e65944SIsaku Yamahata * 6.2.6 Error Message Control 23134e65944SIsaku Yamahata * Figure 6-3 23234e65944SIsaku Yamahata * virtual pci bridge part 23334e65944SIsaku Yamahata */ 23434e65944SIsaku Yamahata static bool pcie_aer_msg_vbridge(PCIDevice *dev, const PCIEAERMsg *msg) 23534e65944SIsaku Yamahata { 23634e65944SIsaku Yamahata uint16_t bridge_control = pci_get_word(dev->config + PCI_BRIDGE_CONTROL); 23734e65944SIsaku Yamahata 23834e65944SIsaku Yamahata if (pcie_aer_msg_is_uncor(msg)) { 23934e65944SIsaku Yamahata /* Received System Error */ 24034e65944SIsaku Yamahata pci_word_test_and_set_mask(dev->config + PCI_SEC_STATUS, 24134e65944SIsaku Yamahata PCI_SEC_STATUS_RCV_SYSTEM_ERROR); 24234e65944SIsaku Yamahata } 24334e65944SIsaku Yamahata 24434e65944SIsaku Yamahata if (!(bridge_control & PCI_BRIDGE_CTL_SERR)) { 24534e65944SIsaku Yamahata return false; 24634e65944SIsaku Yamahata } 24734e65944SIsaku Yamahata return true; 24834e65944SIsaku Yamahata } 24934e65944SIsaku Yamahata 25034e65944SIsaku Yamahata void pcie_aer_root_set_vector(PCIDevice *dev, unsigned int vector) 25134e65944SIsaku Yamahata { 25234e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 25334e65944SIsaku Yamahata assert(vector < PCI_ERR_ROOT_IRQ_MAX); 25434e65944SIsaku Yamahata pci_long_test_and_clear_mask(aer_cap + PCI_ERR_ROOT_STATUS, 25534e65944SIsaku Yamahata PCI_ERR_ROOT_IRQ); 25634e65944SIsaku Yamahata pci_long_test_and_set_mask(aer_cap + PCI_ERR_ROOT_STATUS, 25734e65944SIsaku Yamahata vector << PCI_ERR_ROOT_IRQ_SHIFT); 25834e65944SIsaku Yamahata } 25934e65944SIsaku Yamahata 26034e65944SIsaku Yamahata static unsigned int pcie_aer_root_get_vector(PCIDevice *dev) 26134e65944SIsaku Yamahata { 26234e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 26334e65944SIsaku Yamahata uint32_t root_status = pci_get_long(aer_cap + PCI_ERR_ROOT_STATUS); 26434e65944SIsaku Yamahata return (root_status & PCI_ERR_ROOT_IRQ) >> PCI_ERR_ROOT_IRQ_SHIFT; 26534e65944SIsaku Yamahata } 26634e65944SIsaku Yamahata 267c3f33667SMichael S. Tsirkin /* Given a status register, get corresponding bits in the command register */ 268c3f33667SMichael S. Tsirkin static uint32_t pcie_aer_status_to_cmd(uint32_t status) 269c3f33667SMichael S. Tsirkin { 270c3f33667SMichael S. Tsirkin uint32_t cmd = 0; 271c3f33667SMichael S. Tsirkin if (status & PCI_ERR_ROOT_COR_RCV) { 272c3f33667SMichael S. Tsirkin cmd |= PCI_ERR_ROOT_CMD_COR_EN; 273c3f33667SMichael S. Tsirkin } 274c3f33667SMichael S. Tsirkin if (status & PCI_ERR_ROOT_NONFATAL_RCV) { 275c3f33667SMichael S. Tsirkin cmd |= PCI_ERR_ROOT_CMD_NONFATAL_EN; 276c3f33667SMichael S. Tsirkin } 277c3f33667SMichael S. Tsirkin if (status & PCI_ERR_ROOT_FATAL_RCV) { 278c3f33667SMichael S. Tsirkin cmd |= PCI_ERR_ROOT_CMD_FATAL_EN; 279c3f33667SMichael S. Tsirkin } 280c3f33667SMichael S. Tsirkin return cmd; 281c3f33667SMichael S. Tsirkin } 282c3f33667SMichael S. Tsirkin 283513691b7SMichael S. Tsirkin static void pcie_aer_root_notify(PCIDevice *dev) 284513691b7SMichael S. Tsirkin { 285513691b7SMichael S. Tsirkin if (msix_enabled(dev)) { 286513691b7SMichael S. Tsirkin msix_notify(dev, pcie_aer_root_get_vector(dev)); 287513691b7SMichael S. Tsirkin } else if (msi_enabled(dev)) { 288513691b7SMichael S. Tsirkin msi_notify(dev, pcie_aer_root_get_vector(dev)); 289513691b7SMichael S. Tsirkin } else { 2905a03e708SMarcel Apfelbaum pci_irq_assert(dev); 291513691b7SMichael S. Tsirkin } 292513691b7SMichael S. Tsirkin } 293513691b7SMichael S. Tsirkin 29434e65944SIsaku Yamahata /* 29534e65944SIsaku Yamahata * 6.2.6 Error Message Control 29634e65944SIsaku Yamahata * Figure 6-3 29734e65944SIsaku Yamahata * root port part 29834e65944SIsaku Yamahata */ 2995f47c187SMichael S. Tsirkin static void pcie_aer_msg_root_port(PCIDevice *dev, const PCIEAERMsg *msg) 30034e65944SIsaku Yamahata { 30134e65944SIsaku Yamahata uint16_t cmd; 30234e65944SIsaku Yamahata uint8_t *aer_cap; 30334e65944SIsaku Yamahata uint32_t root_cmd; 304c3f33667SMichael S. Tsirkin uint32_t root_status, prev_status; 30534e65944SIsaku Yamahata 30634e65944SIsaku Yamahata cmd = pci_get_word(dev->config + PCI_COMMAND); 30734e65944SIsaku Yamahata aer_cap = dev->config + dev->exp.aer_cap; 30834e65944SIsaku Yamahata root_cmd = pci_get_long(aer_cap + PCI_ERR_ROOT_COMMAND); 309c3f33667SMichael S. Tsirkin prev_status = root_status = pci_get_long(aer_cap + PCI_ERR_ROOT_STATUS); 31034e65944SIsaku Yamahata 31134e65944SIsaku Yamahata if (cmd & PCI_COMMAND_SERR) { 31234e65944SIsaku Yamahata /* System Error. 31334e65944SIsaku Yamahata * 31434e65944SIsaku Yamahata * The way to report System Error is platform specific and 31534e65944SIsaku Yamahata * it isn't implemented in qemu right now. 31634e65944SIsaku Yamahata * So just discard the error for now. 31734e65944SIsaku Yamahata * OS which cares of aer would receive errors via 31834e65944SIsaku Yamahata * native aer mechanims, so this wouldn't matter. 31934e65944SIsaku Yamahata */ 32034e65944SIsaku Yamahata } 32134e65944SIsaku Yamahata 32234e65944SIsaku Yamahata /* Errro Message Received: Root Error Status register */ 32334e65944SIsaku Yamahata switch (msg->severity) { 32434e65944SIsaku Yamahata case PCI_ERR_ROOT_CMD_COR_EN: 32534e65944SIsaku Yamahata if (root_status & PCI_ERR_ROOT_COR_RCV) { 32634e65944SIsaku Yamahata root_status |= PCI_ERR_ROOT_MULTI_COR_RCV; 32734e65944SIsaku Yamahata } else { 32881486b55SJan Kiszka pci_set_word(aer_cap + PCI_ERR_ROOT_ERR_SRC + PCI_ERR_SRC_COR_OFFS, 32981486b55SJan Kiszka msg->source_id); 33034e65944SIsaku Yamahata } 33134e65944SIsaku Yamahata root_status |= PCI_ERR_ROOT_COR_RCV; 33234e65944SIsaku Yamahata break; 33334e65944SIsaku Yamahata case PCI_ERR_ROOT_CMD_NONFATAL_EN: 33434e65944SIsaku Yamahata root_status |= PCI_ERR_ROOT_NONFATAL_RCV; 33534e65944SIsaku Yamahata break; 33634e65944SIsaku Yamahata case PCI_ERR_ROOT_CMD_FATAL_EN: 33734e65944SIsaku Yamahata if (!(root_status & PCI_ERR_ROOT_UNCOR_RCV)) { 33834e65944SIsaku Yamahata root_status |= PCI_ERR_ROOT_FIRST_FATAL; 33934e65944SIsaku Yamahata } 34034e65944SIsaku Yamahata root_status |= PCI_ERR_ROOT_FATAL_RCV; 34134e65944SIsaku Yamahata break; 34234e65944SIsaku Yamahata default: 34334e65944SIsaku Yamahata abort(); 34434e65944SIsaku Yamahata break; 34534e65944SIsaku Yamahata } 34634e65944SIsaku Yamahata if (pcie_aer_msg_is_uncor(msg)) { 34734e65944SIsaku Yamahata if (root_status & PCI_ERR_ROOT_UNCOR_RCV) { 34834e65944SIsaku Yamahata root_status |= PCI_ERR_ROOT_MULTI_UNCOR_RCV; 34934e65944SIsaku Yamahata } else { 35081486b55SJan Kiszka pci_set_word(aer_cap + PCI_ERR_ROOT_ERR_SRC + 35181486b55SJan Kiszka PCI_ERR_SRC_UNCOR_OFFS, msg->source_id); 35234e65944SIsaku Yamahata } 35334e65944SIsaku Yamahata root_status |= PCI_ERR_ROOT_UNCOR_RCV; 35434e65944SIsaku Yamahata } 35534e65944SIsaku Yamahata pci_set_long(aer_cap + PCI_ERR_ROOT_STATUS, root_status); 35634e65944SIsaku Yamahata 35734e65944SIsaku Yamahata /* 6.2.4.1.2 Interrupt Generation */ 358c3f33667SMichael S. Tsirkin /* All the above did was set some bits in the status register. 359c3f33667SMichael S. Tsirkin * Specifically these that match message severity. 360c3f33667SMichael S. Tsirkin * The below code relies on this fact. */ 361c3f33667SMichael S. Tsirkin if (!(root_cmd & msg->severity) || 362c3f33667SMichael S. Tsirkin (pcie_aer_status_to_cmd(prev_status) & root_cmd)) { 363c3f33667SMichael S. Tsirkin /* Condition is not being set or was already true so nothing to do. */ 3645f47c187SMichael S. Tsirkin return; 365c3f33667SMichael S. Tsirkin } 366c3f33667SMichael S. Tsirkin 367513691b7SMichael S. Tsirkin pcie_aer_root_notify(dev); 36834e65944SIsaku Yamahata } 36934e65944SIsaku Yamahata 37034e65944SIsaku Yamahata /* 37134e65944SIsaku Yamahata * 6.2.6 Error Message Control Figure 6-3 372247c97f3SMichael S. Tsirkin * 373d33d9156SMichael S. Tsirkin * Walk up the bus tree from the device, propagate the error message. 37434e65944SIsaku Yamahata */ 37540f8f0c3SChen Fan void pcie_aer_msg(PCIDevice *dev, const PCIEAERMsg *msg) 376247c97f3SMichael S. Tsirkin { 377d33d9156SMichael S. Tsirkin uint8_t type; 378d33d9156SMichael S. Tsirkin 379247c97f3SMichael S. Tsirkin while (dev) { 380d33d9156SMichael S. Tsirkin if (!pci_is_express(dev)) { 381d33d9156SMichael S. Tsirkin /* just ignore it */ 382d33d9156SMichael S. Tsirkin /* TODO: Shouldn't we set PCI_STATUS_SIG_SYSTEM_ERROR? 383d33d9156SMichael S. Tsirkin * Consider e.g. a PCI bridge above a PCI Express device. */ 384247c97f3SMichael S. Tsirkin return; 385247c97f3SMichael S. Tsirkin } 386d33d9156SMichael S. Tsirkin 387d33d9156SMichael S. Tsirkin type = pcie_cap_get_type(dev); 388d33d9156SMichael S. Tsirkin if ((type == PCI_EXP_TYPE_ROOT_PORT || 389d33d9156SMichael S. Tsirkin type == PCI_EXP_TYPE_UPSTREAM || 390d33d9156SMichael S. Tsirkin type == PCI_EXP_TYPE_DOWNSTREAM) && 391d33d9156SMichael S. Tsirkin !pcie_aer_msg_vbridge(dev, msg)) { 392d33d9156SMichael S. Tsirkin return; 393d33d9156SMichael S. Tsirkin } 394d33d9156SMichael S. Tsirkin if (!pcie_aer_msg_alldev(dev, msg)) { 395d33d9156SMichael S. Tsirkin return; 396d33d9156SMichael S. Tsirkin } 397d33d9156SMichael S. Tsirkin if (type == PCI_EXP_TYPE_ROOT_PORT) { 398d33d9156SMichael S. Tsirkin pcie_aer_msg_root_port(dev, msg); 399d33d9156SMichael S. Tsirkin /* Root port can notify system itself, 400d33d9156SMichael S. Tsirkin or send the error message to root complex event collector. */ 401d33d9156SMichael S. Tsirkin /* 402d33d9156SMichael S. Tsirkin * if root port is associated with an event collector, 403d33d9156SMichael S. Tsirkin * return the root complex event collector here. 404d33d9156SMichael S. Tsirkin * For now root complex event collector isn't supported. 405d33d9156SMichael S. Tsirkin */ 406d33d9156SMichael S. Tsirkin return; 407d33d9156SMichael S. Tsirkin } 408d33d9156SMichael S. Tsirkin dev = pci_bridge_get_device(dev->bus); 409247c97f3SMichael S. Tsirkin } 41034e65944SIsaku Yamahata } 41134e65944SIsaku Yamahata 41234e65944SIsaku Yamahata static void pcie_aer_update_log(PCIDevice *dev, const PCIEAERErr *err) 41334e65944SIsaku Yamahata { 41434e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 415786a4ea8SStefan Hajnoczi uint8_t first_bit = ctz32(err->status); 41634e65944SIsaku Yamahata uint32_t errcap = pci_get_long(aer_cap + PCI_ERR_CAP); 41734e65944SIsaku Yamahata int i; 41834e65944SIsaku Yamahata 41934e65944SIsaku Yamahata assert(err->status); 42074d63b65SIsaku Yamahata assert(!(err->status & (err->status - 1))); 42134e65944SIsaku Yamahata 42234e65944SIsaku Yamahata errcap &= ~(PCI_ERR_CAP_FEP_MASK | PCI_ERR_CAP_TLP); 42334e65944SIsaku Yamahata errcap |= PCI_ERR_CAP_FEP(first_bit); 42434e65944SIsaku Yamahata 42534e65944SIsaku Yamahata if (err->flags & PCIE_AER_ERR_HEADER_VALID) { 42634e65944SIsaku Yamahata for (i = 0; i < ARRAY_SIZE(err->header); ++i) { 42734e65944SIsaku Yamahata /* 7.10.8 Header Log Register */ 42834e65944SIsaku Yamahata uint8_t *header_log = 42934e65944SIsaku Yamahata aer_cap + PCI_ERR_HEADER_LOG + i * sizeof err->header[0]; 4306bd194abSPeter Maydell stl_be_p(header_log, err->header[i]); 43134e65944SIsaku Yamahata } 43234e65944SIsaku Yamahata } else { 43334e65944SIsaku Yamahata assert(!(err->flags & PCIE_AER_ERR_TLP_PREFIX_PRESENT)); 43434e65944SIsaku Yamahata memset(aer_cap + PCI_ERR_HEADER_LOG, 0, PCI_ERR_HEADER_LOG_SIZE); 43534e65944SIsaku Yamahata } 43634e65944SIsaku Yamahata 43734e65944SIsaku Yamahata if ((err->flags & PCIE_AER_ERR_TLP_PREFIX_PRESENT) && 43877a3c1d7SChen Fan (pci_get_long(dev->config + dev->exp.exp_cap + PCI_EXP_DEVCAP2) & 43934e65944SIsaku Yamahata PCI_EXP_DEVCAP2_EETLPP)) { 44034e65944SIsaku Yamahata for (i = 0; i < ARRAY_SIZE(err->prefix); ++i) { 44134e65944SIsaku Yamahata /* 7.10.12 tlp prefix log register */ 44234e65944SIsaku Yamahata uint8_t *prefix_log = 44334e65944SIsaku Yamahata aer_cap + PCI_ERR_TLP_PREFIX_LOG + i * sizeof err->prefix[0]; 4446bd194abSPeter Maydell stl_be_p(prefix_log, err->prefix[i]); 44534e65944SIsaku Yamahata } 44634e65944SIsaku Yamahata errcap |= PCI_ERR_CAP_TLP; 44734e65944SIsaku Yamahata } else { 44834e65944SIsaku Yamahata memset(aer_cap + PCI_ERR_TLP_PREFIX_LOG, 0, 44934e65944SIsaku Yamahata PCI_ERR_TLP_PREFIX_LOG_SIZE); 45034e65944SIsaku Yamahata } 45134e65944SIsaku Yamahata pci_set_long(aer_cap + PCI_ERR_CAP, errcap); 45234e65944SIsaku Yamahata } 45334e65944SIsaku Yamahata 45434e65944SIsaku Yamahata static void pcie_aer_clear_log(PCIDevice *dev) 45534e65944SIsaku Yamahata { 45634e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 45734e65944SIsaku Yamahata 45834e65944SIsaku Yamahata pci_long_test_and_clear_mask(aer_cap + PCI_ERR_CAP, 45934e65944SIsaku Yamahata PCI_ERR_CAP_FEP_MASK | PCI_ERR_CAP_TLP); 46034e65944SIsaku Yamahata 46134e65944SIsaku Yamahata memset(aer_cap + PCI_ERR_HEADER_LOG, 0, PCI_ERR_HEADER_LOG_SIZE); 46234e65944SIsaku Yamahata memset(aer_cap + PCI_ERR_TLP_PREFIX_LOG, 0, PCI_ERR_TLP_PREFIX_LOG_SIZE); 46334e65944SIsaku Yamahata } 46434e65944SIsaku Yamahata 46534e65944SIsaku Yamahata static void pcie_aer_clear_error(PCIDevice *dev) 46634e65944SIsaku Yamahata { 46734e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 46834e65944SIsaku Yamahata uint32_t errcap = pci_get_long(aer_cap + PCI_ERR_CAP); 46934e65944SIsaku Yamahata PCIEAERLog *aer_log = &dev->exp.aer_log; 47034e65944SIsaku Yamahata PCIEAERErr err; 47134e65944SIsaku Yamahata 47234e65944SIsaku Yamahata if (!(errcap & PCI_ERR_CAP_MHRE) || !aer_log->log_num) { 47334e65944SIsaku Yamahata pcie_aer_clear_log(dev); 47434e65944SIsaku Yamahata return; 47534e65944SIsaku Yamahata } 47634e65944SIsaku Yamahata 47734e65944SIsaku Yamahata /* 47834e65944SIsaku Yamahata * If more errors are queued, set corresponding bits in uncorrectable 47934e65944SIsaku Yamahata * error status. 48034e65944SIsaku Yamahata * We emulate uncorrectable error status register as W1CS. 48134e65944SIsaku Yamahata * So set bit in uncorrectable error status here again for multiple 48234e65944SIsaku Yamahata * error recording support. 48334e65944SIsaku Yamahata * 48434e65944SIsaku Yamahata * 6.2.4.2 Multiple Error Handling(Advanced Error Reporting Capability) 48534e65944SIsaku Yamahata */ 48634e65944SIsaku Yamahata pcie_aer_update_uncor_status(dev); 48734e65944SIsaku Yamahata 48834e65944SIsaku Yamahata aer_log_del_err(aer_log, &err); 48934e65944SIsaku Yamahata pcie_aer_update_log(dev, &err); 49034e65944SIsaku Yamahata } 49134e65944SIsaku Yamahata 49234e65944SIsaku Yamahata static int pcie_aer_record_error(PCIDevice *dev, 49334e65944SIsaku Yamahata const PCIEAERErr *err) 49434e65944SIsaku Yamahata { 49534e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 49634e65944SIsaku Yamahata uint32_t errcap = pci_get_long(aer_cap + PCI_ERR_CAP); 49734e65944SIsaku Yamahata int fep = PCI_ERR_CAP_FEP(errcap); 49834e65944SIsaku Yamahata 49934e65944SIsaku Yamahata assert(err->status); 50074d63b65SIsaku Yamahata assert(!(err->status & (err->status - 1))); 50134e65944SIsaku Yamahata 50234e65944SIsaku Yamahata if (errcap & PCI_ERR_CAP_MHRE && 50334e65944SIsaku Yamahata (pci_get_long(aer_cap + PCI_ERR_UNCOR_STATUS) & (1U << fep))) { 50434e65944SIsaku Yamahata /* Not first error. queue error */ 50534e65944SIsaku Yamahata if (aer_log_add_err(&dev->exp.aer_log, err) < 0) { 50634e65944SIsaku Yamahata /* overflow */ 50734e65944SIsaku Yamahata return -1; 50834e65944SIsaku Yamahata } 50934e65944SIsaku Yamahata return 0; 51034e65944SIsaku Yamahata } 51134e65944SIsaku Yamahata 51234e65944SIsaku Yamahata pcie_aer_update_log(dev, err); 51334e65944SIsaku Yamahata return 0; 51434e65944SIsaku Yamahata } 51534e65944SIsaku Yamahata 51634e65944SIsaku Yamahata typedef struct PCIEAERInject { 51734e65944SIsaku Yamahata PCIDevice *dev; 51834e65944SIsaku Yamahata uint8_t *aer_cap; 51934e65944SIsaku Yamahata const PCIEAERErr *err; 52034e65944SIsaku Yamahata uint16_t devctl; 52134e65944SIsaku Yamahata uint16_t devsta; 52234e65944SIsaku Yamahata uint32_t error_status; 52334e65944SIsaku Yamahata bool unsupported_request; 52434e65944SIsaku Yamahata bool log_overflow; 52534e65944SIsaku Yamahata PCIEAERMsg msg; 52634e65944SIsaku Yamahata } PCIEAERInject; 52734e65944SIsaku Yamahata 52834e65944SIsaku Yamahata static bool pcie_aer_inject_cor_error(PCIEAERInject *inj, 52934e65944SIsaku Yamahata uint32_t uncor_status, 53034e65944SIsaku Yamahata bool is_advisory_nonfatal) 53134e65944SIsaku Yamahata { 53234e65944SIsaku Yamahata PCIDevice *dev = inj->dev; 53334e65944SIsaku Yamahata 53434e65944SIsaku Yamahata inj->devsta |= PCI_EXP_DEVSTA_CED; 53534e65944SIsaku Yamahata if (inj->unsupported_request) { 53634e65944SIsaku Yamahata inj->devsta |= PCI_EXP_DEVSTA_URD; 53734e65944SIsaku Yamahata } 53834e65944SIsaku Yamahata pci_set_word(dev->config + dev->exp.exp_cap + PCI_EXP_DEVSTA, inj->devsta); 53934e65944SIsaku Yamahata 54034e65944SIsaku Yamahata if (inj->aer_cap) { 54134e65944SIsaku Yamahata uint32_t mask; 54234e65944SIsaku Yamahata pci_long_test_and_set_mask(inj->aer_cap + PCI_ERR_COR_STATUS, 54334e65944SIsaku Yamahata inj->error_status); 54434e65944SIsaku Yamahata mask = pci_get_long(inj->aer_cap + PCI_ERR_COR_MASK); 54534e65944SIsaku Yamahata if (mask & inj->error_status) { 54634e65944SIsaku Yamahata return false; 54734e65944SIsaku Yamahata } 54834e65944SIsaku Yamahata if (is_advisory_nonfatal) { 54934e65944SIsaku Yamahata uint32_t uncor_mask = 55034e65944SIsaku Yamahata pci_get_long(inj->aer_cap + PCI_ERR_UNCOR_MASK); 55134e65944SIsaku Yamahata if (!(uncor_mask & uncor_status)) { 55234e65944SIsaku Yamahata inj->log_overflow = !!pcie_aer_record_error(dev, inj->err); 55334e65944SIsaku Yamahata } 55434e65944SIsaku Yamahata pci_long_test_and_set_mask(inj->aer_cap + PCI_ERR_UNCOR_STATUS, 55534e65944SIsaku Yamahata uncor_status); 55634e65944SIsaku Yamahata } 55734e65944SIsaku Yamahata } 55834e65944SIsaku Yamahata 55934e65944SIsaku Yamahata if (inj->unsupported_request && !(inj->devctl & PCI_EXP_DEVCTL_URRE)) { 56034e65944SIsaku Yamahata return false; 56134e65944SIsaku Yamahata } 56234e65944SIsaku Yamahata if (!(inj->devctl & PCI_EXP_DEVCTL_CERE)) { 56334e65944SIsaku Yamahata return false; 56434e65944SIsaku Yamahata } 56534e65944SIsaku Yamahata 56634e65944SIsaku Yamahata inj->msg.severity = PCI_ERR_ROOT_CMD_COR_EN; 56734e65944SIsaku Yamahata return true; 56834e65944SIsaku Yamahata } 56934e65944SIsaku Yamahata 57034e65944SIsaku Yamahata static bool pcie_aer_inject_uncor_error(PCIEAERInject *inj, bool is_fatal) 57134e65944SIsaku Yamahata { 57234e65944SIsaku Yamahata PCIDevice *dev = inj->dev; 57334e65944SIsaku Yamahata uint16_t cmd; 57434e65944SIsaku Yamahata 57534e65944SIsaku Yamahata if (is_fatal) { 57634e65944SIsaku Yamahata inj->devsta |= PCI_EXP_DEVSTA_FED; 57734e65944SIsaku Yamahata } else { 57834e65944SIsaku Yamahata inj->devsta |= PCI_EXP_DEVSTA_NFED; 57934e65944SIsaku Yamahata } 58034e65944SIsaku Yamahata if (inj->unsupported_request) { 58134e65944SIsaku Yamahata inj->devsta |= PCI_EXP_DEVSTA_URD; 58234e65944SIsaku Yamahata } 58334e65944SIsaku Yamahata pci_set_long(dev->config + dev->exp.exp_cap + PCI_EXP_DEVSTA, inj->devsta); 58434e65944SIsaku Yamahata 58534e65944SIsaku Yamahata if (inj->aer_cap) { 58634e65944SIsaku Yamahata uint32_t mask = pci_get_long(inj->aer_cap + PCI_ERR_UNCOR_MASK); 58734e65944SIsaku Yamahata if (mask & inj->error_status) { 58834e65944SIsaku Yamahata pci_long_test_and_set_mask(inj->aer_cap + PCI_ERR_UNCOR_STATUS, 58934e65944SIsaku Yamahata inj->error_status); 59034e65944SIsaku Yamahata return false; 59134e65944SIsaku Yamahata } 59234e65944SIsaku Yamahata 59334e65944SIsaku Yamahata inj->log_overflow = !!pcie_aer_record_error(dev, inj->err); 59434e65944SIsaku Yamahata pci_long_test_and_set_mask(inj->aer_cap + PCI_ERR_UNCOR_STATUS, 59534e65944SIsaku Yamahata inj->error_status); 59634e65944SIsaku Yamahata } 59734e65944SIsaku Yamahata 59834e65944SIsaku Yamahata cmd = pci_get_word(dev->config + PCI_COMMAND); 59934e65944SIsaku Yamahata if (inj->unsupported_request && 60034e65944SIsaku Yamahata !(inj->devctl & PCI_EXP_DEVCTL_URRE) && !(cmd & PCI_COMMAND_SERR)) { 60134e65944SIsaku Yamahata return false; 60234e65944SIsaku Yamahata } 60334e65944SIsaku Yamahata if (is_fatal) { 60434e65944SIsaku Yamahata if (!((cmd & PCI_COMMAND_SERR) || 60534e65944SIsaku Yamahata (inj->devctl & PCI_EXP_DEVCTL_FERE))) { 60634e65944SIsaku Yamahata return false; 60734e65944SIsaku Yamahata } 60834e65944SIsaku Yamahata inj->msg.severity = PCI_ERR_ROOT_CMD_FATAL_EN; 60934e65944SIsaku Yamahata } else { 61034e65944SIsaku Yamahata if (!((cmd & PCI_COMMAND_SERR) || 61134e65944SIsaku Yamahata (inj->devctl & PCI_EXP_DEVCTL_NFERE))) { 61234e65944SIsaku Yamahata return false; 61334e65944SIsaku Yamahata } 61434e65944SIsaku Yamahata inj->msg.severity = PCI_ERR_ROOT_CMD_NONFATAL_EN; 61534e65944SIsaku Yamahata } 61634e65944SIsaku Yamahata return true; 61734e65944SIsaku Yamahata } 61834e65944SIsaku Yamahata 61934e65944SIsaku Yamahata /* 62034e65944SIsaku Yamahata * non-Function specific error must be recorded in all functions. 62134e65944SIsaku Yamahata * It is the responsibility of the caller of this function. 622e8e3bb2fSStefan Weil * It is also caller's responsibility to determine which function should 623b01738c2SChen Fan * report the error. 62434e65944SIsaku Yamahata * 62534e65944SIsaku Yamahata * 6.2.4 Error Logging 626b01738c2SChen Fan * 6.2.5 Sequence of Device Error Signaling and Logging Operations 627ce394947SMichael S. Tsirkin * Figure 6-2: Flowchart Showing Sequence of Device Error Signaling and Logging 62834e65944SIsaku Yamahata * Operations 62934e65944SIsaku Yamahata */ 63034e65944SIsaku Yamahata int pcie_aer_inject_error(PCIDevice *dev, const PCIEAERErr *err) 63134e65944SIsaku Yamahata { 63234e65944SIsaku Yamahata uint8_t *aer_cap = NULL; 63334e65944SIsaku Yamahata uint16_t devctl = 0; 63434e65944SIsaku Yamahata uint16_t devsta = 0; 63534e65944SIsaku Yamahata uint32_t error_status = err->status; 63634e65944SIsaku Yamahata PCIEAERInject inj; 63734e65944SIsaku Yamahata 63834e65944SIsaku Yamahata if (!pci_is_express(dev)) { 63934e65944SIsaku Yamahata return -ENOSYS; 64034e65944SIsaku Yamahata } 64134e65944SIsaku Yamahata 64234e65944SIsaku Yamahata if (err->flags & PCIE_AER_ERR_IS_CORRECTABLE) { 64334e65944SIsaku Yamahata error_status &= PCI_ERR_COR_SUPPORTED; 64434e65944SIsaku Yamahata } else { 64534e65944SIsaku Yamahata error_status &= PCI_ERR_UNC_SUPPORTED; 64634e65944SIsaku Yamahata } 64734e65944SIsaku Yamahata 64834e65944SIsaku Yamahata /* invalid status bit. one and only one bit must be set */ 64934e65944SIsaku Yamahata if (!error_status || (error_status & (error_status - 1))) { 65034e65944SIsaku Yamahata return -EINVAL; 65134e65944SIsaku Yamahata } 65234e65944SIsaku Yamahata 65334e65944SIsaku Yamahata if (dev->exp.aer_cap) { 65434e65944SIsaku Yamahata uint8_t *exp_cap = dev->config + dev->exp.exp_cap; 65534e65944SIsaku Yamahata aer_cap = dev->config + dev->exp.aer_cap; 65634e65944SIsaku Yamahata devctl = pci_get_long(exp_cap + PCI_EXP_DEVCTL); 65734e65944SIsaku Yamahata devsta = pci_get_long(exp_cap + PCI_EXP_DEVSTA); 65834e65944SIsaku Yamahata } 65934e65944SIsaku Yamahata 66034e65944SIsaku Yamahata inj.dev = dev; 66134e65944SIsaku Yamahata inj.aer_cap = aer_cap; 66234e65944SIsaku Yamahata inj.err = err; 66334e65944SIsaku Yamahata inj.devctl = devctl; 66434e65944SIsaku Yamahata inj.devsta = devsta; 66534e65944SIsaku Yamahata inj.error_status = error_status; 66634e65944SIsaku Yamahata inj.unsupported_request = !(err->flags & PCIE_AER_ERR_IS_CORRECTABLE) && 66734e65944SIsaku Yamahata err->status == PCI_ERR_UNC_UNSUP; 66834e65944SIsaku Yamahata inj.log_overflow = false; 66934e65944SIsaku Yamahata 67034e65944SIsaku Yamahata if (err->flags & PCIE_AER_ERR_IS_CORRECTABLE) { 67134e65944SIsaku Yamahata if (!pcie_aer_inject_cor_error(&inj, 0, false)) { 67234e65944SIsaku Yamahata return 0; 67334e65944SIsaku Yamahata } 67434e65944SIsaku Yamahata } else { 67534e65944SIsaku Yamahata bool is_fatal = 67634e65944SIsaku Yamahata pcie_aer_uncor_default_severity(error_status) == 67734e65944SIsaku Yamahata PCI_ERR_ROOT_CMD_FATAL_EN; 67834e65944SIsaku Yamahata if (aer_cap) { 67934e65944SIsaku Yamahata is_fatal = 68034e65944SIsaku Yamahata error_status & pci_get_long(aer_cap + PCI_ERR_UNCOR_SEVER); 68134e65944SIsaku Yamahata } 68234e65944SIsaku Yamahata if (!is_fatal && (err->flags & PCIE_AER_ERR_MAYBE_ADVISORY)) { 68334e65944SIsaku Yamahata inj.error_status = PCI_ERR_COR_ADV_NONFATAL; 68434e65944SIsaku Yamahata if (!pcie_aer_inject_cor_error(&inj, error_status, true)) { 68534e65944SIsaku Yamahata return 0; 68634e65944SIsaku Yamahata } 68734e65944SIsaku Yamahata } else { 68834e65944SIsaku Yamahata if (!pcie_aer_inject_uncor_error(&inj, is_fatal)) { 68934e65944SIsaku Yamahata return 0; 69034e65944SIsaku Yamahata } 69134e65944SIsaku Yamahata } 69234e65944SIsaku Yamahata } 69334e65944SIsaku Yamahata 69434e65944SIsaku Yamahata /* send up error message */ 69534e65944SIsaku Yamahata inj.msg.source_id = err->source_id; 69634e65944SIsaku Yamahata pcie_aer_msg(dev, &inj.msg); 69734e65944SIsaku Yamahata 69834e65944SIsaku Yamahata if (inj.log_overflow) { 69934e65944SIsaku Yamahata PCIEAERErr header_log_overflow = { 70034e65944SIsaku Yamahata .status = PCI_ERR_COR_HL_OVERFLOW, 70134e65944SIsaku Yamahata .flags = PCIE_AER_ERR_IS_CORRECTABLE, 70234e65944SIsaku Yamahata }; 70334e65944SIsaku Yamahata int ret = pcie_aer_inject_error(dev, &header_log_overflow); 70434e65944SIsaku Yamahata assert(!ret); 70534e65944SIsaku Yamahata } 70634e65944SIsaku Yamahata return 0; 70734e65944SIsaku Yamahata } 70834e65944SIsaku Yamahata 70934e65944SIsaku Yamahata void pcie_aer_write_config(PCIDevice *dev, 71034e65944SIsaku Yamahata uint32_t addr, uint32_t val, int len) 71134e65944SIsaku Yamahata { 71234e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 71334e65944SIsaku Yamahata uint32_t errcap = pci_get_long(aer_cap + PCI_ERR_CAP); 71434e65944SIsaku Yamahata uint32_t first_error = 1U << PCI_ERR_CAP_FEP(errcap); 71534e65944SIsaku Yamahata uint32_t uncorsta = pci_get_long(aer_cap + PCI_ERR_UNCOR_STATUS); 71634e65944SIsaku Yamahata 71734e65944SIsaku Yamahata /* uncorrectable error */ 71834e65944SIsaku Yamahata if (!(uncorsta & first_error)) { 71934e65944SIsaku Yamahata /* the bit that corresponds to the first error is cleared */ 72034e65944SIsaku Yamahata pcie_aer_clear_error(dev); 72134e65944SIsaku Yamahata } else if (errcap & PCI_ERR_CAP_MHRE) { 72234e65944SIsaku Yamahata /* When PCI_ERR_CAP_MHRE is enabled and the first error isn't cleared 72334e65944SIsaku Yamahata * nothing should happen. So we have to revert the modification to 72434e65944SIsaku Yamahata * the register. 72534e65944SIsaku Yamahata */ 72634e65944SIsaku Yamahata pcie_aer_update_uncor_status(dev); 72734e65944SIsaku Yamahata } else { 72834e65944SIsaku Yamahata /* capability & control 72934e65944SIsaku Yamahata * PCI_ERR_CAP_MHRE might be cleared, so clear of header log. 73034e65944SIsaku Yamahata */ 73134e65944SIsaku Yamahata aer_log_clear_all_err(&dev->exp.aer_log); 73234e65944SIsaku Yamahata } 73334e65944SIsaku Yamahata } 73434e65944SIsaku Yamahata 73534e65944SIsaku Yamahata void pcie_aer_root_init(PCIDevice *dev) 73634e65944SIsaku Yamahata { 73734e65944SIsaku Yamahata uint16_t pos = dev->exp.aer_cap; 73834e65944SIsaku Yamahata 73934e65944SIsaku Yamahata pci_set_long(dev->wmask + pos + PCI_ERR_ROOT_COMMAND, 74034e65944SIsaku Yamahata PCI_ERR_ROOT_CMD_EN_MASK); 74134e65944SIsaku Yamahata pci_set_long(dev->w1cmask + pos + PCI_ERR_ROOT_STATUS, 74234e65944SIsaku Yamahata PCI_ERR_ROOT_STATUS_REPORT_MASK); 7430e180d9cSJason Baron /* PCI_ERR_ROOT_IRQ is RO but devices change it using a 7440e180d9cSJason Baron * device-specific method. 7450e180d9cSJason Baron */ 7460e180d9cSJason Baron pci_set_long(dev->cmask + pos + PCI_ERR_ROOT_STATUS, 7470e180d9cSJason Baron ~PCI_ERR_ROOT_IRQ); 74834e65944SIsaku Yamahata } 74934e65944SIsaku Yamahata 75034e65944SIsaku Yamahata void pcie_aer_root_reset(PCIDevice *dev) 75134e65944SIsaku Yamahata { 75234e65944SIsaku Yamahata uint8_t* aer_cap = dev->config + dev->exp.aer_cap; 75334e65944SIsaku Yamahata 75434e65944SIsaku Yamahata pci_set_long(aer_cap + PCI_ERR_ROOT_COMMAND, 0); 75534e65944SIsaku Yamahata 75634e65944SIsaku Yamahata /* 75734e65944SIsaku Yamahata * Advanced Error Interrupt Message Number in Root Error Status Register 75834e65944SIsaku Yamahata * must be updated by chip dependent code because it's chip dependent 75934e65944SIsaku Yamahata * which number is used. 76034e65944SIsaku Yamahata */ 76134e65944SIsaku Yamahata } 76234e65944SIsaku Yamahata 76334e65944SIsaku Yamahata void pcie_aer_root_write_config(PCIDevice *dev, 76434e65944SIsaku Yamahata uint32_t addr, uint32_t val, int len, 76534e65944SIsaku Yamahata uint32_t root_cmd_prev) 76634e65944SIsaku Yamahata { 76734e65944SIsaku Yamahata uint8_t *aer_cap = dev->config + dev->exp.aer_cap; 76834e65944SIsaku Yamahata uint32_t root_status = pci_get_long(aer_cap + PCI_ERR_ROOT_STATUS); 7692b3cb353SMichael S. Tsirkin uint32_t enabled_cmd = pcie_aer_status_to_cmd(root_status); 7702b3cb353SMichael S. Tsirkin uint32_t root_cmd = pci_get_long(aer_cap + PCI_ERR_ROOT_COMMAND); 7712b3cb353SMichael S. Tsirkin /* 6.2.4.1.2 Interrupt Generation */ 7722b3cb353SMichael S. Tsirkin if (!msix_enabled(dev) && !msi_enabled(dev)) { 7735a03e708SMarcel Apfelbaum pci_set_irq(dev, !!(root_cmd & enabled_cmd)); 7742b3cb353SMichael S. Tsirkin return; 7752b3cb353SMichael S. Tsirkin } 7762b3cb353SMichael S. Tsirkin 7772b3cb353SMichael S. Tsirkin if ((root_cmd_prev & enabled_cmd) || !(root_cmd & enabled_cmd)) { 7782b3cb353SMichael S. Tsirkin /* Send MSI on transition from false to true. */ 7792b3cb353SMichael S. Tsirkin return; 7802b3cb353SMichael S. Tsirkin } 78134e65944SIsaku Yamahata 782513691b7SMichael S. Tsirkin pcie_aer_root_notify(dev); 78334e65944SIsaku Yamahata } 78434e65944SIsaku Yamahata 78534e65944SIsaku Yamahata static const VMStateDescription vmstate_pcie_aer_err = { 78634e65944SIsaku Yamahata .name = "PCIE_AER_ERROR", 78734e65944SIsaku Yamahata .version_id = 1, 78834e65944SIsaku Yamahata .minimum_version_id = 1, 78934e65944SIsaku Yamahata .fields = (VMStateField[]) { 79034e65944SIsaku Yamahata VMSTATE_UINT32(status, PCIEAERErr), 79134e65944SIsaku Yamahata VMSTATE_UINT16(source_id, PCIEAERErr), 79234e65944SIsaku Yamahata VMSTATE_UINT16(flags, PCIEAERErr), 79334e65944SIsaku Yamahata VMSTATE_UINT32_ARRAY(header, PCIEAERErr, 4), 79434e65944SIsaku Yamahata VMSTATE_UINT32_ARRAY(prefix, PCIEAERErr, 4), 79534e65944SIsaku Yamahata VMSTATE_END_OF_LIST() 79634e65944SIsaku Yamahata } 79734e65944SIsaku Yamahata }; 79834e65944SIsaku Yamahata 7995f691ff9SMichael S. Tsirkin static bool pcie_aer_state_log_num_valid(void *opaque, int version_id) 8005f691ff9SMichael S. Tsirkin { 8015f691ff9SMichael S. Tsirkin PCIEAERLog *s = opaque; 8025f691ff9SMichael S. Tsirkin 8035f691ff9SMichael S. Tsirkin return s->log_num <= s->log_max; 8045f691ff9SMichael S. Tsirkin } 8055f691ff9SMichael S. Tsirkin 80634e65944SIsaku Yamahata const VMStateDescription vmstate_pcie_aer_log = { 80734e65944SIsaku Yamahata .name = "PCIE_AER_ERROR_LOG", 80834e65944SIsaku Yamahata .version_id = 1, 80934e65944SIsaku Yamahata .minimum_version_id = 1, 81034e65944SIsaku Yamahata .fields = (VMStateField[]) { 81134e65944SIsaku Yamahata VMSTATE_UINT16(log_num, PCIEAERLog), 8125f691ff9SMichael S. Tsirkin VMSTATE_UINT16_EQUAL(log_max, PCIEAERLog), 8135f691ff9SMichael S. Tsirkin VMSTATE_VALIDATE("log_num <= log_max", pcie_aer_state_log_num_valid), 81447188700SDmitry Eremin-Solenikov VMSTATE_STRUCT_VARRAY_POINTER_UINT16(log, PCIEAERLog, log_num, 81534e65944SIsaku Yamahata vmstate_pcie_aer_err, PCIEAERErr), 81634e65944SIsaku Yamahata VMSTATE_END_OF_LIST() 81734e65944SIsaku Yamahata } 81834e65944SIsaku Yamahata }; 8192ae63bdaSIsaku Yamahata 8202ae63bdaSIsaku Yamahata typedef struct PCIEAERErrorName { 8212ae63bdaSIsaku Yamahata const char *name; 8222ae63bdaSIsaku Yamahata uint32_t val; 8232ae63bdaSIsaku Yamahata bool correctable; 8242ae63bdaSIsaku Yamahata } PCIEAERErrorName; 8252ae63bdaSIsaku Yamahata 8262ae63bdaSIsaku Yamahata /* 82766a0a2cbSDong Xu Wang * AER error name -> value conversion table 8282ae63bdaSIsaku Yamahata * This naming scheme is same to linux aer-injection tool. 8292ae63bdaSIsaku Yamahata */ 8302ae63bdaSIsaku Yamahata static const struct PCIEAERErrorName pcie_aer_error_list[] = { 8312ae63bdaSIsaku Yamahata { 8322ae63bdaSIsaku Yamahata .name = "DLP", 8332ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_DLP, 8342ae63bdaSIsaku Yamahata .correctable = false, 8352ae63bdaSIsaku Yamahata }, { 8362ae63bdaSIsaku Yamahata .name = "SDN", 8372ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_SDN, 8382ae63bdaSIsaku Yamahata .correctable = false, 8392ae63bdaSIsaku Yamahata }, { 8402ae63bdaSIsaku Yamahata .name = "POISON_TLP", 8412ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_POISON_TLP, 8422ae63bdaSIsaku Yamahata .correctable = false, 8432ae63bdaSIsaku Yamahata }, { 8442ae63bdaSIsaku Yamahata .name = "FCP", 8452ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_FCP, 8462ae63bdaSIsaku Yamahata .correctable = false, 8472ae63bdaSIsaku Yamahata }, { 8482ae63bdaSIsaku Yamahata .name = "COMP_TIME", 8492ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_COMP_TIME, 8502ae63bdaSIsaku Yamahata .correctable = false, 8512ae63bdaSIsaku Yamahata }, { 8522ae63bdaSIsaku Yamahata .name = "COMP_ABORT", 8532ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_COMP_ABORT, 8542ae63bdaSIsaku Yamahata .correctable = false, 8552ae63bdaSIsaku Yamahata }, { 8562ae63bdaSIsaku Yamahata .name = "UNX_COMP", 8572ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_UNX_COMP, 8582ae63bdaSIsaku Yamahata .correctable = false, 8592ae63bdaSIsaku Yamahata }, { 8602ae63bdaSIsaku Yamahata .name = "RX_OVER", 8612ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_RX_OVER, 8622ae63bdaSIsaku Yamahata .correctable = false, 8632ae63bdaSIsaku Yamahata }, { 8642ae63bdaSIsaku Yamahata .name = "MALF_TLP", 8652ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_MALF_TLP, 8662ae63bdaSIsaku Yamahata .correctable = false, 8672ae63bdaSIsaku Yamahata }, { 8682ae63bdaSIsaku Yamahata .name = "ECRC", 8692ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_ECRC, 8702ae63bdaSIsaku Yamahata .correctable = false, 8712ae63bdaSIsaku Yamahata }, { 8722ae63bdaSIsaku Yamahata .name = "UNSUP", 8732ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_UNSUP, 8742ae63bdaSIsaku Yamahata .correctable = false, 8752ae63bdaSIsaku Yamahata }, { 8762ae63bdaSIsaku Yamahata .name = "ACSV", 8772ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_ACSV, 8782ae63bdaSIsaku Yamahata .correctable = false, 8792ae63bdaSIsaku Yamahata }, { 8802ae63bdaSIsaku Yamahata .name = "INTN", 8812ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_INTN, 8822ae63bdaSIsaku Yamahata .correctable = false, 8832ae63bdaSIsaku Yamahata }, { 8842ae63bdaSIsaku Yamahata .name = "MCBTLP", 8852ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_MCBTLP, 8862ae63bdaSIsaku Yamahata .correctable = false, 8872ae63bdaSIsaku Yamahata }, { 8882ae63bdaSIsaku Yamahata .name = "ATOP_EBLOCKED", 8892ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_ATOP_EBLOCKED, 8902ae63bdaSIsaku Yamahata .correctable = false, 8912ae63bdaSIsaku Yamahata }, { 8922ae63bdaSIsaku Yamahata .name = "TLP_PRF_BLOCKED", 8932ae63bdaSIsaku Yamahata .val = PCI_ERR_UNC_TLP_PRF_BLOCKED, 8942ae63bdaSIsaku Yamahata .correctable = false, 8952ae63bdaSIsaku Yamahata }, { 8962ae63bdaSIsaku Yamahata .name = "RCVR", 8972ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_RCVR, 8982ae63bdaSIsaku Yamahata .correctable = true, 8992ae63bdaSIsaku Yamahata }, { 9002ae63bdaSIsaku Yamahata .name = "BAD_TLP", 9012ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_BAD_TLP, 9022ae63bdaSIsaku Yamahata .correctable = true, 9032ae63bdaSIsaku Yamahata }, { 9042ae63bdaSIsaku Yamahata .name = "BAD_DLLP", 9052ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_BAD_DLLP, 9062ae63bdaSIsaku Yamahata .correctable = true, 9072ae63bdaSIsaku Yamahata }, { 9082ae63bdaSIsaku Yamahata .name = "REP_ROLL", 9092ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_REP_ROLL, 9102ae63bdaSIsaku Yamahata .correctable = true, 9112ae63bdaSIsaku Yamahata }, { 9122ae63bdaSIsaku Yamahata .name = "REP_TIMER", 9132ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_REP_TIMER, 9142ae63bdaSIsaku Yamahata .correctable = true, 9152ae63bdaSIsaku Yamahata }, { 9162ae63bdaSIsaku Yamahata .name = "ADV_NONFATAL", 9172ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_ADV_NONFATAL, 9182ae63bdaSIsaku Yamahata .correctable = true, 9192ae63bdaSIsaku Yamahata }, { 9202ae63bdaSIsaku Yamahata .name = "INTERNAL", 9212ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_INTERNAL, 9222ae63bdaSIsaku Yamahata .correctable = true, 9232ae63bdaSIsaku Yamahata }, { 9242ae63bdaSIsaku Yamahata .name = "HL_OVERFLOW", 9252ae63bdaSIsaku Yamahata .val = PCI_ERR_COR_HL_OVERFLOW, 9262ae63bdaSIsaku Yamahata .correctable = true, 9272ae63bdaSIsaku Yamahata }, 9282ae63bdaSIsaku Yamahata }; 9292ae63bdaSIsaku Yamahata 9302ae63bdaSIsaku Yamahata static int pcie_aer_parse_error_string(const char *error_name, 9312ae63bdaSIsaku Yamahata uint32_t *status, bool *correctable) 9322ae63bdaSIsaku Yamahata { 9332ae63bdaSIsaku Yamahata int i; 9342ae63bdaSIsaku Yamahata 9352ae63bdaSIsaku Yamahata for (i = 0; i < ARRAY_SIZE(pcie_aer_error_list); i++) { 9362ae63bdaSIsaku Yamahata const PCIEAERErrorName *e = &pcie_aer_error_list[i]; 9372ae63bdaSIsaku Yamahata if (strcmp(error_name, e->name)) { 9382ae63bdaSIsaku Yamahata continue; 9392ae63bdaSIsaku Yamahata } 9402ae63bdaSIsaku Yamahata 9412ae63bdaSIsaku Yamahata *status = e->val; 9422ae63bdaSIsaku Yamahata *correctable = e->correctable; 9432ae63bdaSIsaku Yamahata return 0; 9442ae63bdaSIsaku Yamahata } 9452ae63bdaSIsaku Yamahata return -EINVAL; 9462ae63bdaSIsaku Yamahata } 9472ae63bdaSIsaku Yamahata 94804e00c92SMarkus Armbruster static int do_pcie_aer_inject_error(Monitor *mon, 9492ae63bdaSIsaku Yamahata const QDict *qdict, QObject **ret_data) 9502ae63bdaSIsaku Yamahata { 9512ae63bdaSIsaku Yamahata const char *id = qdict_get_str(qdict, "id"); 9522ae63bdaSIsaku Yamahata const char *error_name; 9532ae63bdaSIsaku Yamahata uint32_t error_status; 9542ae63bdaSIsaku Yamahata bool correctable; 9552ae63bdaSIsaku Yamahata PCIDevice *dev; 9562ae63bdaSIsaku Yamahata PCIEAERErr err; 9572ae63bdaSIsaku Yamahata int ret; 9582ae63bdaSIsaku Yamahata 9592ae63bdaSIsaku Yamahata ret = pci_qdev_find_device(id, &dev); 9602ae63bdaSIsaku Yamahata if (ret < 0) { 9612ae63bdaSIsaku Yamahata monitor_printf(mon, 9622ae63bdaSIsaku Yamahata "id or pci device path is invalid or device not " 9632ae63bdaSIsaku Yamahata "found. %s\n", id); 9642ae63bdaSIsaku Yamahata return ret; 9652ae63bdaSIsaku Yamahata } 9662ae63bdaSIsaku Yamahata if (!pci_is_express(dev)) { 9672ae63bdaSIsaku Yamahata monitor_printf(mon, "the device doesn't support pci express. %s\n", 9682ae63bdaSIsaku Yamahata id); 9692ae63bdaSIsaku Yamahata return -ENOSYS; 9702ae63bdaSIsaku Yamahata } 9712ae63bdaSIsaku Yamahata 9722ae63bdaSIsaku Yamahata error_name = qdict_get_str(qdict, "error_status"); 9732ae63bdaSIsaku Yamahata if (pcie_aer_parse_error_string(error_name, &error_status, &correctable)) { 9742ae63bdaSIsaku Yamahata char *e = NULL; 9752ae63bdaSIsaku Yamahata error_status = strtoul(error_name, &e, 0); 97634acbc95SEric Blake correctable = qdict_get_try_bool(qdict, "correctable", false); 9772ae63bdaSIsaku Yamahata if (!e || *e != '\0') { 9782ae63bdaSIsaku Yamahata monitor_printf(mon, "invalid error status value. \"%s\"", 9792ae63bdaSIsaku Yamahata error_name); 9802ae63bdaSIsaku Yamahata return -EINVAL; 9812ae63bdaSIsaku Yamahata } 9822ae63bdaSIsaku Yamahata } 98374d63b65SIsaku Yamahata err.status = error_status; 984a05f686fSPavel Fedin err.source_id = pci_requester_id(dev); 9852ae63bdaSIsaku Yamahata 9862ae63bdaSIsaku Yamahata err.flags = 0; 9872ae63bdaSIsaku Yamahata if (correctable) { 9882ae63bdaSIsaku Yamahata err.flags |= PCIE_AER_ERR_IS_CORRECTABLE; 9892ae63bdaSIsaku Yamahata } 99034acbc95SEric Blake if (qdict_get_try_bool(qdict, "advisory_non_fatal", false)) { 9912ae63bdaSIsaku Yamahata err.flags |= PCIE_AER_ERR_MAYBE_ADVISORY; 9922ae63bdaSIsaku Yamahata } 9932ae63bdaSIsaku Yamahata if (qdict_haskey(qdict, "header0")) { 9942ae63bdaSIsaku Yamahata err.flags |= PCIE_AER_ERR_HEADER_VALID; 9952ae63bdaSIsaku Yamahata } 9962ae63bdaSIsaku Yamahata if (qdict_haskey(qdict, "prefix0")) { 9972ae63bdaSIsaku Yamahata err.flags |= PCIE_AER_ERR_TLP_PREFIX_PRESENT; 9982ae63bdaSIsaku Yamahata } 9992ae63bdaSIsaku Yamahata 10002ae63bdaSIsaku Yamahata err.header[0] = qdict_get_try_int(qdict, "header0", 0); 10012ae63bdaSIsaku Yamahata err.header[1] = qdict_get_try_int(qdict, "header1", 0); 10022ae63bdaSIsaku Yamahata err.header[2] = qdict_get_try_int(qdict, "header2", 0); 10032ae63bdaSIsaku Yamahata err.header[3] = qdict_get_try_int(qdict, "header3", 0); 10042ae63bdaSIsaku Yamahata 10052ae63bdaSIsaku Yamahata err.prefix[0] = qdict_get_try_int(qdict, "prefix0", 0); 10062ae63bdaSIsaku Yamahata err.prefix[1] = qdict_get_try_int(qdict, "prefix1", 0); 10072ae63bdaSIsaku Yamahata err.prefix[2] = qdict_get_try_int(qdict, "prefix2", 0); 10082ae63bdaSIsaku Yamahata err.prefix[3] = qdict_get_try_int(qdict, "prefix3", 0); 10092ae63bdaSIsaku Yamahata 10102ae63bdaSIsaku Yamahata ret = pcie_aer_inject_error(dev, &err); 10112ae63bdaSIsaku Yamahata *ret_data = qobject_from_jsonf("{'id': %s, " 1012568f0690SDavid Gibson "'root_bus': %s, 'bus': %d, 'devfn': %d, " 10132ae63bdaSIsaku Yamahata "'ret': %d}", 1014568f0690SDavid Gibson id, pci_root_bus_path(dev), 10152ae63bdaSIsaku Yamahata pci_bus_num(dev->bus), dev->devfn, 10162ae63bdaSIsaku Yamahata ret); 10172ae63bdaSIsaku Yamahata assert(*ret_data); 10182ae63bdaSIsaku Yamahata 10192ae63bdaSIsaku Yamahata return 0; 10202ae63bdaSIsaku Yamahata } 102104e00c92SMarkus Armbruster 102204e00c92SMarkus Armbruster void hmp_pcie_aer_inject_error(Monitor *mon, const QDict *qdict) 102304e00c92SMarkus Armbruster { 102404e00c92SMarkus Armbruster QObject *data; 102504e00c92SMarkus Armbruster int devfn; 102604e00c92SMarkus Armbruster 102704e00c92SMarkus Armbruster if (do_pcie_aer_inject_error(mon, qdict, &data) < 0) { 102804e00c92SMarkus Armbruster return; 102904e00c92SMarkus Armbruster } 103004e00c92SMarkus Armbruster 103104e00c92SMarkus Armbruster assert(qobject_type(data) == QTYPE_QDICT); 103204e00c92SMarkus Armbruster qdict = qobject_to_qdict(data); 103304e00c92SMarkus Armbruster 103404e00c92SMarkus Armbruster devfn = (int)qdict_get_int(qdict, "devfn"); 103504e00c92SMarkus Armbruster monitor_printf(mon, "OK id: %s root bus: %s, bus: %x devfn: %x.%x\n", 103604e00c92SMarkus Armbruster qdict_get_str(qdict, "id"), 103704e00c92SMarkus Armbruster qdict_get_str(qdict, "root_bus"), 103804e00c92SMarkus Armbruster (int) qdict_get_int(qdict, "bus"), 103904e00c92SMarkus Armbruster PCI_SLOT(devfn), PCI_FUNC(devfn)); 104004e00c92SMarkus Armbruster } 1041