Commit 54b34aa0 authored by Mika Westerberg's avatar Mika Westerberg Committed by Lee Jones

platform/x86: intel_scu_ipc: Split out SCU IPC functionality from the SCU driver

The SCU IPC functionality is usable outside of Intel MID devices. For
example modern Intel CPUs include the same thing but now it is called
PMC (Power Management Controller) instead of SCU. To make the IPC
available for those split the driver into core part (intel_scu_ipc.c)
and the SCU PCI driver part (intel_scu_pcidrv.c) which then calls the
former before it goes and creates rest of the SCU devices. The SCU IPC
will also register a new class that gets assigned to the device that is
created under the parent PCI device.

We also split the Kconfig symbols so that INTEL_SCU_IPC enables the SCU
IPC library and INTEL_SCU_PCI the SCU driver and convert the users
accordingly. While there remove default y from the INTEL_SCU_PCI symbol
as it is already selected by X86_INTEL_MID.
Signed-off-by: default avatarMika Westerberg <mika.westerberg@linux.intel.com>
Reviewed-by: default avatarAndy Shevchenko <andriy.shevchenko@linux.intel.com>
Signed-off-by: default avatarLee Jones <lee.jones@linaro.org>
parent 8f3d9f35
...@@ -595,7 +595,7 @@ config X86_INTEL_MID ...@@ -595,7 +595,7 @@ config X86_INTEL_MID
select I2C select I2C
select DW_APB_TIMER select DW_APB_TIMER
select APB_TIMER select APB_TIMER
select INTEL_SCU_IPC select INTEL_SCU_PCI
select MFD_INTEL_MSIC select MFD_INTEL_MSIC
---help--- ---help---
Select to build a kernel capable of supporting Intel MID (Mobile Select to build a kernel capable of supporting Intel MID (Mobile
......
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
#ifndef _ASM_X86_INTEL_SCU_IPC_H_ #ifndef _ASM_X86_INTEL_SCU_IPC_H_
#define _ASM_X86_INTEL_SCU_IPC_H_ #define _ASM_X86_INTEL_SCU_IPC_H_
#include <linux/ioport.h>
#include <linux/notifier.h> #include <linux/notifier.h>
#define IPCMSG_INDIRECT_READ 0x02 #define IPCMSG_INDIRECT_READ 0x02
...@@ -19,6 +20,23 @@ ...@@ -19,6 +20,23 @@
#define IPC_CMD_VRTC_SETTIME 1 /* Set time */ #define IPC_CMD_VRTC_SETTIME 1 /* Set time */
#define IPC_CMD_VRTC_SETALARM 2 /* Set alarm */ #define IPC_CMD_VRTC_SETALARM 2 /* Set alarm */
struct device;
struct intel_scu_ipc_dev;
/**
* struct intel_scu_ipc_data - Data used to configure SCU IPC
* @mem: Base address of SCU IPC MMIO registers
* @irq: The IRQ number used for SCU (optional)
*/
struct intel_scu_ipc_data {
struct resource mem;
int irq;
};
struct intel_scu_ipc_dev *
intel_scu_ipc_register(struct device *parent,
const struct intel_scu_ipc_data *scu_data);
/* Read single register */ /* Read single register */
int intel_scu_ipc_ioread8(u16 addr, u8 *data); int intel_scu_ipc_ioread8(u16 addr, u8 *data);
......
...@@ -593,7 +593,7 @@ config INTEL_SOC_PMIC_MRFLD ...@@ -593,7 +593,7 @@ config INTEL_SOC_PMIC_MRFLD
tristate "Support for Intel Merrifield Basin Cove PMIC" tristate "Support for Intel Merrifield Basin Cove PMIC"
depends on GPIOLIB depends on GPIOLIB
depends on ACPI depends on ACPI
depends on INTEL_SCU_IPC depends on INTEL_SCU
select MFD_CORE select MFD_CORE
select REGMAP_IRQ select REGMAP_IRQ
help help
...@@ -625,7 +625,7 @@ config MFD_INTEL_LPSS_PCI ...@@ -625,7 +625,7 @@ config MFD_INTEL_LPSS_PCI
config MFD_INTEL_MSIC config MFD_INTEL_MSIC
bool "Intel MSIC" bool "Intel MSIC"
depends on INTEL_SCU_IPC depends on INTEL_SCU
select MFD_CORE select MFD_CORE
help help
Select this option to enable access to Intel MSIC (Avatele Select this option to enable access to Intel MSIC (Avatele
......
...@@ -1295,7 +1295,7 @@ config INTEL_MFLD_THERMAL ...@@ -1295,7 +1295,7 @@ config INTEL_MFLD_THERMAL
config INTEL_MID_POWER_BUTTON config INTEL_MID_POWER_BUTTON
tristate "power button driver for Intel MID platforms" tristate "power button driver for Intel MID platforms"
depends on INTEL_SCU_IPC && INPUT depends on INTEL_SCU && INPUT
help help
This driver handles the power button on the Intel MID platforms. This driver handles the power button on the Intel MID platforms.
...@@ -1342,17 +1342,25 @@ config INTEL_PUNIT_IPC ...@@ -1342,17 +1342,25 @@ config INTEL_PUNIT_IPC
which is used to bridge the communications between kernel and P-Unit. which is used to bridge the communications between kernel and P-Unit.
config INTEL_SCU_IPC config INTEL_SCU_IPC
bool "Intel SCU IPC Support" bool
config INTEL_SCU
bool
select INTEL_SCU_IPC
config INTEL_SCU_PCI
bool "Intel SCU PCI driver"
depends on X86_INTEL_MID depends on X86_INTEL_MID
default y select INTEL_SCU
---help--- help
IPC is used to bridge the communications between kernel and SCU on This driver is used to bridge the communications between kernel
some embedded Intel x86 platforms. This is not needed for PC-type and SCU on some embedded Intel x86 platforms. It also creates
machines. devices that are connected to the SoC through the SCU. This is
not needed for PC-type machines.
config INTEL_SCU_IPC_UTIL config INTEL_SCU_IPC_UTIL
tristate "Intel SCU IPC utility driver" tristate "Intel SCU IPC utility driver"
depends on INTEL_SCU_IPC depends on INTEL_SCU
---help--- ---help---
The IPC Util driver provides an interface with the SCU enabling The IPC Util driver provides an interface with the SCU enabling
low level access for debug work and updating the firmware. Say low level access for debug work and updating the firmware. Say
......
...@@ -141,6 +141,7 @@ obj-$(CONFIG_INTEL_PMC_CORE) += intel_pmc_core.o intel_pmc_core_pltdrv.o ...@@ -141,6 +141,7 @@ obj-$(CONFIG_INTEL_PMC_CORE) += intel_pmc_core.o intel_pmc_core_pltdrv.o
obj-$(CONFIG_INTEL_PMC_IPC) += intel_pmc_ipc.o obj-$(CONFIG_INTEL_PMC_IPC) += intel_pmc_ipc.o
obj-$(CONFIG_INTEL_PUNIT_IPC) += intel_punit_ipc.o obj-$(CONFIG_INTEL_PUNIT_IPC) += intel_punit_ipc.o
obj-$(CONFIG_INTEL_SCU_IPC) += intel_scu_ipc.o obj-$(CONFIG_INTEL_SCU_IPC) += intel_scu_ipc.o
obj-$(CONFIG_INTEL_SCU_PCI) += intel_scu_pcidrv.o
obj-$(CONFIG_INTEL_SCU_IPC_UTIL) += intel_scu_ipcutil.o obj-$(CONFIG_INTEL_SCU_IPC_UTIL) += intel_scu_ipcutil.o
obj-$(CONFIG_INTEL_TELEMETRY) += intel_telemetry_core.o \ obj-$(CONFIG_INTEL_TELEMETRY) += intel_telemetry_core.o \
intel_telemetry_pltdrv.o \ intel_telemetry_pltdrv.o \
......
...@@ -18,11 +18,10 @@ ...@@ -18,11 +18,10 @@
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/pci.h> #include <linux/io.h>
#include <linux/pm.h> #include <linux/module.h>
#include <linux/sfi.h> #include <linux/slab.h>
#include <asm/intel-mid.h>
#include <asm/intel_scu_ipc.h> #include <asm/intel_scu_ipc.h>
/* IPC defines the following message types */ /* IPC defines the following message types */
...@@ -55,14 +54,13 @@ ...@@ -55,14 +54,13 @@
#define IPC_IOC 0x100 /* IPC command register IOC bit */ #define IPC_IOC 0x100 /* IPC command register IOC bit */
struct intel_scu_ipc_dev { struct intel_scu_ipc_dev {
struct device *dev; struct device dev;
struct resource mem;
int irq;
void __iomem *ipc_base; void __iomem *ipc_base;
struct completion cmd_complete; struct completion cmd_complete;
u8 irq_mode;
}; };
static struct intel_scu_ipc_dev ipcdev; /* Only one for now */
#define IPC_STATUS 0x04 #define IPC_STATUS 0x04
#define IPC_STATUS_IRQ BIT(2) #define IPC_STATUS_IRQ BIT(2)
#define IPC_STATUS_ERR BIT(1) #define IPC_STATUS_ERR BIT(1)
...@@ -78,8 +76,14 @@ static struct intel_scu_ipc_dev ipcdev; /* Only one for now */ ...@@ -78,8 +76,14 @@ static struct intel_scu_ipc_dev ipcdev; /* Only one for now */
/* Timeout in jiffies */ /* Timeout in jiffies */
#define IPC_TIMEOUT (3 * HZ) #define IPC_TIMEOUT (3 * HZ)
static struct intel_scu_ipc_dev *ipcdev; /* Only one for now */
static DEFINE_MUTEX(ipclock); /* lock used to prevent multiple call to SCU */ static DEFINE_MUTEX(ipclock); /* lock used to prevent multiple call to SCU */
static struct class intel_scu_ipc_class = {
.name = "intel_scu_ipc",
.owner = THIS_MODULE,
};
/* /*
* Send ipc command * Send ipc command
* Command Register (Write Only): * Command Register (Write Only):
...@@ -143,7 +147,7 @@ static inline int busy_loop(struct intel_scu_ipc_dev *scu) ...@@ -143,7 +147,7 @@ static inline int busy_loop(struct intel_scu_ipc_dev *scu)
usleep_range(50, 100); usleep_range(50, 100);
} while (time_before(jiffies, end)); } while (time_before(jiffies, end));
dev_err(scu->dev, "IPC timed out"); dev_err(&scu->dev, "IPC timed out");
return -ETIMEDOUT; return -ETIMEDOUT;
} }
...@@ -153,7 +157,7 @@ static inline int ipc_wait_for_interrupt(struct intel_scu_ipc_dev *scu) ...@@ -153,7 +157,7 @@ static inline int ipc_wait_for_interrupt(struct intel_scu_ipc_dev *scu)
int status; int status;
if (!wait_for_completion_timeout(&scu->cmd_complete, IPC_TIMEOUT)) { if (!wait_for_completion_timeout(&scu->cmd_complete, IPC_TIMEOUT)) {
dev_err(scu->dev, "IPC timed out\n"); dev_err(&scu->dev, "IPC timed out\n");
return -ETIMEDOUT; return -ETIMEDOUT;
} }
...@@ -166,13 +170,13 @@ static inline int ipc_wait_for_interrupt(struct intel_scu_ipc_dev *scu) ...@@ -166,13 +170,13 @@ static inline int ipc_wait_for_interrupt(struct intel_scu_ipc_dev *scu)
static int intel_scu_ipc_check_status(struct intel_scu_ipc_dev *scu) static int intel_scu_ipc_check_status(struct intel_scu_ipc_dev *scu)
{ {
return scu->irq_mode ? ipc_wait_for_interrupt(scu) : busy_loop(scu); return scu->irq > 0 ? ipc_wait_for_interrupt(scu) : busy_loop(scu);
} }
/* Read/Write power control(PMIC in Langwell, MSIC in PenWell) registers */ /* Read/Write power control(PMIC in Langwell, MSIC in PenWell) registers */
static int pwr_reg_rdwr(u16 *addr, u8 *data, u32 count, u32 op, u32 id) static int pwr_reg_rdwr(u16 *addr, u8 *data, u32 count, u32 op, u32 id)
{ {
struct intel_scu_ipc_dev *scu = &ipcdev; struct intel_scu_ipc_dev *scu;
int nc; int nc;
u32 offset = 0; u32 offset = 0;
int err; int err;
...@@ -182,11 +186,11 @@ static int pwr_reg_rdwr(u16 *addr, u8 *data, u32 count, u32 op, u32 id) ...@@ -182,11 +186,11 @@ static int pwr_reg_rdwr(u16 *addr, u8 *data, u32 count, u32 op, u32 id)
memset(cbuf, 0, sizeof(cbuf)); memset(cbuf, 0, sizeof(cbuf));
mutex_lock(&ipclock); mutex_lock(&ipclock);
if (!ipcdev) {
if (scu->dev == NULL) {
mutex_unlock(&ipclock); mutex_unlock(&ipclock);
return -ENODEV; return -ENODEV;
} }
scu = ipcdev;
for (nc = 0; nc < count; nc++, offset += 2) { for (nc = 0; nc < count; nc++, offset += 2) {
cbuf[offset] = addr[nc]; cbuf[offset] = addr[nc];
...@@ -326,14 +330,15 @@ EXPORT_SYMBOL(intel_scu_ipc_update_register); ...@@ -326,14 +330,15 @@ EXPORT_SYMBOL(intel_scu_ipc_update_register);
*/ */
int intel_scu_ipc_simple_command(int cmd, int sub) int intel_scu_ipc_simple_command(int cmd, int sub)
{ {
struct intel_scu_ipc_dev *scu = &ipcdev; struct intel_scu_ipc_dev *scu;
int err; int err;
mutex_lock(&ipclock); mutex_lock(&ipclock);
if (scu->dev == NULL) { if (!ipcdev) {
mutex_unlock(&ipclock); mutex_unlock(&ipclock);
return -ENODEV; return -ENODEV;
} }
scu = ipcdev;
ipc_command(scu, sub << 12 | cmd); ipc_command(scu, sub << 12 | cmd);
err = intel_scu_ipc_check_status(scu); err = intel_scu_ipc_check_status(scu);
mutex_unlock(&ipclock); mutex_unlock(&ipclock);
...@@ -356,14 +361,15 @@ EXPORT_SYMBOL(intel_scu_ipc_simple_command); ...@@ -356,14 +361,15 @@ EXPORT_SYMBOL(intel_scu_ipc_simple_command);
int intel_scu_ipc_command(int cmd, int sub, u32 *in, int inlen, int intel_scu_ipc_command(int cmd, int sub, u32 *in, int inlen,
u32 *out, int outlen) u32 *out, int outlen)
{ {
struct intel_scu_ipc_dev *scu = &ipcdev; struct intel_scu_ipc_dev *scu;
int i, err; int i, err;
mutex_lock(&ipclock); mutex_lock(&ipclock);
if (scu->dev == NULL) { if (!ipcdev) {
mutex_unlock(&ipclock); mutex_unlock(&ipclock);
return -ENODEV; return -ENODEV;
} }
scu = ipcdev;
for (i = 0; i < inlen; i++) for (i = 0; i < inlen; i++)
ipc_data_writel(scu, *in++, 4 * i); ipc_data_writel(scu, *in++, 4 * i);
...@@ -399,61 +405,113 @@ static irqreturn_t ioc(int irq, void *dev_id) ...@@ -399,61 +405,113 @@ static irqreturn_t ioc(int irq, void *dev_id)
return IRQ_HANDLED; return IRQ_HANDLED;
} }
static void intel_scu_ipc_release(struct device *dev)
{
struct intel_scu_ipc_dev *scu;
scu = container_of(dev, struct intel_scu_ipc_dev, dev);
if (scu->irq > 0)
free_irq(scu->irq, scu);
iounmap(scu->ipc_base);
release_mem_region(scu->mem.start, resource_size(&scu->mem));
kfree(scu);
}
/** /**
* ipc_probe - probe an Intel SCU IPC * intel_scu_ipc_register() - Register SCU IPC device
* @pdev: the PCI device matching * @parent: Parent device
* @id: entry in the match table * @scu_data: Data used to configure SCU IPC
* *
* Enable and install an intel SCU IPC. This appears in the PCI space * Call this function to register SCU IPC mechanism under @parent.
* but uses some hard coded addresses as well. * Returns pointer to the new SCU IPC device or ERR_PTR() in case of
* failure.
*/ */
static int ipc_probe(struct pci_dev *pdev, const struct pci_device_id *id) struct intel_scu_ipc_dev *
intel_scu_ipc_register(struct device *parent,
const struct intel_scu_ipc_data *scu_data)
{ {
int err; int err;
struct intel_scu_ipc_dev *scu = &ipcdev; struct intel_scu_ipc_dev *scu;
void __iomem *ipc_base;
if (scu->dev) /* We support only one SCU */ mutex_lock(&ipclock);
return -EBUSY; /* We support only one IPC */
if (ipcdev) {
err = -EBUSY;
goto err_unlock;
}
err = pcim_enable_device(pdev); scu = kzalloc(sizeof(*scu), GFP_KERNEL);
if (err) if (!scu) {
return err; err = -ENOMEM;
goto err_unlock;
}
err = pcim_iomap_regions(pdev, 1 << 0, pci_name(pdev)); scu->dev.parent = parent;
if (err) scu->dev.class = &intel_scu_ipc_class;
return err; scu->dev.release = intel_scu_ipc_release;
dev_set_name(&scu->dev, "intel_scu_ipc");
if (!request_mem_region(scu_data->mem.start, resource_size(&scu_data->mem),
"intel_scu_ipc")) {
err = -EBUSY;
goto err_free;
}
ipc_base = ioremap(scu_data->mem.start, resource_size(&scu_data->mem));
if (!ipc_base) {
err = -ENOMEM;
goto err_release;
}
scu->ipc_base = ipc_base;
scu->mem = scu_data->mem;
scu->irq = scu_data->irq;
init_completion(&scu->cmd_complete); init_completion(&scu->cmd_complete);
scu->ipc_base = pcim_iomap_table(pdev)[0]; if (scu->irq > 0) {
err = request_irq(scu->irq, ioc, 0, "intel_scu_ipc", scu);
if (err)
goto err_unmap;
}
err = devm_request_irq(&pdev->dev, pdev->irq, ioc, 0, "intel_scu_ipc", /*
scu); * After this point intel_scu_ipc_release() takes care of
if (err) * releasing the SCU IPC resources once refcount drops to zero.
return err; */
err = device_register(&scu->dev);
if (err) {
put_device(&scu->dev);
goto err_unlock;
}
/* Assign device at last */ /* Assign device at last */
scu->dev = &pdev->dev; ipcdev = scu;
mutex_unlock(&ipclock);
intel_scu_devices_create(); return scu;
pci_set_drvdata(pdev, scu); err_unmap:
return 0; iounmap(ipc_base);
err_release:
release_mem_region(scu_data->mem.start, resource_size(&scu_data->mem));
err_free:
kfree(scu);
err_unlock:
mutex_unlock(&ipclock);
return ERR_PTR(err);
} }
EXPORT_SYMBOL_GPL(intel_scu_ipc_register);
static const struct pci_device_id pci_ids[] = { static int __init intel_scu_ipc_init(void)
{ PCI_VDEVICE(INTEL, 0x080e) }, {
{ PCI_VDEVICE(INTEL, 0x08ea) }, return class_register(&intel_scu_ipc_class);
{ PCI_VDEVICE(INTEL, 0x11a0) }, }
{} subsys_initcall(intel_scu_ipc_init);
};
static struct pci_driver ipc_driver = { static void __exit intel_scu_ipc_exit(void)
.driver = { {
.suppress_bind_attrs = true, class_unregister(&intel_scu_ipc_class);
}, }
.name = "intel_scu_ipc", module_exit(intel_scu_ipc_exit);
.id_table = pci_ids,
.probe = ipc_probe,
};
builtin_pci_driver(ipc_driver);
// SPDX-License-Identifier: GPL-2.0
/*
* PCI driver for the Intel SCU.
*
* Copyright (C) 2008-2010, 2015, 2020 Intel Corporation
* Authors: Sreedhara DS (sreedhara.ds@intel.com)
* Mika Westerberg <mika.westerberg@linux.intel.com>
*/
#include <linux/errno.h>
#include <linux/init.h>
#include <linux/pci.h>
#include <asm/intel-mid.h>
#include <asm/intel_scu_ipc.h>
static int intel_scu_pci_probe(struct pci_dev *pdev,
const struct pci_device_id *id)
{
struct intel_scu_ipc_data scu_data = {};
struct intel_scu_ipc_dev *scu;
int ret;
ret = pcim_enable_device(pdev);
if (ret)
return ret;
scu_data.mem = pdev->resource[0];
scu_data.irq = pdev->irq;
scu = intel_scu_ipc_register(&pdev->dev, &scu_data);
if (IS_ERR(scu))
return PTR_ERR(scu);
intel_scu_devices_create();
return 0;
}
static const struct pci_device_id pci_ids[] = {
{ PCI_VDEVICE(INTEL, 0x080e) },
{ PCI_VDEVICE(INTEL, 0x08ea) },
{ PCI_VDEVICE(INTEL, 0x11a0) },
{}
};
static struct pci_driver intel_scu_pci_driver = {
.driver = {
.suppress_bind_attrs = true,
},
.name = "intel_scu",
.id_table = pci_ids,
.probe = intel_scu_pci_probe,
};
builtin_pci_driver(intel_scu_pci_driver);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment