| /* |
| * QEMU PowerPC PowerNV Emulation of some SBE behaviour |
| * |
| * Copyright (c) 2022, IBM Corporation. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License, version 2, as |
| * published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, see <http://www.gnu.org/licenses/>. |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "target/ppc/cpu.h" |
| #include "qapi/error.h" |
| #include "qemu/log.h" |
| #include "qemu/module.h" |
| #include "hw/irq.h" |
| #include "hw/qdev-properties.h" |
| #include "hw/ppc/pnv.h" |
| #include "hw/ppc/pnv_xscom.h" |
| #include "hw/ppc/pnv_sbe.h" |
| #include "trace.h" |
| |
| /* |
| * Most register and command definitions come from skiboot. |
| * |
| * xscom addresses are adjusted to be relative to xscom subregion bases |
| */ |
| |
| /* |
| * SBE MBOX register address |
| * Reg 0 - 3 : Host to send command packets to SBE |
| * Reg 4 - 7 : SBE to send response packets to Host |
| */ |
| #define PSU_HOST_SBE_MBOX_REG0 0x00000000 |
| #define PSU_HOST_SBE_MBOX_REG1 0x00000001 |
| #define PSU_HOST_SBE_MBOX_REG2 0x00000002 |
| #define PSU_HOST_SBE_MBOX_REG3 0x00000003 |
| #define PSU_HOST_SBE_MBOX_REG4 0x00000004 |
| #define PSU_HOST_SBE_MBOX_REG5 0x00000005 |
| #define PSU_HOST_SBE_MBOX_REG6 0x00000006 |
| #define PSU_HOST_SBE_MBOX_REG7 0x00000007 |
| #define PSU_SBE_DOORBELL_REG_RW 0x00000010 |
| #define PSU_SBE_DOORBELL_REG_AND 0x00000011 |
| #define PSU_SBE_DOORBELL_REG_OR 0x00000012 |
| #define PSU_HOST_DOORBELL_REG_RW 0x00000013 |
| #define PSU_HOST_DOORBELL_REG_AND 0x00000014 |
| #define PSU_HOST_DOORBELL_REG_OR 0x00000015 |
| |
| /* |
| * Doorbell register to trigger SBE interrupt. Set by OPAL to inform |
| * the SBE about a waiting message in the Host/SBE mailbox registers |
| */ |
| #define HOST_SBE_MSG_WAITING PPC_BIT(0) |
| |
| /* |
| * Doorbell register for host bridge interrupt. Set by the SBE to inform |
| * host about a response message in the Host/SBE mailbox registers |
| */ |
| #define SBE_HOST_RESPONSE_WAITING PPC_BIT(0) |
| #define SBE_HOST_MSG_READ PPC_BIT(1) |
| #define SBE_HOST_STOP15_EXIT PPC_BIT(2) |
| #define SBE_HOST_RESET PPC_BIT(3) |
| #define SBE_HOST_PASSTHROUGH PPC_BIT(4) |
| #define SBE_HOST_TIMER_EXPIRY PPC_BIT(14) |
| #define SBE_HOST_RESPONSE_MASK (PPC_BITMASK(0, 4) | \ |
| SBE_HOST_TIMER_EXPIRY) |
| |
| /* SBE Control Register */ |
| #define SBE_CONTROL_REG_RW 0x00000000 |
| |
| /* SBE interrupt s0/s1 bits */ |
| #define SBE_CONTROL_REG_S0 PPC_BIT(14) |
| #define SBE_CONTROL_REG_S1 PPC_BIT(15) |
| |
| struct sbe_msg { |
| uint64_t reg[4]; |
| }; |
| |
| static uint64_t pnv_sbe_power9_xscom_ctrl_read(void *opaque, hwaddr addr, |
| unsigned size) |
| { |
| uint32_t offset = addr >> 3; |
| uint64_t val = 0; |
| |
| switch (offset) { |
| default: |
| qemu_log_mask(LOG_UNIMP, "SBE Unimplemented register: Ox%" |
| HWADDR_PRIx "\n", addr >> 3); |
| } |
| |
| trace_pnv_sbe_xscom_ctrl_read(addr, val); |
| |
| return val; |
| } |
| |
| static void pnv_sbe_power9_xscom_ctrl_write(void *opaque, hwaddr addr, |
| uint64_t val, unsigned size) |
| { |
| uint32_t offset = addr >> 3; |
| |
| trace_pnv_sbe_xscom_ctrl_write(addr, val); |
| |
| switch (offset) { |
| default: |
| qemu_log_mask(LOG_UNIMP, "SBE Unimplemented register: Ox%" |
| HWADDR_PRIx "\n", addr >> 3); |
| } |
| } |
| |
| static const MemoryRegionOps pnv_sbe_power9_xscom_ctrl_ops = { |
| .read = pnv_sbe_power9_xscom_ctrl_read, |
| .write = pnv_sbe_power9_xscom_ctrl_write, |
| .valid.min_access_size = 8, |
| .valid.max_access_size = 8, |
| .impl.min_access_size = 8, |
| .impl.max_access_size = 8, |
| .endianness = DEVICE_BIG_ENDIAN, |
| }; |
| |
| static void pnv_sbe_set_host_doorbell(PnvSBE *sbe, uint64_t val) |
| { |
| val &= SBE_HOST_RESPONSE_MASK; /* Is this right? What does HW do? */ |
| sbe->host_doorbell = val; |
| |
| trace_pnv_sbe_reg_set_host_doorbell(val); |
| qemu_set_irq(sbe->psi_irq, !!val); |
| } |
| |
| /* SBE Target Type */ |
| #define SBE_TARGET_TYPE_PROC 0x00 |
| #define SBE_TARGET_TYPE_EX 0x01 |
| #define SBE_TARGET_TYPE_PERV 0x02 |
| #define SBE_TARGET_TYPE_MCS 0x03 |
| #define SBE_TARGET_TYPE_EQ 0x04 |
| #define SBE_TARGET_TYPE_CORE 0x05 |
| |
| /* SBE MBOX command class */ |
| #define SBE_MCLASS_FIRST 0xD1 |
| #define SBE_MCLASS_CORE_STATE 0xD1 |
| #define SBE_MCLASS_SCOM 0xD2 |
| #define SBE_MCLASS_RING 0xD3 |
| #define SBE_MCLASS_TIMER 0xD4 |
| #define SBE_MCLASS_MPIPL 0xD5 |
| #define SBE_MCLASS_SECURITY 0xD6 |
| #define SBE_MCLASS_GENERIC 0xD7 |
| #define SBE_MCLASS_LAST 0xD7 |
| |
| /* |
| * Commands are provided in xxyy form where: |
| * - xx : command class |
| * - yy : command |
| * |
| * Both request and response message uses same seq ID, |
| * command class and command. |
| */ |
| #define SBE_CMD_CTRL_DEADMAN_LOOP 0xD101 |
| #define SBE_CMD_MULTI_SCOM 0xD201 |
| #define SBE_CMD_PUT_RING_FORM_IMAGE 0xD301 |
| #define SBE_CMD_CONTROL_TIMER 0xD401 |
| #define SBE_CMD_GET_ARCHITECTED_REG 0xD501 |
| #define SBE_CMD_CLR_ARCHITECTED_REG 0xD502 |
| #define SBE_CMD_SET_UNSEC_MEM_WINDOW 0xD601 |
| #define SBE_CMD_GET_SBE_FFDC 0xD701 |
| #define SBE_CMD_GET_CAPABILITY 0xD702 |
| #define SBE_CMD_READ_SBE_SEEPROM 0xD703 |
| #define SBE_CMD_SET_FFDC_ADDR 0xD704 |
| #define SBE_CMD_QUIESCE_SBE 0xD705 |
| #define SBE_CMD_SET_FABRIC_ID_MAP 0xD706 |
| #define SBE_CMD_STASH_MPIPL_CONFIG 0xD707 |
| |
| /* SBE MBOX control flags */ |
| |
| /* Generic flags */ |
| #define SBE_CMD_CTRL_RESP_REQ 0x0100 |
| #define SBE_CMD_CTRL_ACK_REQ 0x0200 |
| |
| /* Deadman loop */ |
| #define CTRL_DEADMAN_LOOP_START 0x0001 |
| #define CTRL_DEADMAN_LOOP_STOP 0x0002 |
| |
| /* Control timer */ |
| #define CONTROL_TIMER_START 0x0001 |
| #define CONTROL_TIMER_STOP 0x0002 |
| |
| /* Stash MPIPL config */ |
| #define SBE_STASH_KEY_SKIBOOT_BASE 0x03 |
| |
| static void sbe_timer(void *opaque) |
| { |
| PnvSBE *sbe = opaque; |
| |
| trace_pnv_sbe_cmd_timer_expired(); |
| |
| pnv_sbe_set_host_doorbell(sbe, sbe->host_doorbell | SBE_HOST_TIMER_EXPIRY); |
| } |
| |
| static void do_sbe_msg(PnvSBE *sbe) |
| { |
| struct sbe_msg msg; |
| uint16_t cmd, ctrl_flags, seq_id; |
| int i; |
| |
| memset(&msg, 0, sizeof(msg)); |
| |
| for (i = 0; i < 4; i++) { |
| msg.reg[i] = sbe->mbox[i]; |
| } |
| |
| cmd = msg.reg[0]; |
| seq_id = msg.reg[0] >> 16; |
| ctrl_flags = msg.reg[0] >> 32; |
| |
| trace_pnv_sbe_msg_recv(cmd, seq_id, ctrl_flags); |
| |
| if (ctrl_flags & SBE_CMD_CTRL_ACK_REQ) { |
| pnv_sbe_set_host_doorbell(sbe, sbe->host_doorbell | SBE_HOST_MSG_READ); |
| } |
| |
| switch (cmd) { |
| case SBE_CMD_CONTROL_TIMER: |
| if (ctrl_flags & CONTROL_TIMER_START) { |
| uint64_t us = msg.reg[1]; |
| trace_pnv_sbe_cmd_timer_start(us); |
| timer_mod(sbe->timer, qemu_clock_get_us(QEMU_CLOCK_VIRTUAL) + us); |
| } |
| if (ctrl_flags & CONTROL_TIMER_STOP) { |
| trace_pnv_sbe_cmd_timer_stop(); |
| timer_del(sbe->timer); |
| } |
| break; |
| default: |
| qemu_log_mask(LOG_UNIMP, "SBE Unimplemented command: 0x%x\n", cmd); |
| } |
| } |
| |
| static void pnv_sbe_set_sbe_doorbell(PnvSBE *sbe, uint64_t val) |
| { |
| val &= HOST_SBE_MSG_WAITING; |
| sbe->sbe_doorbell = val; |
| |
| if (val & HOST_SBE_MSG_WAITING) { |
| sbe->sbe_doorbell &= ~HOST_SBE_MSG_WAITING; |
| do_sbe_msg(sbe); |
| } |
| } |
| |
| static uint64_t pnv_sbe_power9_xscom_mbox_read(void *opaque, hwaddr addr, |
| unsigned size) |
| { |
| PnvSBE *sbe = PNV_SBE(opaque); |
| uint32_t offset = addr >> 3; |
| uint64_t val = 0; |
| |
| if (offset <= PSU_HOST_SBE_MBOX_REG7) { |
| uint32_t idx = offset - PSU_HOST_SBE_MBOX_REG0; |
| val = sbe->mbox[idx]; |
| } else { |
| switch (offset) { |
| case PSU_SBE_DOORBELL_REG_RW: |
| val = sbe->sbe_doorbell; |
| break; |
| case PSU_HOST_DOORBELL_REG_RW: |
| val = sbe->host_doorbell; |
| break; |
| default: |
| qemu_log_mask(LOG_UNIMP, "SBE Unimplemented register: Ox%" |
| HWADDR_PRIx "\n", addr >> 3); |
| } |
| } |
| |
| trace_pnv_sbe_xscom_mbox_read(addr, val); |
| |
| return val; |
| } |
| |
| static void pnv_sbe_power9_xscom_mbox_write(void *opaque, hwaddr addr, |
| uint64_t val, unsigned size) |
| { |
| PnvSBE *sbe = PNV_SBE(opaque); |
| uint32_t offset = addr >> 3; |
| |
| trace_pnv_sbe_xscom_mbox_write(addr, val); |
| |
| if (offset <= PSU_HOST_SBE_MBOX_REG7) { |
| uint32_t idx = offset - PSU_HOST_SBE_MBOX_REG0; |
| sbe->mbox[idx] = val; |
| } else { |
| switch (offset) { |
| case PSU_SBE_DOORBELL_REG_RW: |
| pnv_sbe_set_sbe_doorbell(sbe, val); |
| break; |
| case PSU_SBE_DOORBELL_REG_AND: |
| pnv_sbe_set_sbe_doorbell(sbe, sbe->sbe_doorbell & val); |
| break; |
| case PSU_SBE_DOORBELL_REG_OR: |
| pnv_sbe_set_sbe_doorbell(sbe, sbe->sbe_doorbell | val); |
| break; |
| |
| case PSU_HOST_DOORBELL_REG_RW: |
| pnv_sbe_set_host_doorbell(sbe, val); |
| break; |
| case PSU_HOST_DOORBELL_REG_AND: |
| pnv_sbe_set_host_doorbell(sbe, sbe->host_doorbell & val); |
| break; |
| case PSU_HOST_DOORBELL_REG_OR: |
| pnv_sbe_set_host_doorbell(sbe, sbe->host_doorbell | val); |
| break; |
| |
| default: |
| qemu_log_mask(LOG_UNIMP, "SBE Unimplemented register: Ox%" |
| HWADDR_PRIx "\n", addr >> 3); |
| } |
| } |
| } |
| |
| static const MemoryRegionOps pnv_sbe_power9_xscom_mbox_ops = { |
| .read = pnv_sbe_power9_xscom_mbox_read, |
| .write = pnv_sbe_power9_xscom_mbox_write, |
| .valid.min_access_size = 8, |
| .valid.max_access_size = 8, |
| .impl.min_access_size = 8, |
| .impl.max_access_size = 8, |
| .endianness = DEVICE_BIG_ENDIAN, |
| }; |
| |
| static void pnv_sbe_power9_class_init(ObjectClass *klass, void *data) |
| { |
| PnvSBEClass *psc = PNV_SBE_CLASS(klass); |
| DeviceClass *dc = DEVICE_CLASS(klass); |
| |
| dc->desc = "PowerNV SBE Controller (POWER9)"; |
| psc->xscom_ctrl_size = PNV9_XSCOM_SBE_CTRL_SIZE; |
| psc->xscom_ctrl_ops = &pnv_sbe_power9_xscom_ctrl_ops; |
| psc->xscom_mbox_size = PNV9_XSCOM_SBE_MBOX_SIZE; |
| psc->xscom_mbox_ops = &pnv_sbe_power9_xscom_mbox_ops; |
| } |
| |
| static const TypeInfo pnv_sbe_power9_type_info = { |
| .name = TYPE_PNV9_SBE, |
| .parent = TYPE_PNV_SBE, |
| .instance_size = sizeof(PnvSBE), |
| .class_init = pnv_sbe_power9_class_init, |
| }; |
| |
| static void pnv_sbe_power10_class_init(ObjectClass *klass, void *data) |
| { |
| PnvSBEClass *psc = PNV_SBE_CLASS(klass); |
| DeviceClass *dc = DEVICE_CLASS(klass); |
| |
| dc->desc = "PowerNV SBE Controller (POWER10)"; |
| psc->xscom_ctrl_size = PNV10_XSCOM_SBE_CTRL_SIZE; |
| psc->xscom_ctrl_ops = &pnv_sbe_power9_xscom_ctrl_ops; |
| psc->xscom_mbox_size = PNV10_XSCOM_SBE_MBOX_SIZE; |
| psc->xscom_mbox_ops = &pnv_sbe_power9_xscom_mbox_ops; |
| } |
| |
| static const TypeInfo pnv_sbe_power10_type_info = { |
| .name = TYPE_PNV10_SBE, |
| .parent = TYPE_PNV9_SBE, |
| .class_init = pnv_sbe_power10_class_init, |
| }; |
| |
| static void pnv_sbe_realize(DeviceState *dev, Error **errp) |
| { |
| PnvSBE *sbe = PNV_SBE(dev); |
| PnvSBEClass *psc = PNV_SBE_GET_CLASS(sbe); |
| |
| /* XScom regions for SBE registers */ |
| pnv_xscom_region_init(&sbe->xscom_ctrl_regs, OBJECT(dev), |
| psc->xscom_ctrl_ops, sbe, "xscom-sbe-ctrl", |
| psc->xscom_ctrl_size); |
| pnv_xscom_region_init(&sbe->xscom_mbox_regs, OBJECT(dev), |
| psc->xscom_mbox_ops, sbe, "xscom-sbe-mbox", |
| psc->xscom_mbox_size); |
| |
| qdev_init_gpio_out(dev, &sbe->psi_irq, 1); |
| |
| sbe->timer = timer_new_us(QEMU_CLOCK_VIRTUAL, sbe_timer, sbe); |
| } |
| |
| static void pnv_sbe_class_init(ObjectClass *klass, void *data) |
| { |
| DeviceClass *dc = DEVICE_CLASS(klass); |
| |
| dc->realize = pnv_sbe_realize; |
| dc->desc = "PowerNV SBE Controller"; |
| dc->user_creatable = false; |
| } |
| |
| static const TypeInfo pnv_sbe_type_info = { |
| .name = TYPE_PNV_SBE, |
| .parent = TYPE_DEVICE, |
| .instance_size = sizeof(PnvSBE), |
| .class_init = pnv_sbe_class_init, |
| .class_size = sizeof(PnvSBEClass), |
| .abstract = true, |
| }; |
| |
| static void pnv_sbe_register_types(void) |
| { |
| type_register_static(&pnv_sbe_type_info); |
| type_register_static(&pnv_sbe_power9_type_info); |
| type_register_static(&pnv_sbe_power10_type_info); |
| } |
| |
| type_init(pnv_sbe_register_types); |