1aeaef83dSCédric Le Goater /* 2aeaef83dSCédric Le Goater * QEMU PowerNV, BMC related functions 3aeaef83dSCédric Le Goater * 4aeaef83dSCédric Le Goater * Copyright (c) 2016-2017, IBM Corporation. 5aeaef83dSCédric Le Goater * 6aeaef83dSCédric Le Goater * This program is free software; you can redistribute it and/or modify 7aeaef83dSCédric Le Goater * it under the terms of the GNU General Public License, version 2, as 8aeaef83dSCédric Le Goater * published by the Free Software Foundation. 9aeaef83dSCédric Le Goater * 10aeaef83dSCédric Le Goater * This program is distributed in the hope that it will be useful, 11aeaef83dSCédric Le Goater * but WITHOUT ANY WARRANTY; without even the implied warranty of 12aeaef83dSCédric Le Goater * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13aeaef83dSCédric Le Goater * GNU General Public License for more details. 14aeaef83dSCédric Le Goater * 15aeaef83dSCédric Le Goater * You should have received a copy of the GNU General Public License 16aeaef83dSCédric Le Goater * along with this program; if not, see <http://www.gnu.org/licenses/>. 17aeaef83dSCédric Le Goater */ 18aeaef83dSCédric Le Goater 19aeaef83dSCédric Le Goater #include "qemu/osdep.h" 20aeaef83dSCédric Le Goater #include "hw/hw.h" 21aeaef83dSCédric Le Goater #include "sysemu/sysemu.h" 22aeaef83dSCédric Le Goater #include "target/ppc/cpu.h" 23aeaef83dSCédric Le Goater #include "qapi/error.h" 24aeaef83dSCédric Le Goater #include "qemu/log.h" 25aeaef83dSCédric Le Goater #include "hw/ipmi/ipmi.h" 26aeaef83dSCédric Le Goater #include "hw/ppc/fdt.h" 27aeaef83dSCédric Le Goater 28aeaef83dSCédric Le Goater #include "hw/ppc/pnv.h" 29aeaef83dSCédric Le Goater 30aeaef83dSCédric Le Goater #include <libfdt.h> 31aeaef83dSCédric Le Goater 32aeaef83dSCédric Le Goater /* TODO: include definition in ipmi.h */ 33aeaef83dSCédric Le Goater #define IPMI_SDR_FULL_TYPE 1 34aeaef83dSCédric Le Goater 35bce0b691SCédric Le Goater /* 36bce0b691SCédric Le Goater * OEM SEL Event data packet sent by BMC in response of a Read Event 37bce0b691SCédric Le Goater * Message Buffer command 38bce0b691SCédric Le Goater */ 39bce0b691SCédric Le Goater typedef struct OemSel { 40bce0b691SCédric Le Goater /* SEL header */ 41bce0b691SCédric Le Goater uint8_t id[2]; 42bce0b691SCédric Le Goater uint8_t type; 43bce0b691SCédric Le Goater uint8_t timestamp[4]; 44bce0b691SCédric Le Goater uint8_t manuf_id[3]; 45bce0b691SCédric Le Goater 46bce0b691SCédric Le Goater /* OEM SEL data (6 bytes) follows */ 47bce0b691SCédric Le Goater uint8_t netfun; 48bce0b691SCédric Le Goater uint8_t cmd; 49bce0b691SCédric Le Goater uint8_t data[4]; 50bce0b691SCédric Le Goater } OemSel; 51bce0b691SCédric Le Goater 52bce0b691SCédric Le Goater #define SOFT_OFF 0x00 53bce0b691SCédric Le Goater #define SOFT_REBOOT 0x01 54bce0b691SCédric Le Goater 55bce0b691SCédric Le Goater static void pnv_gen_oem_sel(IPMIBmc *bmc, uint8_t reboot) 56bce0b691SCédric Le Goater { 57bce0b691SCédric Le Goater /* IPMI SEL Event are 16 bytes long */ 58bce0b691SCédric Le Goater OemSel sel = { 59bce0b691SCédric Le Goater .id = { 0x55 , 0x55 }, 60bce0b691SCédric Le Goater .type = 0xC0, /* OEM */ 61bce0b691SCédric Le Goater .manuf_id = { 0x0, 0x0, 0x0 }, 62bce0b691SCédric Le Goater .timestamp = { 0x0, 0x0, 0x0, 0x0 }, 63bce0b691SCédric Le Goater .netfun = 0x3A, /* IBM */ 64bce0b691SCédric Le Goater .cmd = 0x04, /* AMI OEM SEL Power Notification */ 65bce0b691SCédric Le Goater .data = { reboot, 0xFF, 0xFF, 0xFF }, 66bce0b691SCédric Le Goater }; 67bce0b691SCédric Le Goater 68bce0b691SCédric Le Goater ipmi_bmc_gen_event(bmc, (uint8_t *) &sel, 0 /* do not log the event */); 69bce0b691SCédric Le Goater } 70bce0b691SCédric Le Goater 71bce0b691SCédric Le Goater void pnv_bmc_powerdown(IPMIBmc *bmc) 72bce0b691SCédric Le Goater { 73bce0b691SCédric Le Goater pnv_gen_oem_sel(bmc, SOFT_OFF); 74bce0b691SCédric Le Goater } 75bce0b691SCédric Le Goater 76*b168a138SCédric Le Goater void pnv_dt_bmc_sensors(IPMIBmc *bmc, void *fdt) 77aeaef83dSCédric Le Goater { 78aeaef83dSCédric Le Goater int offset; 79aeaef83dSCédric Le Goater int i; 80aeaef83dSCédric Le Goater const struct ipmi_sdr_compact *sdr; 81aeaef83dSCédric Le Goater uint16_t nextrec; 82aeaef83dSCédric Le Goater 83aeaef83dSCédric Le Goater offset = fdt_add_subnode(fdt, 0, "/bmc"); 84aeaef83dSCédric Le Goater _FDT(offset); 85aeaef83dSCédric Le Goater 86aeaef83dSCédric Le Goater _FDT((fdt_setprop_string(fdt, offset, "name", "bmc"))); 87aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, offset, "#address-cells", 0x1))); 88aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, offset, "#size-cells", 0x0))); 89aeaef83dSCédric Le Goater 90aeaef83dSCédric Le Goater offset = fdt_add_subnode(fdt, offset, "sensors"); 91aeaef83dSCédric Le Goater _FDT(offset); 92aeaef83dSCédric Le Goater 93aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, offset, "#address-cells", 0x1))); 94aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, offset, "#size-cells", 0x0))); 95aeaef83dSCédric Le Goater 96aeaef83dSCédric Le Goater for (i = 0; !ipmi_bmc_sdr_find(bmc, i, &sdr, &nextrec); i++) { 97aeaef83dSCédric Le Goater int off; 98aeaef83dSCédric Le Goater char *name; 99aeaef83dSCédric Le Goater 100aeaef83dSCédric Le Goater if (sdr->header.rec_type != IPMI_SDR_COMPACT_TYPE && 101aeaef83dSCédric Le Goater sdr->header.rec_type != IPMI_SDR_FULL_TYPE) { 102aeaef83dSCédric Le Goater continue; 103aeaef83dSCédric Le Goater } 104aeaef83dSCédric Le Goater 105aeaef83dSCédric Le Goater name = g_strdup_printf("sensor@%x", sdr->sensor_owner_number); 106aeaef83dSCédric Le Goater off = fdt_add_subnode(fdt, offset, name); 107aeaef83dSCédric Le Goater _FDT(off); 108aeaef83dSCédric Le Goater g_free(name); 109aeaef83dSCédric Le Goater 110aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, off, "reg", sdr->sensor_owner_number))); 111aeaef83dSCédric Le Goater _FDT((fdt_setprop_string(fdt, off, "name", "sensor"))); 112aeaef83dSCédric Le Goater _FDT((fdt_setprop_string(fdt, off, "compatible", "ibm,ipmi-sensor"))); 113aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, off, "ipmi-sensor-reading-type", 114aeaef83dSCédric Le Goater sdr->reading_type))); 115aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, off, "ipmi-entity-id", 116aeaef83dSCédric Le Goater sdr->entity_id))); 117aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, off, "ipmi-entity-instance", 118aeaef83dSCédric Le Goater sdr->entity_instance))); 119aeaef83dSCédric Le Goater _FDT((fdt_setprop_cell(fdt, off, "ipmi-sensor-type", 120aeaef83dSCédric Le Goater sdr->sensor_type))); 121aeaef83dSCédric Le Goater } 122aeaef83dSCédric Le Goater } 123