historical/m0-applesillicon.git/xnu-qemu-arm64-5.1.0/roms/skiboot/hw/lpc.c

1402 lines
34 KiB
C
Raw Normal View History

2024-01-16 17:20:27 +00:00
/* Copyright 2013-2014 IBM Corp.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#define pr_fmt(fmt) "LPC: " fmt
#include <skiboot.h>
#include <xscom.h>
#include <io.h>
#include <lock.h>
#include <chip.h>
#include <lpc.h>
#include <timebase.h>
#include <errorlog.h>
#include <opal-api.h>
#include <platform.h>
#include <psi.h>
#include <interrupts.h>
//#define DBG_IRQ(fmt...) prerror(fmt)
#define DBG_IRQ(fmt...) do { } while(0)
DEFINE_LOG_ENTRY(OPAL_RC_LPC_READ, OPAL_PLATFORM_ERR_EVT, OPAL_LPC,
OPAL_MISC_SUBSYSTEM, OPAL_PREDICTIVE_ERR_GENERAL,
OPAL_NA);
DEFINE_LOG_ENTRY(OPAL_RC_LPC_WRITE, OPAL_PLATFORM_ERR_EVT, OPAL_LPC,
OPAL_MISC_SUBSYSTEM, OPAL_PREDICTIVE_ERR_GENERAL,
OPAL_NA);
DEFINE_LOG_ENTRY(OPAL_RC_LPC_SYNC, OPAL_PLATFORM_ERR_EVT, OPAL_LPC,
OPAL_MISC_SUBSYSTEM, OPAL_PREDICTIVE_ERR_GENERAL,
OPAL_NA);
/* Used exclusively in manufacturing mode */
DEFINE_LOG_ENTRY(OPAL_RC_LPC_SYNC_PERF, OPAL_PLATFORM_ERR_EVT, OPAL_LPC,
OPAL_MISC_SUBSYSTEM, OPAL_UNRECOVERABLE_ERR_DEGRADE_PERF,
OPAL_NA);
#define ECCB_CTL 0 /* b0020 -> b00200 */
#define ECCB_STAT 2 /* b0022 -> b00210 */
#define ECCB_DATA 3 /* b0023 -> b00218 */
#define ECCB_CTL_MAGIC 0xd000000000000000ul
#define ECCB_CTL_DATASZ PPC_BITMASK(4,7)
#define ECCB_CTL_READ PPC_BIT(15)
#define ECCB_CTL_ADDRLEN PPC_BITMASK(23,25)
#define ECCB_ADDRLEN_4B 0x4
#define ECCB_CTL_ADDR PPC_BITMASK(32,63)
#define ECCB_STAT_PIB_ERR PPC_BITMASK(0,5)
#define ECCB_STAT_RD_DATA PPC_BITMASK(6,37)
#define ECCB_STAT_BUSY PPC_BIT(44)
#define ECCB_STAT_ERRORS1 PPC_BITMASK(45,51)
#define ECCB_STAT_OP_DONE PPC_BIT(52)
#define ECCB_STAT_ERRORS2 PPC_BITMASK(53,55)
#define ECCB_STAT_ERR_MASK (ECCB_STAT_PIB_ERR | \
ECCB_STAT_ERRORS1 | \
ECCB_STAT_ERRORS2)
#define ECCB_TIMEOUT 1000000
/* OPB Master LS registers */
#define OPB_MASTER_LS_IRQ_STAT 0x50
#define OPB_MASTER_LS_IRQ_MASK 0x54
#define OPB_MASTER_LS_IRQ_POL 0x58
#define OPB_MASTER_IRQ_LPC 0x00000800
/* LPC HC registers */
#define LPC_HC_FW_SEG_IDSEL 0x24
#define LPC_HC_FW_RD_ACC_SIZE 0x28
#define LPC_HC_FW_RD_1B 0x00000000
#define LPC_HC_FW_RD_2B 0x01000000
#define LPC_HC_FW_RD_4B 0x02000000
#define LPC_HC_FW_RD_16B 0x04000000
#define LPC_HC_FW_RD_128B 0x07000000
#define LPC_HC_IRQSER_CTRL 0x30
#define LPC_HC_IRQSER_EN 0x80000000
#define LPC_HC_IRQSER_QMODE 0x40000000
#define LPC_HC_IRQSER_START_MASK 0x03000000
#define LPC_HC_IRQSER_START_4CLK 0x00000000
#define LPC_HC_IRQSER_START_6CLK 0x01000000
#define LPC_HC_IRQSER_START_8CLK 0x02000000
#define LPC_HC_IRQSER_AUTO_CLEAR 0x00800000
#define LPC_HC_IRQMASK 0x34 /* same bit defs as LPC_HC_IRQSTAT */
#define LPC_HC_IRQSTAT 0x38
#define LPC_HC_IRQ_SERIRQ0 0x80000000u /* all bits down to ... */
#define LPC_HC_IRQ_SERIRQ16 0x00008000 /* IRQ16=IOCHK#, IRQ2=SMI# */
#define LPC_HC_IRQ_SERIRQ_ALL 0xffff8000
#define LPC_HC_IRQ_LRESET 0x00000400
#define LPC_HC_IRQ_SYNC_ABNORM_ERR 0x00000080
#define LPC_HC_IRQ_SYNC_NORESP_ERR 0x00000040
#define LPC_HC_IRQ_SYNC_NORM_ERR 0x00000020
#define LPC_HC_IRQ_SYNC_TIMEOUT_ERR 0x00000010
#define LPC_HC_IRQ_TARG_TAR_ERR 0x00000008
#define LPC_HC_IRQ_BM_TAR_ERR 0x00000004
#define LPC_HC_IRQ_BM0_REQ 0x00000002
#define LPC_HC_IRQ_BM1_REQ 0x00000001
#define LPC_HC_IRQ_BASE_IRQS ( \
LPC_HC_IRQ_LRESET | \
LPC_HC_IRQ_SYNC_ABNORM_ERR | \
LPC_HC_IRQ_SYNC_NORESP_ERR | \
LPC_HC_IRQ_SYNC_NORM_ERR | \
LPC_HC_IRQ_SYNC_TIMEOUT_ERR | \
LPC_HC_IRQ_TARG_TAR_ERR | \
LPC_HC_IRQ_BM_TAR_ERR)
#define LPC_HC_ERROR_ADDRESS 0x40
#define LPC_NUM_SERIRQ 17
enum {
LPC_ROUTE_FREE = 0,
LPC_ROUTE_OPAL,
LPC_ROUTE_LINUX
};
struct lpc_error_entry {
int64_t rc;
const char *description;
};
struct lpcm {
uint32_t chip_id;
uint32_t xbase;
void *mbase;
struct lock lock;
uint8_t fw_idsel;
uint8_t fw_rdsz;
struct list_head clients;
bool has_serirq;
uint8_t sirq_routes[LPC_NUM_SERIRQ];
bool sirq_routed[LPC_NUM_SERIRQ];
uint32_t sirq_rmasks[4];
uint8_t sirq_ralloc[4];
struct dt_node *node;
};
#define LPC_BUS_DEGRADED_PERF_THRESHOLD 5
struct lpc_client_entry {
struct list_node node;
const struct lpc_client *clt;
uint32_t policy;
};
/* Default LPC bus */
static int32_t lpc_default_chip_id = -1;
static bool lpc_irqs_ready;
/*
* These are expected to be the same on all chips and should probably
* be read (or configured) dynamically. This is how things are configured
* today on Tuletta.
*/
static uint32_t lpc_io_opb_base = 0xd0010000;
static uint32_t lpc_mem_opb_base = 0xe0000000;
static uint32_t lpc_fw_opb_base = 0xf0000000;
static uint32_t lpc_reg_opb_base = 0xc0012000;
static uint32_t opb_master_reg_base = 0xc0010000;
static int64_t opb_mmio_write(struct lpcm *lpc, uint32_t addr, uint32_t data,
uint32_t sz)
{
switch (sz) {
case 1:
out_8(lpc->mbase + addr, data);
return OPAL_SUCCESS;
case 2:
out_be16(lpc->mbase + addr, data);
return OPAL_SUCCESS;
case 4:
out_be32(lpc->mbase + addr, data);
return OPAL_SUCCESS;
}
prerror("Invalid data size %d\n", sz);
return OPAL_PARAMETER;
}
static int64_t opb_write(struct lpcm *lpc, uint32_t addr, uint32_t data,
uint32_t sz)
{
uint64_t ctl = ECCB_CTL_MAGIC, stat;
int64_t rc, tout;
uint64_t data_reg;
if (lpc->mbase)
return opb_mmio_write(lpc, addr, data, sz);
switch(sz) {
case 1:
data_reg = ((uint64_t)data) << 56;
break;
case 2:
data_reg = ((uint64_t)data) << 48;
break;
case 4:
data_reg = ((uint64_t)data) << 32;
break;
default:
prerror("Invalid data size %d\n", sz);
return OPAL_PARAMETER;
}
rc = xscom_write(lpc->chip_id, lpc->xbase + ECCB_DATA, data_reg);
if (rc) {
log_simple_error(&e_info(OPAL_RC_LPC_WRITE),
"LPC: XSCOM write to ECCB DATA error %lld\n", rc);
return rc;
}
ctl = SETFIELD(ECCB_CTL_DATASZ, ctl, sz);
ctl = SETFIELD(ECCB_CTL_ADDRLEN, ctl, ECCB_ADDRLEN_4B);
ctl = SETFIELD(ECCB_CTL_ADDR, ctl, addr);
rc = xscom_write(lpc->chip_id, lpc->xbase + ECCB_CTL, ctl);
if (rc) {
log_simple_error(&e_info(OPAL_RC_LPC_WRITE),
"LPC: XSCOM write to ECCB CTL error %lld\n", rc);
return rc;
}
for (tout = 0; tout < ECCB_TIMEOUT; tout++) {
rc = xscom_read(lpc->chip_id, lpc->xbase + ECCB_STAT,
&stat);
if (rc) {
log_simple_error(&e_info(OPAL_RC_LPC_WRITE),
"LPC: XSCOM read from ECCB STAT err %lld\n",
rc);
return rc;
}
if (stat & ECCB_STAT_OP_DONE) {
if (stat & ECCB_STAT_ERR_MASK) {
log_simple_error(&e_info(OPAL_RC_LPC_WRITE),
"LPC: Error status: 0x%llx\n", stat);
return OPAL_HARDWARE;
}
return OPAL_SUCCESS;
}
time_wait_nopoll(100);
}
log_simple_error(&e_info(OPAL_RC_LPC_WRITE), "LPC: Write timeout !\n");
return OPAL_HARDWARE;
}
static int64_t opb_mmio_read(struct lpcm *lpc, uint32_t addr, uint32_t *data,
uint32_t sz)
{
switch (sz) {
case 1:
*data = in_8(lpc->mbase + addr);
return OPAL_SUCCESS;
case 2:
*data = in_be16(lpc->mbase + addr);
return OPAL_SUCCESS;
case 4:
*data = in_be32(lpc->mbase + addr);
return OPAL_SUCCESS;
}
prerror("Invalid data size %d\n", sz);
return OPAL_PARAMETER;
}
static int64_t opb_read(struct lpcm *lpc, uint32_t addr, uint32_t *data,
uint32_t sz)
{
uint64_t ctl = ECCB_CTL_MAGIC | ECCB_CTL_READ, stat;
int64_t rc, tout;
if (lpc->mbase)
return opb_mmio_read(lpc, addr, data, sz);
if (sz != 1 && sz != 2 && sz != 4) {
prerror("Invalid data size %d\n", sz);
return OPAL_PARAMETER;
}
ctl = SETFIELD(ECCB_CTL_DATASZ, ctl, sz);
ctl = SETFIELD(ECCB_CTL_ADDRLEN, ctl, ECCB_ADDRLEN_4B);
ctl = SETFIELD(ECCB_CTL_ADDR, ctl, addr);
rc = xscom_write(lpc->chip_id, lpc->xbase + ECCB_CTL, ctl);
if (rc) {
log_simple_error(&e_info(OPAL_RC_LPC_READ),
"LPC: XSCOM write to ECCB CTL error %lld\n", rc);
return rc;
}
for (tout = 0; tout < ECCB_TIMEOUT; tout++) {
rc = xscom_read(lpc->chip_id, lpc->xbase + ECCB_STAT,
&stat);
if (rc) {
log_simple_error(&e_info(OPAL_RC_LPC_READ),
"LPC: XSCOM read from ECCB STAT err %lld\n",
rc);
return rc;
}
if (stat & ECCB_STAT_OP_DONE) {
uint32_t rdata = GETFIELD(ECCB_STAT_RD_DATA, stat);
if (stat & ECCB_STAT_ERR_MASK) {
log_simple_error(&e_info(OPAL_RC_LPC_READ),
"LPC: Error status: 0x%llx\n", stat);
return OPAL_HARDWARE;
}
switch(sz) {
case 1:
*data = rdata >> 24;
break;
case 2:
*data = rdata >> 16;
break;
default:
*data = rdata;
break;
}
return 0;
}
time_wait_nopoll(100);
}
log_simple_error(&e_info(OPAL_RC_LPC_READ), "LPC: Read timeout !\n");
return OPAL_HARDWARE;
}
static int64_t lpc_set_fw_idsel(struct lpcm *lpc, uint8_t idsel)
{
uint32_t val;
int64_t rc;
if (idsel == lpc->fw_idsel)
return OPAL_SUCCESS;
if (idsel > 0xf)
return OPAL_PARAMETER;
rc = opb_read(lpc, lpc_reg_opb_base + LPC_HC_FW_SEG_IDSEL,
&val, 4);
if (rc) {
prerror("Failed to read HC_FW_SEG_IDSEL register !\n");
return rc;
}
val = (val & 0xfffffff0) | idsel;
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_FW_SEG_IDSEL,
val, 4);
if (rc) {
prerror("Failed to write HC_FW_SEG_IDSEL register !\n");
return rc;
}
lpc->fw_idsel = idsel;
return OPAL_SUCCESS;
}
static int64_t lpc_set_fw_rdsz(struct lpcm *lpc, uint8_t rdsz)
{
uint32_t val;
int64_t rc;
if (rdsz == lpc->fw_rdsz)
return OPAL_SUCCESS;
switch(rdsz) {
case 1:
val = LPC_HC_FW_RD_1B;
break;
case 2:
val = LPC_HC_FW_RD_2B;
break;
case 4:
val = LPC_HC_FW_RD_4B;
break;
default:
/*
* The HW supports 16 and 128 via a buffer/cache
* but I have never exprimented with it and am not
* sure it works the way we expect so let's leave it
* at that for now
*/
return OPAL_PARAMETER;
}
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_FW_RD_ACC_SIZE,
val, 4);
if (rc) {
prerror("Failed to write LPC_HC_FW_RD_ACC_SIZE !\n");
return rc;
}
lpc->fw_rdsz = rdsz;
return OPAL_SUCCESS;
}
static int64_t lpc_opb_prepare(struct lpcm *lpc,
enum OpalLPCAddressType addr_type,
uint32_t addr, uint32_t sz,
uint32_t *opb_base, bool is_write)
{
uint32_t top = addr + sz;
uint8_t fw_idsel;
int64_t rc;
/* Address wraparound */
if (top < addr)
return OPAL_PARAMETER;
/*
* Bound check access and get the OPB base address for
* the window corresponding to the access type
*/
switch(addr_type) {
case OPAL_LPC_IO:
/* IO space is 64K */
if (top > 0x10000)
return OPAL_PARAMETER;
/* And only supports byte accesses */
if (sz != 1)
return OPAL_PARAMETER;
*opb_base = lpc_io_opb_base;
break;
case OPAL_LPC_MEM:
/* MEM space is 256M */
if (top > 0x10000000)
return OPAL_PARAMETER;
/* And only supports byte accesses */
if (sz != 1)
return OPAL_PARAMETER;
*opb_base = lpc_mem_opb_base;
break;
case OPAL_LPC_FW:
/*
* FW space is in segments of 256M controlled
* by IDSEL, make sure we don't cross segments
*/
*opb_base = lpc_fw_opb_base;
fw_idsel = (addr >> 28);
if (((top - 1) >> 28) != fw_idsel)
return OPAL_PARAMETER;
/* Set segment */
rc = lpc_set_fw_idsel(lpc, fw_idsel);
if (rc)
return rc;
/* Set read access size */
if (!is_write) {
rc = lpc_set_fw_rdsz(lpc, sz);
if (rc)
return rc;
}
break;
default:
return OPAL_PARAMETER;
}
return OPAL_SUCCESS;
}
#define LPC_ERROR_IDX(x) (__builtin_ffs(x) - 1 - 2)
#define LPC_ERROR(_sts, _rc, _description) \
[LPC_ERROR_IDX(_sts)] = { _rc, _description }
static const struct lpc_error_entry lpc_error_table[] = {
LPC_ERROR(LPC_HC_IRQ_BM_TAR_ERR, OPAL_WRONG_STATE, "Got bus master TAR error."),
LPC_ERROR(LPC_HC_IRQ_TARG_TAR_ERR, OPAL_WRONG_STATE, "Got abnormal TAR error."),
LPC_ERROR(LPC_HC_IRQ_SYNC_TIMEOUT_ERR, OPAL_TIMEOUT, "Got SYNC timeout error."),
LPC_ERROR(LPC_HC_IRQ_SYNC_NORM_ERR, OPAL_WRONG_STATE, "Got SYNC normal error."),
LPC_ERROR(LPC_HC_IRQ_SYNC_NORESP_ERR, OPAL_HARDWARE, "Got SYNC no-response error."),
LPC_ERROR(LPC_HC_IRQ_SYNC_ABNORM_ERR, OPAL_WRONG_STATE, "Got SYNC abnormal error."),
};
static int64_t lpc_probe_prepare(struct lpcm *lpc)
{
const uint32_t irqmask_addr = lpc_reg_opb_base + LPC_HC_IRQMASK;
const uint32_t irqstat_addr = lpc_reg_opb_base + LPC_HC_IRQSTAT;
uint32_t irqmask;
int rc;
rc = opb_read(lpc, irqmask_addr, &irqmask, 4);
if (rc)
return rc;
irqmask &= ~LPC_HC_IRQ_SYNC_NORESP_ERR;
rc = opb_write(lpc, irqmask_addr, irqmask, 4);
if (rc)
return rc;
return opb_write(lpc, irqstat_addr, LPC_HC_IRQ_SYNC_NORESP_ERR, 4);
}
static int64_t lpc_probe_test(struct lpcm *lpc)
{
const uint32_t irqmask_addr = lpc_reg_opb_base + LPC_HC_IRQMASK;
const uint32_t irqstat_addr = lpc_reg_opb_base + LPC_HC_IRQSTAT;
uint32_t irqmask, irqstat;
int64_t idx;
int rc;
rc = opb_read(lpc, irqstat_addr, &irqstat, 4);
if (rc)
return rc;
rc = opb_write(lpc, irqstat_addr, LPC_HC_IRQ_SYNC_NORESP_ERR, 4);
if (rc)
return rc;
rc = opb_read(lpc, irqmask_addr, &irqmask, 4);
if (rc)
return rc;
irqmask |= LPC_HC_IRQ_SYNC_NORESP_ERR;
rc = opb_write(lpc, irqmask_addr, irqmask, 4);
if (rc)
return rc;
if (!(irqstat & LPC_HC_IRQ_BASE_IRQS))
return OPAL_SUCCESS;
/* Ensure we can perform a valid lookup in the error table */
idx = LPC_ERROR_IDX(irqstat);
if (idx < 0 || idx >= ARRAY_SIZE(lpc_error_table)) {
prerror("LPC bus error translation failed with status 0x%x\n",
irqstat);
return OPAL_PARAMETER;
}
rc = lpc_error_table[idx].rc;
return rc;
}
static int64_t __lpc_write(struct lpcm *lpc, enum OpalLPCAddressType addr_type,
uint32_t addr, uint32_t data, uint32_t sz,
bool probe)
{
uint32_t opb_base;
int64_t rc;
lock(&lpc->lock);
if (probe) {
rc = lpc_probe_prepare(lpc);
if (rc)
goto bail;
}
/*
* Convert to an OPB access and handle LPC HC configuration
* for FW accesses (IDSEL)
*/
rc = lpc_opb_prepare(lpc, addr_type, addr, sz, &opb_base, true);
if (rc)
goto bail;
/* Perform OPB access */
rc = opb_write(lpc, opb_base + addr, data, sz);
if (rc)
goto bail;
if (probe)
rc = lpc_probe_test(lpc);
bail:
unlock(&lpc->lock);
return rc;
}
static int64_t __lpc_write_sanity(enum OpalLPCAddressType addr_type,
uint32_t addr, uint32_t data, uint32_t sz,
bool probe)
{
struct proc_chip *chip;
if (lpc_default_chip_id < 0)
return OPAL_PARAMETER;
chip = get_chip(lpc_default_chip_id);
if (!chip || !chip->lpc)
return OPAL_PARAMETER;
return __lpc_write(chip->lpc, addr_type, addr, data, sz, probe);
}
int64_t lpc_write(enum OpalLPCAddressType addr_type, uint32_t addr,
uint32_t data, uint32_t sz)
{
return __lpc_write_sanity(addr_type, addr, data, sz, false);
}
int64_t lpc_probe_write(enum OpalLPCAddressType addr_type, uint32_t addr,
uint32_t data, uint32_t sz)
{
return __lpc_write_sanity(addr_type, addr, data, sz, true);
}
/*
* The "OPAL" variant add the emulation of 2 and 4 byte accesses using
* byte accesses for IO and MEM space in order to be compatible with
* existing Linux expectations
*/
static int64_t opal_lpc_write(uint32_t chip_id, enum OpalLPCAddressType addr_type,
uint32_t addr, uint32_t data, uint32_t sz)
{
struct proc_chip *chip;
int64_t rc;
chip = get_chip(chip_id);
if (!chip || !chip->lpc)
return OPAL_PARAMETER;
if (addr_type == OPAL_LPC_FW || sz == 1)
return __lpc_write(chip->lpc, addr_type, addr, data, sz, false);
while(sz--) {
rc = __lpc_write(chip->lpc, addr_type, addr, data & 0xff, 1, false);
if (rc)
return rc;
addr++;
data >>= 8;
}
return OPAL_SUCCESS;
}
static int64_t __lpc_read(struct lpcm *lpc, enum OpalLPCAddressType addr_type,
uint32_t addr, uint32_t *data, uint32_t sz,
bool probe)
{
uint32_t opb_base;
int64_t rc;
lock(&lpc->lock);
if (probe) {
rc = lpc_probe_prepare(lpc);
if (rc)
goto bail;
}
/*
* Convert to an OPB access and handle LPC HC configuration
* for FW accesses (IDSEL and read size)
*/
rc = lpc_opb_prepare(lpc, addr_type, addr, sz, &opb_base, false);
if (rc)
goto bail;
/* Perform OPB access */
rc = opb_read(lpc, opb_base + addr, data, sz);
if (rc)
goto bail;
if (probe)
rc = lpc_probe_test(lpc);
bail:
unlock(&lpc->lock);
return rc;
}
static int64_t __lpc_read_sanity(enum OpalLPCAddressType addr_type,
uint32_t addr, uint32_t *data, uint32_t sz,
bool probe)
{
struct proc_chip *chip;
if (lpc_default_chip_id < 0)
return OPAL_PARAMETER;
chip = get_chip(lpc_default_chip_id);
if (!chip || !chip->lpc)
return OPAL_PARAMETER;
return __lpc_read(chip->lpc, addr_type, addr, data, sz, probe);
}
int64_t lpc_read(enum OpalLPCAddressType addr_type, uint32_t addr,
uint32_t *data, uint32_t sz)
{
return __lpc_read_sanity(addr_type, addr, data, sz, false);
}
int64_t lpc_probe_read(enum OpalLPCAddressType addr_type, uint32_t addr,
uint32_t *data, uint32_t sz)
{
return __lpc_read_sanity(addr_type, addr, data, sz, true);
}
/*
* The "OPAL" variant add the emulation of 2 and 4 byte accesses using
* byte accesses for IO and MEM space in order to be compatible with
* existing Linux expectations
*/
static int64_t opal_lpc_read(uint32_t chip_id, enum OpalLPCAddressType addr_type,
uint32_t addr, uint32_t *data, uint32_t sz)
{
struct proc_chip *chip;
int64_t rc;
chip = get_chip(chip_id);
if (!chip || !chip->lpc)
return OPAL_PARAMETER;
if (addr_type == OPAL_LPC_FW || sz == 1)
return __lpc_read(chip->lpc, addr_type, addr, data, sz, false);
*data = 0;
while(sz--) {
uint32_t byte;
rc = __lpc_read(chip->lpc, addr_type, addr, &byte, 1, false);
if (rc)
return rc;
*data = *data | (byte << (8 * sz));
addr++;
}
return OPAL_SUCCESS;
}
bool lpc_present(void)
{
return lpc_default_chip_id >= 0;
}
/* Called with LPC lock held */
static void lpc_setup_serirq(struct lpcm *lpc)
{
struct lpc_client_entry *ent;
uint32_t mask = LPC_HC_IRQ_BASE_IRQS;
int rc;
if (!lpc_irqs_ready)
return;
/* Collect serirq enable bits */
list_for_each(&lpc->clients, ent, node)
mask |= ent->clt->interrupts & LPC_HC_IRQ_SERIRQ_ALL;
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQMASK, mask, 4);
if (rc) {
prerror("Failed to update irq mask\n");
return;
}
DBG_IRQ("IRQ mask set to 0x%08x\n", mask);
/* Enable the LPC interrupt in the OPB Master */
opb_write(lpc, opb_master_reg_base + OPB_MASTER_LS_IRQ_POL, 0, 4);
rc = opb_write(lpc, opb_master_reg_base + OPB_MASTER_LS_IRQ_MASK,
OPB_MASTER_IRQ_LPC, 4);
if (rc)
prerror("Failed to enable IRQs in OPB\n");
/* Check whether we should enable serirq */
if (mask & LPC_HC_IRQ_SERIRQ_ALL) {
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQSER_CTRL,
LPC_HC_IRQSER_EN |
LPC_HC_IRQSER_START_4CLK |
/*
* New mode bit for P9N DD2.0 (ignored otherwise)
* when set we no longer have to manually clear
* the SerIRQs on EOI.
*/
LPC_HC_IRQSER_AUTO_CLEAR, 4);
DBG_IRQ("SerIRQ enabled\n");
} else {
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQSER_CTRL,
0, 4);
DBG_IRQ("SerIRQ disabled\n");
}
if (rc)
prerror("Failed to configure SerIRQ\n");
{
u32 val;
rc = opb_read(lpc, lpc_reg_opb_base + LPC_HC_IRQMASK, &val, 4);
if (rc)
prerror("Failed to readback mask");
else
DBG_IRQ("MASK READBACK=%x\n", val);
rc = opb_read(lpc, lpc_reg_opb_base + LPC_HC_IRQSER_CTRL,
&val, 4);
if (rc)
prerror("Failed to readback ctrl");
else
DBG_IRQ("CTRL READBACK=%x\n", val);
}
}
static void lpc_route_serirq(struct lpcm *lpc, uint32_t sirq,
uint32_t psi_idx)
{
uint32_t reg, shift, val, psi_old;
int64_t rc;
psi_old = lpc->sirq_routes[sirq];
lpc->sirq_rmasks[psi_old] &= ~(LPC_HC_IRQ_SERIRQ0 >> sirq);
lpc->sirq_rmasks[psi_idx] |= (LPC_HC_IRQ_SERIRQ0 >> sirq);
lpc->sirq_routes[sirq] = psi_idx;
lpc->sirq_routed[sirq] = true;
/* We may not be ready yet ... */
if (!lpc->has_serirq)
return;
if (sirq < 14) {
reg = 0xc;
shift = 4 + (sirq << 1);
} else {
reg = 0x8;
shift = 8 + ((sirq - 14) << 1);
}
shift = 30-shift;
rc = opb_read(lpc, opb_master_reg_base + reg, &val, 4);
if (rc)
return;
val = val & ~(3 << shift);
val |= (psi_idx & 3) << shift;
opb_write(lpc, opb_master_reg_base + reg, val, 4);
}
static void lpc_alloc_route(struct lpcm *lpc, unsigned int irq,
unsigned int policy)
{
unsigned int i, r, c;
int route = -1;
if (policy == IRQ_ATTR_TARGET_OPAL)
r = LPC_ROUTE_OPAL;
else
r = LPC_ROUTE_LINUX;
prlog(PR_DEBUG, "Routing irq %d, policy: %d (r=%d)\n",
irq, policy, r);
/* Are we already routed ? */
if (lpc->sirq_routed[irq] &&
r != lpc->sirq_ralloc[lpc->sirq_routes[irq]]) {
prerror("irq %d has conflicting policies\n", irq);
return;
}
/* First try to find a free route. Leave one for another
* policy though
*/
for (i = 0, c = 0; i < 4; i++) {
/* Count routes with identical policy */
if (lpc->sirq_ralloc[i] == r)
c++;
/* Use the route if it's free and there is no more
* than 3 existing routes with that policy
*/
if (lpc->sirq_ralloc[i] == LPC_ROUTE_FREE && c < 4) {
lpc->sirq_ralloc[i] = r;
route = i;
break;
}
}
/* If we couldn't get a free one, try to find an existing one
* with a matching policy
*/
for (i = 0; route < 0 && i < 4; i++) {
if (lpc->sirq_ralloc[i] == r)
route = i;
}
/* Still no route ? bail. That should never happen */
if (route < 0) {
prerror("Can't find a route for irq %d\n", irq);
return;
}
/* Program route */
lpc_route_serirq(lpc, irq, route);
prlog(PR_DEBUG, "SerIRQ %d using route %d targetted at %s\n",
irq, route, r == LPC_ROUTE_LINUX ? "OS" : "OPAL");
}
unsigned int lpc_get_irq_policy(uint32_t chip_id, uint32_t psi_idx)
{
struct proc_chip *c = get_chip(chip_id);
if (!c || !c->lpc)
return IRQ_ATTR_TARGET_LINUX;
if (c->lpc->sirq_ralloc[psi_idx] == LPC_ROUTE_LINUX)
return IRQ_ATTR_TARGET_LINUX;
else
return IRQ_ATTR_TARGET_OPAL | IRQ_ATTR_TYPE_LSI;
}
static void lpc_create_int_map(struct lpcm *lpc, struct dt_node *psi_node)
{
uint32_t map[LPC_NUM_SERIRQ * 5], *pmap;
uint32_t i;
if (!psi_node)
return;
pmap = map;
for (i = 0; i < LPC_NUM_SERIRQ; i++) {
if (!lpc->sirq_routed[i])
continue;
*(pmap++) = 0;
*(pmap++) = 0;
*(pmap++) = i;
*(pmap++) = psi_node->phandle;
*(pmap++) = lpc->sirq_routes[i] + P9_PSI_IRQ_LPC_SIRQ0;
}
if (pmap == map)
return;
dt_add_property(lpc->node, "interrupt-map", map,
(pmap - map) * sizeof(uint32_t));
dt_add_property_cells(lpc->node, "interrupt-map-mask", 0, 0, 0xff);
dt_add_property_cells(lpc->node, "#interrupt-cells", 1);
}
void lpc_finalize_interrupts(void)
{
struct proc_chip *chip;
lpc_irqs_ready = true;
for_each_chip(chip) {
if (chip->lpc && chip->psi &&
(chip->type == PROC_CHIP_P9_NIMBUS ||
chip->type == PROC_CHIP_P9_CUMULUS))
lpc_create_int_map(chip->lpc, chip->psi->node);
}
}
static void lpc_init_interrupts_one(struct proc_chip *chip)
{
struct lpcm *lpc = chip->lpc;
int i, rc;
lock(&lpc->lock);
/* First mask them all */
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQMASK, 0, 4);
if (rc) {
prerror("Failed to init interrutps\n");
goto bail;
}
switch(chip->type) {
case PROC_CHIP_P8_MURANO:
case PROC_CHIP_P8_VENICE:
/* On Murano/Venice, there is no SerIRQ, only enable error
* interrupts
*/
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQMASK,
LPC_HC_IRQ_BASE_IRQS, 4);
if (rc) {
prerror("Failed to set interrupt mask\n");
goto bail;
}
opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQSER_CTRL, 0, 4);
break;
case PROC_CHIP_P8_NAPLES:
/* On Naples, we support LPC interrupts, enable them based
* on what clients requests. This will setup the mask and
* enable processing
*/
lpc->has_serirq = true;
lpc_setup_serirq(lpc);
break;
case PROC_CHIP_P9_NIMBUS:
case PROC_CHIP_P9_CUMULUS:
/* On P9, we additionally setup the routing. */
lpc->has_serirq = true;
for (i = 0; i < LPC_NUM_SERIRQ; i++) {
if (lpc->sirq_routed[i])
lpc_route_serirq(lpc, i, lpc->sirq_routes[i]);
}
lpc_setup_serirq(lpc);
break;
default:
;
}
bail:
unlock(&lpc->lock);
}
void lpc_init_interrupts(void)
{
struct proc_chip *chip;
lpc_irqs_ready = true;
for_each_chip(chip) {
if (chip->lpc)
lpc_init_interrupts_one(chip);
}
}
static void lpc_dispatch_reset(struct lpcm *lpc)
{
struct lpc_client_entry *ent;
/* XXX We are going to hit this repeatedly while reset is
* asserted which might be sub-optimal. We should instead
* detect assertion and start a poller that will wait for
* de-assertion. We could notify clients of LPC being
* on/off rather than just reset
*/
prerror("Got LPC reset on chip 0x%x !\n", lpc->chip_id);
/* Collect serirq enable bits */
list_for_each(&lpc->clients, ent, node) {
if (!ent->clt->reset)
continue;
unlock(&lpc->lock);
ent->clt->reset(lpc->chip_id);
lock(&lpc->lock);
}
/* Reconfigure serial interrupts */
if (lpc->has_serirq)
lpc_setup_serirq(lpc);
}
static void lpc_dispatch_err_irqs(struct lpcm *lpc, uint32_t irqs)
{
const struct lpc_error_entry *err;
static int lpc_bus_err_count;
struct opal_err_info *info;
uint32_t addr;
int64_t idx;
int rc;
/* Write back to clear error interrupts, we clear SerIRQ later
* as they are handled as level interrupts
*/
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQSTAT,
LPC_HC_IRQ_BASE_IRQS, 4);
if (rc)
prerror("Failed to clear IRQ error latches !\n");
if (irqs & LPC_HC_IRQ_LRESET) {
lpc_dispatch_reset(lpc);
return;
}
/* Ensure we can perform a valid lookup in the error table */
idx = LPC_ERROR_IDX(irqs);
if (idx < 0 || idx >= ARRAY_SIZE(lpc_error_table)) {
prerror("LPC bus error translation failed with status 0x%x\n",
irqs);
return;
}
/* Find and report the error */
err = &lpc_error_table[idx];
lpc_bus_err_count++;
if (manufacturing_mode && (lpc_bus_err_count > LPC_BUS_DEGRADED_PERF_THRESHOLD))
info = &e_info(OPAL_RC_LPC_SYNC_PERF);
else
info = &e_info(OPAL_RC_LPC_SYNC);
rc = opb_read(lpc, lpc_reg_opb_base + LPC_HC_ERROR_ADDRESS, &addr, 4);
if (rc)
log_simple_error(info, "LPC[%03x]: %s "
"Error reading error address register\n",
lpc->chip_id, err->description);
else
log_simple_error(info, "LPC[%03x]: %s Error address reg: "
"0x%08x\n",
lpc->chip_id, err->description, addr);
}
static void lpc_dispatch_ser_irqs(struct lpcm *lpc, uint32_t irqs,
bool clear_latch)
{
struct lpc_client_entry *ent;
uint32_t cirqs;
int rc;
irqs &= LPC_HC_IRQ_SERIRQ_ALL;
/* Collect serirq enable bits */
list_for_each(&lpc->clients, ent, node) {
if (!ent->clt->interrupt)
continue;
cirqs = ent->clt->interrupts & irqs;
if (cirqs) {
unlock(&lpc->lock);
ent->clt->interrupt(lpc->chip_id, cirqs);
lock(&lpc->lock);
}
}
/* Our SerIRQ are level sensitive, we clear the latch after
* we call the handler.
*/
if (!clear_latch)
return;
rc = opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQSTAT, irqs, 4);
if (rc)
prerror("Failed to clear SerIRQ latches !\n");
}
void lpc_interrupt(uint32_t chip_id)
{
struct proc_chip *chip = get_chip(chip_id);
struct lpcm *lpc;
uint32_t irqs, opb_irqs;
int rc;
/* No initialized LPC controller on that chip */
if (!chip || !chip->lpc)
return;
lpc = chip->lpc;
lock(&lpc->lock);
/* Grab OPB Master LS interrupt status */
rc = opb_read(lpc, opb_master_reg_base + OPB_MASTER_LS_IRQ_STAT,
&opb_irqs, 4);
if (rc) {
prerror("Failed to read OPB IRQ state\n");
unlock(&lpc->lock);
return;
}
DBG_IRQ("OPB IRQ on chip 0x%x, oirqs=0x%08x\n", chip_id, opb_irqs);
/* Check if it's an LPC interrupt */
if (!(opb_irqs & OPB_MASTER_IRQ_LPC)) {
/* Something we don't support ? Ack it anyway... */
goto bail;
}
/* Handle the lpc interrupt source (errors etc...) */
rc = opb_read(lpc, lpc_reg_opb_base + LPC_HC_IRQSTAT, &irqs, 4);
if (rc) {
prerror("Failed to read LPC IRQ state\n");
goto bail;
}
DBG_IRQ("LPC IRQ on chip 0x%x, irqs=0x%08x\n", chip_id, irqs);
/* Handle error interrupts */
if (irqs & LPC_HC_IRQ_BASE_IRQS)
lpc_dispatch_err_irqs(lpc, irqs);
/* Handle SerIRQ interrupts */
if (irqs & LPC_HC_IRQ_SERIRQ_ALL)
lpc_dispatch_ser_irqs(lpc, irqs, true);
bail:
/* Ack it at the OPB level */
opb_write(lpc, opb_master_reg_base + OPB_MASTER_LS_IRQ_STAT,
opb_irqs, 4);
unlock(&lpc->lock);
}
void lpc_serirq(uint32_t chip_id, uint32_t index)
{
struct proc_chip *chip = get_chip(chip_id);
struct lpcm *lpc;
uint32_t irqs, rmask;
int rc;
/* No initialized LPC controller on that chip */
if (!chip || !chip->lpc)
return;
lpc = chip->lpc;
lock(&lpc->lock);
/* Handle the lpc interrupt source (errors etc...) */
rc = opb_read(lpc, lpc_reg_opb_base + LPC_HC_IRQSTAT, &irqs, 4);
if (rc) {
prerror("Failed to read LPC IRQ state\n");
goto bail;
}
rmask = lpc->sirq_rmasks[index];
DBG_IRQ("IRQ on chip 0x%x, irqs=0x%08x rmask=0x%08x\n",
chip_id, irqs, rmask);
irqs &= rmask;
/*
* Handle SerIRQ interrupts. Don't clear the latch,
* it will be done in our special EOI callback if
* necessary on DD1
*/
if (irqs)
lpc_dispatch_ser_irqs(lpc, irqs, false);
bail:
unlock(&lpc->lock);
}
void lpc_all_interrupts(uint32_t chip_id)
{
struct proc_chip *chip = get_chip(chip_id);
struct lpcm *lpc;
/* No initialized LPC controller on that chip */
if (!chip || !chip->lpc)
return;
lpc = chip->lpc;
/* Dispatch all */
lock(&lpc->lock);
lpc_dispatch_ser_irqs(lpc, LPC_HC_IRQ_SERIRQ_ALL, false);
unlock(&lpc->lock);
}
static void lpc_init_chip_p8(struct dt_node *xn)
{
uint32_t gcid = dt_get_chip_id(xn);
struct proc_chip *chip;
struct lpcm *lpc;
chip = get_chip(gcid);
assert(chip);
lpc = zalloc(sizeof(struct lpcm));
assert(lpc);
lpc->chip_id = gcid;
lpc->xbase = dt_get_address(xn, 0, NULL);
lpc->fw_idsel = 0xff;
lpc->fw_rdsz = 0xff;
lpc->node = xn;
list_head_init(&lpc->clients);
init_lock(&lpc->lock);
if (lpc_default_chip_id < 0 ||
dt_has_node_property(xn, "primary", NULL)) {
lpc_default_chip_id = gcid;
}
/* Mask all interrupts for now */
opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQMASK, 0, 4);
printf("LPC[%03x]: Initialized, access via XSCOM @0x%x\n",
gcid, lpc->xbase);
dt_add_property(xn, "interrupt-controller", NULL, 0);
dt_add_property_cells(xn, "#interrupt-cells", 1);
assert(dt_prop_get_u32(xn, "#address-cells") == 2);
chip->lpc = lpc;
}
static void lpc_init_chip_p9(struct dt_node *opb_node)
{
uint32_t gcid = dt_get_chip_id(opb_node);
struct dt_node *lpc_node;
struct proc_chip *chip;
struct lpcm *lpc;
u64 addr;
u32 val;
chip = get_chip(gcid);
assert(chip);
/* Grab OPB base address */
addr = dt_prop_get_cell(opb_node, "ranges", 1);
addr <<= 32;
addr |= dt_prop_get_cell(opb_node, "ranges", 2);
/* Find the "lpc" child node */
lpc_node = dt_find_compatible_node(opb_node, NULL, "ibm,power9-lpc");
if (!lpc_node)
return;
lpc = zalloc(sizeof(struct lpcm));
assert(lpc);
lpc->chip_id = gcid;
lpc->mbase = (void *)addr;
lpc->fw_idsel = 0xff;
lpc->fw_rdsz = 0xff;
lpc->node = lpc_node;
list_head_init(&lpc->clients);
init_lock(&lpc->lock);
if (lpc_default_chip_id < 0 ||
dt_has_node_property(opb_node, "primary", NULL)) {
lpc_default_chip_id = gcid;
}
/* Mask all interrupts for now */
opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQMASK, 0, 4);
/* Clear any stale LPC bus errors */
opb_write(lpc, lpc_reg_opb_base + LPC_HC_IRQSTAT,
LPC_HC_IRQ_BASE_IRQS, 4);
/* Default with routing to PSI SerIRQ 0, this will be updated
* later when interrupts are initialized.
*/
opb_read(lpc, opb_master_reg_base + 8, &val, 4);
val &= 0xff03ffff;
opb_write(lpc, opb_master_reg_base + 8, val, 4);
opb_read(lpc, opb_master_reg_base + 0xc, &val, 4);
val &= 0xf0000000;
opb_write(lpc, opb_master_reg_base + 0xc, val, 4);
prlog(PR_INFO, "LPC[%03x]: Initialized\n", gcid);
prlog(PR_DEBUG,"access via MMIO @%p\n", lpc->mbase);
chip->lpc = lpc;
}
void lpc_init(void)
{
struct dt_node *xn;
bool has_lpc = false;
/* Look for P9 first as the DT is compatile for both 8 and 9 */
dt_for_each_compatible(dt_root, xn, "ibm,power9-lpcm-opb") {
lpc_init_chip_p9(xn);
has_lpc = true;
}
if (!has_lpc) {
dt_for_each_compatible(dt_root, xn, "ibm,power8-lpc") {
lpc_init_chip_p8(xn);
has_lpc = true;
}
}
if (lpc_default_chip_id >= 0)
prlog(PR_DEBUG, "Default bus on chip 0x%x\n",
lpc_default_chip_id);
if (has_lpc) {
opal_register(OPAL_LPC_WRITE, opal_lpc_write, 5);
opal_register(OPAL_LPC_READ, opal_lpc_read, 5);
}
}
void lpc_used_by_console(void)
{
struct proc_chip *chip;
xscom_used_by_console();
for_each_chip(chip) {
struct lpcm *lpc = chip->lpc;
if (lpc) {
lpc->lock.in_con_path = true;
lock(&lpc->lock);
unlock(&lpc->lock);
}
}
}
bool lpc_ok(void)
{
struct proc_chip *chip;
if (lpc_default_chip_id < 0)
return false;
if (!xscom_ok())
return false;
chip = get_chip(lpc_default_chip_id);
if (!chip->lpc)
return false;
return !lock_held_by_me(&chip->lpc->lock);
}
void lpc_register_client(uint32_t chip_id,
const struct lpc_client *clt,
uint32_t policy)
{
struct lpc_client_entry *ent;
struct proc_chip *chip;
struct lpcm *lpc;
bool has_routes;
chip = get_chip(chip_id);
assert(chip);
lpc = chip->lpc;
if (!lpc) {
prerror("Attempt to register client on bad chip 0x%x\n",
chip_id);
return;
}
has_routes =
chip->type == PROC_CHIP_P9_NIMBUS ||
chip->type == PROC_CHIP_P9_CUMULUS;
if (policy != IRQ_ATTR_TARGET_OPAL && !has_routes) {
prerror("Chip doesn't support OS interrupt policy\n");
return;
}
ent = malloc(sizeof(*ent));
assert(ent);
ent->clt = clt;
ent->policy = policy;
lock(&lpc->lock);
list_add(&lpc->clients, &ent->node);
if (has_routes) {
unsigned int i;
for (i = 0; i < LPC_NUM_SERIRQ; i++)
if (clt->interrupts & LPC_IRQ(i))
lpc_alloc_route(lpc, i, policy);
}
if (lpc->has_serirq)
lpc_setup_serirq(lpc);
unlock(&lpc->lock);
}