355 lines
8.4 KiB
C
355 lines
8.4 KiB
C
/* Copyright 2017 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-MBOX: " fmt
|
|
|
|
#include <skiboot.h>
|
|
#include <lpc.h>
|
|
#include <console.h>
|
|
#include <opal.h>
|
|
#include <device.h>
|
|
#include <interrupts.h>
|
|
#include <processor.h>
|
|
#include <errorlog.h>
|
|
#include <trace.h>
|
|
#include <timebase.h>
|
|
#include <timer.h>
|
|
#include <cpu.h>
|
|
#include <chip.h>
|
|
#include <io.h>
|
|
|
|
#include <lpc-mbox.h>
|
|
|
|
#define MBOX_FLAG_REG 0x0f
|
|
#define MBOX_STATUS_0 0x10
|
|
#define MBOX_STATUS_1 0x11
|
|
#define MBOX_STATUS_1_ATTN (1 << 7)
|
|
#define MBOX_STATUS_1_RESP (1 << 5)
|
|
#define MBOX_BMC_CTRL 0x12
|
|
#define MBOX_CTRL_INT_STATUS (1 << 7)
|
|
#define MBOX_CTRL_INT_MASK (1 << 1)
|
|
#define MBOX_CTRL_INT_PING (1 << 0)
|
|
#define MBOX_CTRL_INT_SEND (MBOX_CTRL_INT_PING | MBOX_CTRL_INT_MASK)
|
|
#define MBOX_HOST_CTRL 0x13
|
|
#define MBOX_BMC_INT_EN_0 0x14
|
|
#define MBOX_BMC_INT_EN_1 0x15
|
|
#define MBOX_HOST_INT_EN_0 0x16
|
|
#define MBOX_HOST_INT_EN_1 0x17
|
|
|
|
#define MBOX_MAX_QUEUE_LEN 5
|
|
|
|
struct mbox {
|
|
uint32_t base;
|
|
int queue_len;
|
|
bool irq_ok;
|
|
uint8_t seq;
|
|
struct timer poller;
|
|
void (*callback)(struct bmc_mbox_msg *msg, void *priv);
|
|
void *drv_data;
|
|
void (*attn)(uint8_t bits, void *priv);
|
|
void *attn_data;
|
|
struct lock lock;
|
|
uint8_t sequence;
|
|
unsigned long timeout;
|
|
};
|
|
|
|
static struct mbox mbox;
|
|
|
|
/*
|
|
* MBOX accesses
|
|
*/
|
|
|
|
static void bmc_mbox_outb(uint8_t val, uint8_t reg)
|
|
{
|
|
lpc_outb(val, mbox.base + reg);
|
|
}
|
|
|
|
static uint8_t bmc_mbox_inb(uint8_t reg)
|
|
{
|
|
return lpc_inb(mbox.base + reg);
|
|
}
|
|
|
|
static void bmc_mbox_recv_message(struct bmc_mbox_msg *msg)
|
|
{
|
|
uint8_t *msg_data = (uint8_t *)msg;
|
|
int i;
|
|
|
|
for (i = 0; i < BMC_MBOX_READ_REGS; i++)
|
|
msg_data[i] = bmc_mbox_inb(i);
|
|
}
|
|
|
|
/* This needs work, don't write the data bytes that aren't needed */
|
|
static void bmc_mbox_send_message(struct bmc_mbox_msg *msg)
|
|
{
|
|
uint8_t *msg_data = (uint8_t *)msg;
|
|
int i;
|
|
|
|
if (!lpc_ok())
|
|
/* We're going to have to handle this better */
|
|
prlog(PR_ERR, "LPC isn't ok\n");
|
|
|
|
for (i = 0; i < BMC_MBOX_WRITE_REGS; i++)
|
|
bmc_mbox_outb(msg_data[i], i);
|
|
|
|
/*
|
|
* Don't touch the response byte - it's setup to generate an interrupt
|
|
* to the host (us) when written to, or the host status reg - we don't
|
|
* currently use it, or the BMC status reg - we're not allowed to.
|
|
*/
|
|
|
|
/* Ping */
|
|
prlog(PR_TRACE, "Sending BMC interrupt\n");
|
|
bmc_mbox_outb(MBOX_CTRL_INT_SEND, MBOX_HOST_CTRL);
|
|
}
|
|
|
|
int bmc_mbox_enqueue(struct bmc_mbox_msg *msg, unsigned int timeout_sec)
|
|
{
|
|
if (!mbox.base) {
|
|
prlog(PR_CRIT, "Using MBOX without init!\n");
|
|
return OPAL_WRONG_STATE;
|
|
}
|
|
|
|
lock(&mbox.lock);
|
|
if (mbox.timeout) {
|
|
prlog(PR_DEBUG, "MBOX message already in flight\n");
|
|
if (mftb() > mbox.timeout) {
|
|
prlog(PR_ERR, "In flight message dropped on the floor\n");
|
|
} else {
|
|
unlock(&mbox.lock);
|
|
return OPAL_BUSY;
|
|
}
|
|
}
|
|
|
|
mbox.timeout = mftb() + secs_to_tb(timeout_sec);
|
|
msg->seq = ++mbox.sequence;
|
|
|
|
bmc_mbox_send_message(msg);
|
|
unlock(&mbox.lock);
|
|
|
|
schedule_timer(&mbox.poller, mbox.irq_ok ?
|
|
TIMER_POLL : msecs_to_tb(MBOX_DEFAULT_POLL_MS));
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void mbox_poll(struct timer *t __unused, void *data __unused,
|
|
uint64_t now __unused)
|
|
{
|
|
struct bmc_mbox_msg msg;
|
|
|
|
if (!lpc_ok())
|
|
return;
|
|
|
|
/*
|
|
* This status bit being high means that someone touched the
|
|
* response byte (byte 13).
|
|
* There is probably a response for the previously sent commant
|
|
*/
|
|
lock(&mbox.lock);
|
|
if (bmc_mbox_inb(MBOX_STATUS_1) & MBOX_STATUS_1_RESP) {
|
|
/* W1C on that reg */
|
|
bmc_mbox_outb(MBOX_STATUS_1_RESP, MBOX_STATUS_1);
|
|
|
|
prlog(PR_INSANE, "Got a regular interrupt\n");
|
|
|
|
bmc_mbox_recv_message(&msg);
|
|
if (mbox.sequence != msg.seq) {
|
|
prlog(PR_ERR, "Got a response to a message we no longer care about\n");
|
|
goto out_response;
|
|
}
|
|
|
|
mbox.timeout = 0;
|
|
if (mbox.callback)
|
|
mbox.callback(&msg, mbox.drv_data);
|
|
else
|
|
prlog(PR_ERR, "Detected NULL callback for mbox message\n");
|
|
}
|
|
|
|
out_response:
|
|
|
|
/*
|
|
* The BMC has touched byte 15 to get our attention as it has
|
|
* something to tell us.
|
|
*/
|
|
if (bmc_mbox_inb(MBOX_STATUS_1) & MBOX_STATUS_1_ATTN) {
|
|
uint8_t action, all;
|
|
|
|
/* W1C on that reg */
|
|
bmc_mbox_outb(MBOX_STATUS_1_ATTN, MBOX_STATUS_1);
|
|
|
|
all = action = bmc_mbox_inb(MBOX_FLAG_REG);
|
|
prlog(PR_TRACE, "Got a status register interrupt with action 0x%02x\n",
|
|
action);
|
|
if (action & MBOX_ATTN_BMC_REBOOT) {
|
|
/*
|
|
* It's unlikely that something needs to be done at the
|
|
* driver level. Let libflash deal with it.
|
|
* Print something just in case, it is quite a signficant
|
|
* event.
|
|
*/
|
|
prlog(PR_WARNING, "BMC reset detected\n");
|
|
action &= ~MBOX_ATTN_BMC_REBOOT;
|
|
}
|
|
|
|
if (action & MBOX_ATTN_BMC_WINDOW_RESET)
|
|
action &= ~MBOX_ATTN_BMC_WINDOW_RESET;
|
|
|
|
if (action & MBOX_ATTN_BMC_FLASH_LOST)
|
|
action &= ~MBOX_ATTN_BMC_FLASH_LOST;
|
|
|
|
if (action & MBOX_ATTN_BMC_DAEMON_READY)
|
|
action &= ~MBOX_ATTN_BMC_DAEMON_READY;
|
|
|
|
if (action)
|
|
prlog(PR_ERR, "Got a status bit set that don't know about: 0x%02x\n",
|
|
action);
|
|
|
|
mbox.attn(all, mbox.attn_data);
|
|
}
|
|
|
|
unlock(&mbox.lock);
|
|
|
|
schedule_timer(&mbox.poller,
|
|
mbox.irq_ok ? TIMER_POLL : msecs_to_tb(MBOX_DEFAULT_POLL_MS));
|
|
}
|
|
|
|
static void mbox_irq(uint32_t chip_id __unused, uint32_t irq_mask __unused)
|
|
{
|
|
mbox.irq_ok = true;
|
|
mbox_poll(NULL, NULL, 0);
|
|
}
|
|
|
|
static struct lpc_client mbox_lpc_client = {
|
|
.interrupt = mbox_irq,
|
|
};
|
|
|
|
static bool mbox_init_hw(void)
|
|
{
|
|
/* Disable all status interrupts except attentions */
|
|
bmc_mbox_outb(0x00, MBOX_HOST_INT_EN_0);
|
|
bmc_mbox_outb(MBOX_STATUS_1_ATTN, MBOX_HOST_INT_EN_1);
|
|
|
|
/* Cleanup host interrupt and status */
|
|
bmc_mbox_outb(MBOX_CTRL_INT_STATUS, MBOX_HOST_CTRL);
|
|
|
|
/* Disable host control interrupt for now (will be
|
|
* re-enabled when needed). Clear BMC interrupts
|
|
*/
|
|
bmc_mbox_outb(MBOX_CTRL_INT_MASK, MBOX_BMC_CTRL);
|
|
|
|
return true;
|
|
}
|
|
|
|
int bmc_mbox_register_callback(void (*callback)(struct bmc_mbox_msg *msg, void *priv),
|
|
void *drv_data)
|
|
{
|
|
mbox.callback = callback;
|
|
mbox.drv_data = drv_data;
|
|
return 0;
|
|
}
|
|
|
|
int bmc_mbox_register_attn(void (*callback)(uint8_t bits, void *priv),
|
|
void *drv_data)
|
|
{
|
|
mbox.attn = callback;
|
|
mbox.attn_data = drv_data;
|
|
return 0;
|
|
}
|
|
|
|
uint8_t bmc_mbox_get_attn_reg(void)
|
|
{
|
|
return bmc_mbox_inb(MBOX_FLAG_REG);
|
|
}
|
|
|
|
void mbox_init(void)
|
|
{
|
|
const struct dt_property *prop;
|
|
struct dt_node *np;
|
|
uint32_t irq, chip_id;
|
|
|
|
if (mbox.base) {
|
|
prlog(PR_ERR, "Duplicate call to mbox_init()\n");
|
|
return;
|
|
}
|
|
|
|
prlog(PR_DEBUG, "Attempting mbox init\n");
|
|
np = dt_find_compatible_node(dt_root, NULL, "mbox");
|
|
if (!np) {
|
|
/* Only an ERROR on P9 and above, otherwise just
|
|
* a warning for someone doing development
|
|
*/
|
|
prlog((proc_gen <= proc_gen_p8) ? PR_DEBUG : PR_ERR,
|
|
"No device tree entry\n");
|
|
return;
|
|
}
|
|
|
|
/* Read the interrupts property if any */
|
|
irq = dt_prop_get_u32_def(np, "interrupts", 0);
|
|
if (!irq) {
|
|
prlog(PR_ERR, "No interrupts property\n");
|
|
return;
|
|
}
|
|
|
|
if (!lpc_present()) {
|
|
prlog(PR_ERR, "LPC not present\n");
|
|
return;
|
|
}
|
|
|
|
/* Get IO base */
|
|
prop = dt_find_property(np, "reg");
|
|
if (!prop) {
|
|
prlog(PR_ERR, "Can't find reg property\n");
|
|
return;
|
|
}
|
|
if (dt_property_get_cell(prop, 0) != OPAL_LPC_IO) {
|
|
prlog(PR_ERR, "Only supports IO addresses\n");
|
|
return;
|
|
}
|
|
mbox.base = dt_property_get_cell(prop, 1);
|
|
|
|
if (!mbox_init_hw()) {
|
|
prlog(PR_DEBUG, "Couldn't init HW\n");
|
|
return;
|
|
}
|
|
|
|
/* Disable the standard interrupt we don't care */
|
|
bmc_mbox_outb(MBOX_CTRL_INT_MASK, MBOX_HOST_CTRL);
|
|
|
|
/* Clear the status reg bits that we intend to use for interrupts */
|
|
/* W1C */
|
|
bmc_mbox_outb(MBOX_STATUS_1_RESP | MBOX_STATUS_1_ATTN, MBOX_STATUS_1);
|
|
|
|
mbox.queue_len = 0;
|
|
mbox.callback = NULL;
|
|
mbox.drv_data = NULL;
|
|
mbox.timeout = 0;
|
|
mbox.sequence = 0;
|
|
init_lock(&mbox.lock);
|
|
|
|
init_timer(&mbox.poller, mbox_poll, NULL);
|
|
|
|
chip_id = dt_get_chip_id(np);
|
|
mbox_lpc_client.interrupts = LPC_IRQ(irq);
|
|
lpc_register_client(chip_id, &mbox_lpc_client, IRQ_ATTR_TARGET_OPAL);
|
|
|
|
/* Enable interrupts */
|
|
bmc_mbox_outb(MBOX_STATUS_1_ATTN | MBOX_STATUS_1_RESP, MBOX_HOST_INT_EN_1);
|
|
|
|
prlog(PR_DEBUG, "Enabled on chip %d, IO port 0x%x, IRQ %d\n",
|
|
chip_id, mbox.base, irq);
|
|
}
|
|
|
|
|