mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2025-01-20 23:28:03 +07:00
dc0f0a026d
When kcs_bmc_handle_event calls kcs_force_abort function to handle the not open (no user running) KCS channel transaction, the returned status value -ENODEV causes the low level IRQ handler indicating that the irq was not for him by returning IRQ_NONE. After some time, this IRQ will be treated to be spurious one, and the exception dump happens. irq 30: nobody cared (try booting with the "irqpoll" option) CPU: 0 PID: 0 Comm: swapper/0 Not tainted 4.10.15-npcm750 #1 Hardware name: NPCMX50 Chip family [<c010b264>] (unwind_backtrace) from [<c0106930>] (show_stack+0x20/0x24) [<c0106930>] (show_stack) from [<c03dad38>] (dump_stack+0x8c/0xa0) [<c03dad38>] (dump_stack) from [<c0168810>] (__report_bad_irq+0x3c/0xdc) [<c0168810>] (__report_bad_irq) from [<c0168c34>] (note_interrupt+0x29c/0x2ec) [<c0168c34>] (note_interrupt) from [<c0165c80>] (handle_irq_event_percpu+0x5c/0x68) [<c0165c80>] (handle_irq_event_percpu) from [<c0165cd4>] (handle_irq_event+0x48/0x6c) [<c0165cd4>] (handle_irq_event) from [<c0169664>] (handle_fasteoi_irq+0xc8/0x198) [<c0169664>] (handle_fasteoi_irq) from [<c016529c>] (__handle_domain_irq+0x90/0xe8) [<c016529c>] (__handle_domain_irq) from [<c01014bc>] (gic_handle_irq+0x58/0x9c) [<c01014bc>] (gic_handle_irq) from [<c010752c>] (__irq_svc+0x6c/0x90) Exception stack(0xc0a01de8 to 0xc0a01e30) 1de0: 00002080 c0a6fbc0 00000000 00000000 00000000 c096d294 1e00: 00000000 00000001 dc406400 f03ff100 00000082 c0a01e94 c0a6fbc0 c0a01e38 1e20: 00200102 c01015bc 60000113 ffffffff [<c010752c>] (__irq_svc) from [<c01015bc>] (__do_softirq+0xbc/0x358) [<c01015bc>] (__do_softirq) from [<c011c798>] (irq_exit+0xb8/0xec) [<c011c798>] (irq_exit) from [<c01652a0>] (__handle_domain_irq+0x94/0xe8) [<c01652a0>] (__handle_domain_irq) from [<c01014bc>] (gic_handle_irq+0x58/0x9c) [<c01014bc>] (gic_handle_irq) from [<c010752c>] (__irq_svc+0x6c/0x90) Exception stack(0xc0a01ef8 to 0xc0a01f40) 1ee0: 00000000 000003ae 1f00: dcc0f338 c0111060 c0a00000 c0a0cc44 c0a0cbe4 c0a1c22b c07bc218 00000001 1f20: dcffca40 c0a01f54 c0a01f58 c0a01f48 c0103524 c0103528 60000013 ffffffff [<c010752c>] (__irq_svc) from [<c0103528>] (arch_cpu_idle+0x48/0x4c) [<c0103528>] (arch_cpu_idle) from [<c0681390>] (default_idle_call+0x30/0x3c) [<c0681390>] (default_idle_call) from [<c0156f24>] (do_idle+0xc8/0x134) [<c0156f24>] (do_idle) from [<c015722c>] (cpu_startup_entry+0x28/0x2c) [<c015722c>] (cpu_startup_entry) from [<c067ad74>] (rest_init+0x84/0x88) [<c067ad74>] (rest_init) from [<c0900d44>] (start_kernel+0x388/0x394) [<c0900d44>] (start_kernel) from [<0000807c>] (0x807c) handlers: [<c041c5dc>] npcm7xx_kcs_irq Disabling IRQ #30 It needs to change the returned status from -ENODEV to 0. The -ENODEV was originally used to tell the low level IRQ handler that no user was running, but not consider the IRQ handling desgin. And multiple KCS channels share one IRQ handler, it needs to check the IBF flag before doing force abort. If the IBF is set, after handling, return 0 to low level IRQ handler to indicate that the IRQ is handled. Signed-off-by: Haiyue Wang <haiyue.wang@linux.intel.com> Signed-off-by: Corey Minyard <cminyard@mvista.com>
457 lines
10 KiB
C
457 lines
10 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (c) 2015-2018, Intel Corporation.
|
|
*/
|
|
|
|
#define pr_fmt(fmt) "kcs-bmc: " fmt
|
|
|
|
#include <linux/errno.h>
|
|
#include <linux/io.h>
|
|
#include <linux/ipmi_bmc.h>
|
|
#include <linux/module.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/poll.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/slab.h>
|
|
|
|
#include "kcs_bmc.h"
|
|
|
|
#define KCS_MSG_BUFSIZ 1000
|
|
|
|
#define KCS_ZERO_DATA 0
|
|
|
|
|
|
/* IPMI 2.0 - Table 9-1, KCS Interface Status Register Bits */
|
|
#define KCS_STATUS_STATE(state) (state << 6)
|
|
#define KCS_STATUS_STATE_MASK GENMASK(7, 6)
|
|
#define KCS_STATUS_CMD_DAT BIT(3)
|
|
#define KCS_STATUS_SMS_ATN BIT(2)
|
|
#define KCS_STATUS_IBF BIT(1)
|
|
#define KCS_STATUS_OBF BIT(0)
|
|
|
|
/* IPMI 2.0 - Table 9-2, KCS Interface State Bits */
|
|
enum kcs_states {
|
|
IDLE_STATE = 0,
|
|
READ_STATE = 1,
|
|
WRITE_STATE = 2,
|
|
ERROR_STATE = 3,
|
|
};
|
|
|
|
/* IPMI 2.0 - Table 9-3, KCS Interface Control Codes */
|
|
#define KCS_CMD_GET_STATUS_ABORT 0x60
|
|
#define KCS_CMD_WRITE_START 0x61
|
|
#define KCS_CMD_WRITE_END 0x62
|
|
#define KCS_CMD_READ_BYTE 0x68
|
|
|
|
static inline u8 read_data(struct kcs_bmc *kcs_bmc)
|
|
{
|
|
return kcs_bmc->io_inputb(kcs_bmc, kcs_bmc->ioreg.idr);
|
|
}
|
|
|
|
static inline void write_data(struct kcs_bmc *kcs_bmc, u8 data)
|
|
{
|
|
kcs_bmc->io_outputb(kcs_bmc, kcs_bmc->ioreg.odr, data);
|
|
}
|
|
|
|
static inline u8 read_status(struct kcs_bmc *kcs_bmc)
|
|
{
|
|
return kcs_bmc->io_inputb(kcs_bmc, kcs_bmc->ioreg.str);
|
|
}
|
|
|
|
static inline void write_status(struct kcs_bmc *kcs_bmc, u8 data)
|
|
{
|
|
kcs_bmc->io_outputb(kcs_bmc, kcs_bmc->ioreg.str, data);
|
|
}
|
|
|
|
static void update_status_bits(struct kcs_bmc *kcs_bmc, u8 mask, u8 val)
|
|
{
|
|
u8 tmp = read_status(kcs_bmc);
|
|
|
|
tmp &= ~mask;
|
|
tmp |= val & mask;
|
|
|
|
write_status(kcs_bmc, tmp);
|
|
}
|
|
|
|
static inline void set_state(struct kcs_bmc *kcs_bmc, u8 state)
|
|
{
|
|
update_status_bits(kcs_bmc, KCS_STATUS_STATE_MASK,
|
|
KCS_STATUS_STATE(state));
|
|
}
|
|
|
|
static void kcs_force_abort(struct kcs_bmc *kcs_bmc)
|
|
{
|
|
set_state(kcs_bmc, ERROR_STATE);
|
|
read_data(kcs_bmc);
|
|
write_data(kcs_bmc, KCS_ZERO_DATA);
|
|
|
|
kcs_bmc->phase = KCS_PHASE_ERROR;
|
|
kcs_bmc->data_in_avail = false;
|
|
kcs_bmc->data_in_idx = 0;
|
|
}
|
|
|
|
static void kcs_bmc_handle_data(struct kcs_bmc *kcs_bmc)
|
|
{
|
|
u8 data;
|
|
|
|
switch (kcs_bmc->phase) {
|
|
case KCS_PHASE_WRITE_START:
|
|
kcs_bmc->phase = KCS_PHASE_WRITE_DATA;
|
|
/* fall through */
|
|
|
|
case KCS_PHASE_WRITE_DATA:
|
|
if (kcs_bmc->data_in_idx < KCS_MSG_BUFSIZ) {
|
|
set_state(kcs_bmc, WRITE_STATE);
|
|
write_data(kcs_bmc, KCS_ZERO_DATA);
|
|
kcs_bmc->data_in[kcs_bmc->data_in_idx++] =
|
|
read_data(kcs_bmc);
|
|
} else {
|
|
kcs_force_abort(kcs_bmc);
|
|
kcs_bmc->error = KCS_LENGTH_ERROR;
|
|
}
|
|
break;
|
|
|
|
case KCS_PHASE_WRITE_END_CMD:
|
|
if (kcs_bmc->data_in_idx < KCS_MSG_BUFSIZ) {
|
|
set_state(kcs_bmc, READ_STATE);
|
|
kcs_bmc->data_in[kcs_bmc->data_in_idx++] =
|
|
read_data(kcs_bmc);
|
|
kcs_bmc->phase = KCS_PHASE_WRITE_DONE;
|
|
kcs_bmc->data_in_avail = true;
|
|
wake_up_interruptible(&kcs_bmc->queue);
|
|
} else {
|
|
kcs_force_abort(kcs_bmc);
|
|
kcs_bmc->error = KCS_LENGTH_ERROR;
|
|
}
|
|
break;
|
|
|
|
case KCS_PHASE_READ:
|
|
if (kcs_bmc->data_out_idx == kcs_bmc->data_out_len)
|
|
set_state(kcs_bmc, IDLE_STATE);
|
|
|
|
data = read_data(kcs_bmc);
|
|
if (data != KCS_CMD_READ_BYTE) {
|
|
set_state(kcs_bmc, ERROR_STATE);
|
|
write_data(kcs_bmc, KCS_ZERO_DATA);
|
|
break;
|
|
}
|
|
|
|
if (kcs_bmc->data_out_idx == kcs_bmc->data_out_len) {
|
|
write_data(kcs_bmc, KCS_ZERO_DATA);
|
|
kcs_bmc->phase = KCS_PHASE_IDLE;
|
|
break;
|
|
}
|
|
|
|
write_data(kcs_bmc,
|
|
kcs_bmc->data_out[kcs_bmc->data_out_idx++]);
|
|
break;
|
|
|
|
case KCS_PHASE_ABORT_ERROR1:
|
|
set_state(kcs_bmc, READ_STATE);
|
|
read_data(kcs_bmc);
|
|
write_data(kcs_bmc, kcs_bmc->error);
|
|
kcs_bmc->phase = KCS_PHASE_ABORT_ERROR2;
|
|
break;
|
|
|
|
case KCS_PHASE_ABORT_ERROR2:
|
|
set_state(kcs_bmc, IDLE_STATE);
|
|
read_data(kcs_bmc);
|
|
write_data(kcs_bmc, KCS_ZERO_DATA);
|
|
kcs_bmc->phase = KCS_PHASE_IDLE;
|
|
break;
|
|
|
|
default:
|
|
kcs_force_abort(kcs_bmc);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void kcs_bmc_handle_cmd(struct kcs_bmc *kcs_bmc)
|
|
{
|
|
u8 cmd;
|
|
|
|
set_state(kcs_bmc, WRITE_STATE);
|
|
write_data(kcs_bmc, KCS_ZERO_DATA);
|
|
|
|
cmd = read_data(kcs_bmc);
|
|
switch (cmd) {
|
|
case KCS_CMD_WRITE_START:
|
|
kcs_bmc->phase = KCS_PHASE_WRITE_START;
|
|
kcs_bmc->error = KCS_NO_ERROR;
|
|
kcs_bmc->data_in_avail = false;
|
|
kcs_bmc->data_in_idx = 0;
|
|
break;
|
|
|
|
case KCS_CMD_WRITE_END:
|
|
if (kcs_bmc->phase != KCS_PHASE_WRITE_DATA) {
|
|
kcs_force_abort(kcs_bmc);
|
|
break;
|
|
}
|
|
|
|
kcs_bmc->phase = KCS_PHASE_WRITE_END_CMD;
|
|
break;
|
|
|
|
case KCS_CMD_GET_STATUS_ABORT:
|
|
if (kcs_bmc->error == KCS_NO_ERROR)
|
|
kcs_bmc->error = KCS_ABORTED_BY_COMMAND;
|
|
|
|
kcs_bmc->phase = KCS_PHASE_ABORT_ERROR1;
|
|
kcs_bmc->data_in_avail = false;
|
|
kcs_bmc->data_in_idx = 0;
|
|
break;
|
|
|
|
default:
|
|
kcs_force_abort(kcs_bmc);
|
|
kcs_bmc->error = KCS_ILLEGAL_CONTROL_CODE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
int kcs_bmc_handle_event(struct kcs_bmc *kcs_bmc)
|
|
{
|
|
unsigned long flags;
|
|
int ret = -ENODATA;
|
|
u8 status;
|
|
|
|
spin_lock_irqsave(&kcs_bmc->lock, flags);
|
|
|
|
status = read_status(kcs_bmc);
|
|
if (status & KCS_STATUS_IBF) {
|
|
if (!kcs_bmc->running)
|
|
kcs_force_abort(kcs_bmc);
|
|
else if (status & KCS_STATUS_CMD_DAT)
|
|
kcs_bmc_handle_cmd(kcs_bmc);
|
|
else
|
|
kcs_bmc_handle_data(kcs_bmc);
|
|
|
|
ret = 0;
|
|
}
|
|
|
|
spin_unlock_irqrestore(&kcs_bmc->lock, flags);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL(kcs_bmc_handle_event);
|
|
|
|
static inline struct kcs_bmc *to_kcs_bmc(struct file *filp)
|
|
{
|
|
return container_of(filp->private_data, struct kcs_bmc, miscdev);
|
|
}
|
|
|
|
static int kcs_bmc_open(struct inode *inode, struct file *filp)
|
|
{
|
|
struct kcs_bmc *kcs_bmc = to_kcs_bmc(filp);
|
|
int ret = 0;
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
if (!kcs_bmc->running)
|
|
kcs_bmc->running = 1;
|
|
else
|
|
ret = -EBUSY;
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static __poll_t kcs_bmc_poll(struct file *filp, poll_table *wait)
|
|
{
|
|
struct kcs_bmc *kcs_bmc = to_kcs_bmc(filp);
|
|
__poll_t mask = 0;
|
|
|
|
poll_wait(filp, &kcs_bmc->queue, wait);
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
if (kcs_bmc->data_in_avail)
|
|
mask |= EPOLLIN;
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
return mask;
|
|
}
|
|
|
|
static ssize_t kcs_bmc_read(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct kcs_bmc *kcs_bmc = to_kcs_bmc(filp);
|
|
bool data_avail;
|
|
size_t data_len;
|
|
ssize_t ret;
|
|
|
|
if (!(filp->f_flags & O_NONBLOCK))
|
|
wait_event_interruptible(kcs_bmc->queue,
|
|
kcs_bmc->data_in_avail);
|
|
|
|
mutex_lock(&kcs_bmc->mutex);
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
data_avail = kcs_bmc->data_in_avail;
|
|
if (data_avail) {
|
|
data_len = kcs_bmc->data_in_idx;
|
|
memcpy(kcs_bmc->kbuffer, kcs_bmc->data_in, data_len);
|
|
}
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
if (!data_avail) {
|
|
ret = -EAGAIN;
|
|
goto out_unlock;
|
|
}
|
|
|
|
if (count < data_len) {
|
|
pr_err("channel=%u with too large data : %zu\n",
|
|
kcs_bmc->channel, data_len);
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
kcs_force_abort(kcs_bmc);
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
ret = -EOVERFLOW;
|
|
goto out_unlock;
|
|
}
|
|
|
|
if (copy_to_user(buf, kcs_bmc->kbuffer, data_len)) {
|
|
ret = -EFAULT;
|
|
goto out_unlock;
|
|
}
|
|
|
|
ret = data_len;
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
if (kcs_bmc->phase == KCS_PHASE_WRITE_DONE) {
|
|
kcs_bmc->phase = KCS_PHASE_WAIT_READ;
|
|
kcs_bmc->data_in_avail = false;
|
|
kcs_bmc->data_in_idx = 0;
|
|
} else {
|
|
ret = -EAGAIN;
|
|
}
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
out_unlock:
|
|
mutex_unlock(&kcs_bmc->mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t kcs_bmc_write(struct file *filp, const char __user *buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct kcs_bmc *kcs_bmc = to_kcs_bmc(filp);
|
|
ssize_t ret;
|
|
|
|
/* a minimum response size '3' : netfn + cmd + ccode */
|
|
if (count < 3 || count > KCS_MSG_BUFSIZ)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&kcs_bmc->mutex);
|
|
|
|
if (copy_from_user(kcs_bmc->kbuffer, buf, count)) {
|
|
ret = -EFAULT;
|
|
goto out_unlock;
|
|
}
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
if (kcs_bmc->phase == KCS_PHASE_WAIT_READ) {
|
|
kcs_bmc->phase = KCS_PHASE_READ;
|
|
kcs_bmc->data_out_idx = 1;
|
|
kcs_bmc->data_out_len = count;
|
|
memcpy(kcs_bmc->data_out, kcs_bmc->kbuffer, count);
|
|
write_data(kcs_bmc, kcs_bmc->data_out[0]);
|
|
ret = count;
|
|
} else {
|
|
ret = -EINVAL;
|
|
}
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
out_unlock:
|
|
mutex_unlock(&kcs_bmc->mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static long kcs_bmc_ioctl(struct file *filp, unsigned int cmd,
|
|
unsigned long arg)
|
|
{
|
|
struct kcs_bmc *kcs_bmc = to_kcs_bmc(filp);
|
|
long ret = 0;
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
|
|
switch (cmd) {
|
|
case IPMI_BMC_IOCTL_SET_SMS_ATN:
|
|
update_status_bits(kcs_bmc, KCS_STATUS_SMS_ATN,
|
|
KCS_STATUS_SMS_ATN);
|
|
break;
|
|
|
|
case IPMI_BMC_IOCTL_CLEAR_SMS_ATN:
|
|
update_status_bits(kcs_bmc, KCS_STATUS_SMS_ATN,
|
|
0);
|
|
break;
|
|
|
|
case IPMI_BMC_IOCTL_FORCE_ABORT:
|
|
kcs_force_abort(kcs_bmc);
|
|
break;
|
|
|
|
default:
|
|
ret = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int kcs_bmc_release(struct inode *inode, struct file *filp)
|
|
{
|
|
struct kcs_bmc *kcs_bmc = to_kcs_bmc(filp);
|
|
|
|
spin_lock_irq(&kcs_bmc->lock);
|
|
kcs_bmc->running = 0;
|
|
kcs_force_abort(kcs_bmc);
|
|
spin_unlock_irq(&kcs_bmc->lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct file_operations kcs_bmc_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = kcs_bmc_open,
|
|
.read = kcs_bmc_read,
|
|
.write = kcs_bmc_write,
|
|
.release = kcs_bmc_release,
|
|
.poll = kcs_bmc_poll,
|
|
.unlocked_ioctl = kcs_bmc_ioctl,
|
|
};
|
|
|
|
struct kcs_bmc *kcs_bmc_alloc(struct device *dev, int sizeof_priv, u32 channel)
|
|
{
|
|
struct kcs_bmc *kcs_bmc;
|
|
|
|
kcs_bmc = devm_kzalloc(dev, sizeof(*kcs_bmc) + sizeof_priv, GFP_KERNEL);
|
|
if (!kcs_bmc)
|
|
return NULL;
|
|
|
|
dev_set_name(dev, "ipmi-kcs%u", channel);
|
|
|
|
spin_lock_init(&kcs_bmc->lock);
|
|
kcs_bmc->channel = channel;
|
|
|
|
mutex_init(&kcs_bmc->mutex);
|
|
init_waitqueue_head(&kcs_bmc->queue);
|
|
|
|
kcs_bmc->data_in = devm_kmalloc(dev, KCS_MSG_BUFSIZ, GFP_KERNEL);
|
|
kcs_bmc->data_out = devm_kmalloc(dev, KCS_MSG_BUFSIZ, GFP_KERNEL);
|
|
kcs_bmc->kbuffer = devm_kmalloc(dev, KCS_MSG_BUFSIZ, GFP_KERNEL);
|
|
if (!kcs_bmc->data_in || !kcs_bmc->data_out || !kcs_bmc->kbuffer)
|
|
return NULL;
|
|
|
|
kcs_bmc->miscdev.minor = MISC_DYNAMIC_MINOR;
|
|
kcs_bmc->miscdev.name = dev_name(dev);
|
|
kcs_bmc->miscdev.fops = &kcs_bmc_fops;
|
|
|
|
return kcs_bmc;
|
|
}
|
|
EXPORT_SYMBOL(kcs_bmc_alloc);
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
MODULE_AUTHOR("Haiyue Wang <haiyue.wang@linux.intel.com>");
|
|
MODULE_DESCRIPTION("KCS BMC to handle the IPMI request from system software");
|