summaryrefslogtreecommitdiffstats
path: root/hw/lpc-mbox.c
diff options
context:
space:
mode:
authorCyril Bur <cyril.bur@au1.ibm.com>2017-02-22 20:30:59 +1100
committerStewart Smith <stewart@linux.vnet.ibm.com>2017-02-23 16:04:27 +1100
commit25ab8e2ee55136cfd9f864f65686e312fd38b667 (patch)
tree362926d4d1c72c909ca8ad769c04a82c79181aac /hw/lpc-mbox.c
parent271b00366a84cfb45e160ac90faee178035d67ac (diff)
downloadblackbird-skiboot-25ab8e2ee55136cfd9f864f65686e312fd38b667.tar.gz
blackbird-skiboot-25ab8e2ee55136cfd9f864f65686e312fd38b667.zip
hw/lpc-mbox: Add skiboot driver for the AST BMC mbox registers
This adds an internal skiboot API for accessing the AST BMC mbox registers. Signed-off-by: Cyril Bur <cyril.bur@au1.ibm.com> Signed-off-by: Michael Neuling <mikey@neuling.org> Signed-off-by: Stewart Smith <stewart@linux.vnet.ibm.com>
Diffstat (limited to 'hw/lpc-mbox.c')
-rw-r--r--hw/lpc-mbox.c280
1 files changed, 280 insertions, 0 deletions
diff --git a/hw/lpc-mbox.c b/hw/lpc-mbox.c
new file mode 100644
index 00000000..fb8852a5
--- /dev/null
+++ b/hw/lpc-mbox.c
@@ -0,0 +1,280 @@
+/* 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_ATTN (1 << 7)
+#define MBOX_STATUS_1 0x11
+#define MBOX_BMC_CTRL 0x12
+#define MBOX_CTRL_INT_STATUS (1 << 7)
+#define MBOX_CTRL_INT_MASK (1 << 1)
+#define MBOX_CTRL_INT_SEND (1 << 0)
+#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
+
+#define BMC_RESET 1
+#define BMC_COMPLETE 2
+
+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;
+ struct lock lock; /* Protect in_flight */
+ struct bmc_mbox_msg *in_flight;
+};
+
+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_DATA_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_DATA_REGS; i++)
+ bmc_mbox_outb(msg_data[i], i);
+
+ /* Ping */
+ prlog(PR_DEBUG, "Sending BMC interrupt\n");
+ bmc_mbox_outb(MBOX_CTRL_INT_SEND, MBOX_HOST_CTRL);
+}
+
+int bmc_mbox_enqueue(struct bmc_mbox_msg *msg)
+{
+ if (!mbox.base) {
+ prlog(PR_CRIT, "Using MBOX without init!\n");
+ return OPAL_WRONG_STATE;
+ }
+
+ lock(&mbox.lock);
+ if (mbox.in_flight) {
+ prlog(PR_DEBUG, "MBOX message already in flight\n");
+ unlock(&mbox.lock);
+ return OPAL_BUSY;
+ }
+
+ mbox.in_flight = msg;
+ unlock(&mbox.lock);
+
+ bmc_mbox_send_message(msg);
+
+ 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;
+
+ /* This is a 'registered' the message you just sent me */
+ if (bmc_mbox_inb(MBOX_HOST_CTRL) & MBOX_CTRL_INT_STATUS) {
+ /* W1C on that reg */
+ bmc_mbox_outb(MBOX_CTRL_INT_STATUS, MBOX_HOST_CTRL);
+
+ prlog(PR_INSANE, "Got a regular interrupt\n");
+ /*
+ * This should be safe lockless
+ */
+ msg = mbox.in_flight;
+ if (msg == NULL) {
+ prlog(PR_CRIT, "Couldn't find the message!!\n");
+ return;
+ }
+ bmc_mbox_recv_message(msg);
+ if (mbox.callback)
+ mbox.callback(msg, mbox.drv_data);
+ else
+ prlog(PR_ERR, "Detected NULL callback for mbox message\n");
+
+ /* Yeah we'll need locks here */
+ lock(&mbox.lock);
+ mbox.in_flight = NULL;
+ unlock(&mbox.lock);
+ }
+
+ /* This is to indicate that the BMC has information to tell us */
+ if (bmc_mbox_inb(MBOX_STATUS_1) & MBOX_STATUS_ATTN) {
+ uint8_t action;
+
+ /* W1C on that reg */
+ bmc_mbox_outb(MBOX_STATUS_ATTN, MBOX_STATUS_1);
+
+ action = bmc_mbox_inb(MBOX_FLAG_REG);
+ prlog(PR_INSANE, "Got a status register interrupt with action 0x%02x\n",
+ action);
+
+ if (action & BMC_RESET) {
+ /* TODO Freak */
+ prlog(PR_WARNING, "BMC reset detected\n");
+ action &= ~BMC_RESET;
+ }
+
+ if (action)
+ prlog(PR_ERR, "Got a status bit set that don't know about: 0x%02x\n",
+ action);
+ }
+
+ 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)
+{
+ /*
+ * Turns out there isn't anything to do.
+ * It might be a good idea to santise the registers though.
+ * TODO
+ */
+ 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;
+}
+
+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) {
+ prlog(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;
+ }
+
+ mbox.queue_len = 0;
+ mbox.in_flight = NULL;
+ mbox.callback = NULL;
+ mbox.drv_data = NULL;
+ 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);
+ prlog(PR_DEBUG, "Using chipid: %d and IRQ: %d at 0x%08x\n", chip_id, irq, mbox.base);
+}
+
+
OpenPOWER on IntegriCloud