arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
/*
|
|
|
|
* IOSF-SB MailBox Interface Driver
|
|
|
|
* Copyright (c) 2013, Intel Corporation.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* The IOSF-SB is a fabric bus available on Atom based SOC's that uses a
|
|
|
|
* mailbox interface (MBI) to communicate with mutiple devices. This
|
|
|
|
* driver implements access to this interface for those platforms that can
|
|
|
|
* enumerate the device using PCI.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/pci.h>
|
2014-08-28 04:40:40 +07:00
|
|
|
#include <linux/debugfs.h>
|
|
|
|
#include <linux/capability.h>
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
|
|
|
|
#include <asm/iosf_mbi.h>
|
|
|
|
|
2014-05-10 03:44:08 +07:00
|
|
|
#define PCI_DEVICE_ID_BAYTRAIL 0x0F00
|
2014-09-18 12:13:49 +07:00
|
|
|
#define PCI_DEVICE_ID_BRASWELL 0x2280
|
2014-05-10 03:44:08 +07:00
|
|
|
#define PCI_DEVICE_ID_QUARK_X1000 0x0958
|
2015-07-08 21:45:09 +07:00
|
|
|
#define PCI_DEVICE_ID_TANGIER 0x1170
|
2014-05-10 03:44:08 +07:00
|
|
|
|
2015-07-08 21:45:08 +07:00
|
|
|
static struct pci_dev *mbi_pdev;
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
static DEFINE_SPINLOCK(iosf_mbi_lock);
|
2017-02-10 17:27:51 +07:00
|
|
|
static DEFINE_MUTEX(iosf_mbi_punit_mutex);
|
2017-02-24 16:29:02 +07:00
|
|
|
static BLOCKING_NOTIFIER_HEAD(iosf_mbi_pmic_bus_access_notifier);
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
|
|
|
|
static inline u32 iosf_mbi_form_mcr(u8 op, u8 port, u8 offset)
|
|
|
|
{
|
|
|
|
return (op << 24) | (port << 16) | (offset << 8) | MBI_ENABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int iosf_mbi_pci_read_mdr(u32 mcrx, u32 mcr, u32 *mdr)
|
|
|
|
{
|
|
|
|
int result;
|
|
|
|
|
|
|
|
if (!mbi_pdev)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
if (mcrx) {
|
|
|
|
result = pci_write_config_dword(mbi_pdev, MBI_MCRX_OFFSET,
|
|
|
|
mcrx);
|
|
|
|
if (result < 0)
|
|
|
|
goto fail_read;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = pci_write_config_dword(mbi_pdev, MBI_MCR_OFFSET, mcr);
|
|
|
|
if (result < 0)
|
|
|
|
goto fail_read;
|
|
|
|
|
|
|
|
result = pci_read_config_dword(mbi_pdev, MBI_MDR_OFFSET, mdr);
|
|
|
|
if (result < 0)
|
|
|
|
goto fail_read;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail_read:
|
|
|
|
dev_err(&mbi_pdev->dev, "PCI config access failed with %d\n", result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int iosf_mbi_pci_write_mdr(u32 mcrx, u32 mcr, u32 mdr)
|
|
|
|
{
|
|
|
|
int result;
|
|
|
|
|
|
|
|
if (!mbi_pdev)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
result = pci_write_config_dword(mbi_pdev, MBI_MDR_OFFSET, mdr);
|
|
|
|
if (result < 0)
|
|
|
|
goto fail_write;
|
|
|
|
|
|
|
|
if (mcrx) {
|
|
|
|
result = pci_write_config_dword(mbi_pdev, MBI_MCRX_OFFSET,
|
|
|
|
mcrx);
|
|
|
|
if (result < 0)
|
|
|
|
goto fail_write;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = pci_write_config_dword(mbi_pdev, MBI_MCR_OFFSET, mcr);
|
|
|
|
if (result < 0)
|
|
|
|
goto fail_write;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail_write:
|
|
|
|
dev_err(&mbi_pdev->dev, "PCI config access failed with %d\n", result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
|
|
|
|
{
|
|
|
|
u32 mcr, mcrx;
|
|
|
|
unsigned long flags;
|
|
|
|
int ret;
|
|
|
|
|
2015-07-08 21:45:08 +07:00
|
|
|
/* Access to the GFX unit is handled by GPU code */
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
if (port == BT_MBI_UNIT_GFX) {
|
|
|
|
WARN_ON(1);
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
|
|
|
|
mcr = iosf_mbi_form_mcr(opcode, port, offset & MBI_MASK_LO);
|
|
|
|
mcrx = offset & MBI_MASK_HI;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&iosf_mbi_lock, flags);
|
|
|
|
ret = iosf_mbi_pci_read_mdr(mcrx, mcr, mdr);
|
|
|
|
spin_unlock_irqrestore(&iosf_mbi_lock, flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_read);
|
|
|
|
|
|
|
|
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
|
|
|
|
{
|
|
|
|
u32 mcr, mcrx;
|
|
|
|
unsigned long flags;
|
|
|
|
int ret;
|
|
|
|
|
2015-07-08 21:45:08 +07:00
|
|
|
/* Access to the GFX unit is handled by GPU code */
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
if (port == BT_MBI_UNIT_GFX) {
|
|
|
|
WARN_ON(1);
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
|
|
|
|
mcr = iosf_mbi_form_mcr(opcode, port, offset & MBI_MASK_LO);
|
|
|
|
mcrx = offset & MBI_MASK_HI;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&iosf_mbi_lock, flags);
|
|
|
|
ret = iosf_mbi_pci_write_mdr(mcrx, mcr, mdr);
|
|
|
|
spin_unlock_irqrestore(&iosf_mbi_lock, flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_write);
|
|
|
|
|
|
|
|
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
|
|
|
|
{
|
|
|
|
u32 mcr, mcrx;
|
|
|
|
u32 value;
|
|
|
|
unsigned long flags;
|
|
|
|
int ret;
|
|
|
|
|
2015-07-08 21:45:08 +07:00
|
|
|
/* Access to the GFX unit is handled by GPU code */
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
if (port == BT_MBI_UNIT_GFX) {
|
|
|
|
WARN_ON(1);
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
|
|
|
|
mcr = iosf_mbi_form_mcr(opcode, port, offset & MBI_MASK_LO);
|
|
|
|
mcrx = offset & MBI_MASK_HI;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&iosf_mbi_lock, flags);
|
|
|
|
|
|
|
|
/* Read current mdr value */
|
|
|
|
ret = iosf_mbi_pci_read_mdr(mcrx, mcr & MBI_RD_MASK, &value);
|
|
|
|
if (ret < 0) {
|
|
|
|
spin_unlock_irqrestore(&iosf_mbi_lock, flags);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Apply mask */
|
|
|
|
value &= ~mask;
|
|
|
|
mdr &= mask;
|
|
|
|
value |= mdr;
|
|
|
|
|
|
|
|
/* Write back */
|
|
|
|
ret = iosf_mbi_pci_write_mdr(mcrx, mcr | MBI_WR_MASK, value);
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&iosf_mbi_lock, flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_modify);
|
|
|
|
|
2014-05-10 03:44:05 +07:00
|
|
|
bool iosf_mbi_available(void)
|
|
|
|
{
|
|
|
|
/* Mbi isn't hot-pluggable. No remove routine is provided */
|
|
|
|
return mbi_pdev;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_available);
|
|
|
|
|
2017-02-10 17:27:51 +07:00
|
|
|
void iosf_mbi_punit_acquire(void)
|
|
|
|
{
|
|
|
|
mutex_lock(&iosf_mbi_punit_mutex);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_punit_acquire);
|
|
|
|
|
|
|
|
void iosf_mbi_punit_release(void)
|
|
|
|
{
|
|
|
|
mutex_unlock(&iosf_mbi_punit_mutex);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_punit_release);
|
|
|
|
|
2017-02-24 16:29:02 +07:00
|
|
|
int iosf_mbi_register_pmic_bus_access_notifier(struct notifier_block *nb)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Wait for the bus to go inactive before registering */
|
|
|
|
mutex_lock(&iosf_mbi_punit_mutex);
|
|
|
|
ret = blocking_notifier_chain_register(
|
|
|
|
&iosf_mbi_pmic_bus_access_notifier, nb);
|
|
|
|
mutex_unlock(&iosf_mbi_punit_mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_register_pmic_bus_access_notifier);
|
|
|
|
|
2017-10-19 18:16:19 +07:00
|
|
|
int iosf_mbi_unregister_pmic_bus_access_notifier_unlocked(
|
|
|
|
struct notifier_block *nb)
|
|
|
|
{
|
|
|
|
iosf_mbi_assert_punit_acquired();
|
|
|
|
|
|
|
|
return blocking_notifier_chain_unregister(
|
|
|
|
&iosf_mbi_pmic_bus_access_notifier, nb);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_unregister_pmic_bus_access_notifier_unlocked);
|
|
|
|
|
2017-02-24 16:29:02 +07:00
|
|
|
int iosf_mbi_unregister_pmic_bus_access_notifier(struct notifier_block *nb)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Wait for the bus to go inactive before unregistering */
|
|
|
|
mutex_lock(&iosf_mbi_punit_mutex);
|
2017-10-19 18:16:19 +07:00
|
|
|
ret = iosf_mbi_unregister_pmic_bus_access_notifier_unlocked(nb);
|
2017-02-24 16:29:02 +07:00
|
|
|
mutex_unlock(&iosf_mbi_punit_mutex);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_unregister_pmic_bus_access_notifier);
|
|
|
|
|
|
|
|
int iosf_mbi_call_pmic_bus_access_notifier_chain(unsigned long val, void *v)
|
|
|
|
{
|
|
|
|
return blocking_notifier_call_chain(
|
|
|
|
&iosf_mbi_pmic_bus_access_notifier, val, v);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_call_pmic_bus_access_notifier_chain);
|
|
|
|
|
2017-10-19 18:16:19 +07:00
|
|
|
void iosf_mbi_assert_punit_acquired(void)
|
|
|
|
{
|
|
|
|
WARN_ON(!mutex_is_locked(&iosf_mbi_punit_mutex));
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(iosf_mbi_assert_punit_acquired);
|
|
|
|
|
2014-09-18 12:13:51 +07:00
|
|
|
#ifdef CONFIG_IOSF_MBI_DEBUG
|
2014-08-28 04:40:40 +07:00
|
|
|
static u32 dbg_mdr;
|
|
|
|
static u32 dbg_mcr;
|
|
|
|
static u32 dbg_mcrx;
|
|
|
|
|
|
|
|
static int mcr_get(void *data, u64 *val)
|
|
|
|
{
|
|
|
|
*val = *(u32 *)data;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mcr_set(void *data, u64 val)
|
|
|
|
{
|
|
|
|
u8 command = ((u32)val & 0xFF000000) >> 24,
|
|
|
|
port = ((u32)val & 0x00FF0000) >> 16,
|
|
|
|
offset = ((u32)val & 0x0000FF00) >> 8;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
*(u32 *)data = val;
|
|
|
|
|
|
|
|
if (!capable(CAP_SYS_RAWIO))
|
|
|
|
return -EACCES;
|
|
|
|
|
|
|
|
if (command & 1u)
|
|
|
|
err = iosf_mbi_write(port,
|
|
|
|
command,
|
|
|
|
dbg_mcrx | offset,
|
|
|
|
dbg_mdr);
|
|
|
|
else
|
|
|
|
err = iosf_mbi_read(port,
|
|
|
|
command,
|
|
|
|
dbg_mcrx | offset,
|
|
|
|
&dbg_mdr);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
DEFINE_SIMPLE_ATTRIBUTE(iosf_mcr_fops, mcr_get, mcr_set , "%llx\n");
|
|
|
|
|
|
|
|
static struct dentry *iosf_dbg;
|
2014-09-18 12:13:51 +07:00
|
|
|
|
2014-08-28 04:40:40 +07:00
|
|
|
static void iosf_sideband_debug_init(void)
|
|
|
|
{
|
|
|
|
struct dentry *d;
|
|
|
|
|
|
|
|
iosf_dbg = debugfs_create_dir("iosf_sb", NULL);
|
|
|
|
if (IS_ERR_OR_NULL(iosf_dbg))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* mdr */
|
|
|
|
d = debugfs_create_x32("mdr", 0660, iosf_dbg, &dbg_mdr);
|
2015-07-08 21:45:06 +07:00
|
|
|
if (!d)
|
2014-08-28 04:40:40 +07:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* mcrx */
|
2015-07-08 21:45:06 +07:00
|
|
|
d = debugfs_create_x32("mcrx", 0660, iosf_dbg, &dbg_mcrx);
|
|
|
|
if (!d)
|
2014-08-28 04:40:40 +07:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* mcr - initiates mailbox tranaction */
|
2015-07-08 21:45:06 +07:00
|
|
|
d = debugfs_create_file("mcr", 0660, iosf_dbg, &dbg_mcr, &iosf_mcr_fops);
|
|
|
|
if (!d)
|
2014-08-28 04:40:40 +07:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
debugfs_remove_recursive(d);
|
|
|
|
}
|
2014-09-18 12:13:51 +07:00
|
|
|
|
|
|
|
static void iosf_debugfs_init(void)
|
|
|
|
{
|
|
|
|
iosf_sideband_debug_init();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void iosf_debugfs_remove(void)
|
|
|
|
{
|
|
|
|
debugfs_remove_recursive(iosf_dbg);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static inline void iosf_debugfs_init(void) { }
|
|
|
|
static inline void iosf_debugfs_remove(void) { }
|
|
|
|
#endif /* CONFIG_IOSF_MBI_DEBUG */
|
2014-08-28 04:40:40 +07:00
|
|
|
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
static int iosf_mbi_probe(struct pci_dev *pdev,
|
|
|
|
const struct pci_device_id *unused)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = pci_enable_device(pdev);
|
|
|
|
if (ret < 0) {
|
|
|
|
dev_err(&pdev->dev, "error: could not enable device\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
mbi_pdev = pci_dev_get(pdev);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-08-08 20:56:03 +07:00
|
|
|
static const struct pci_device_id iosf_mbi_pci_ids[] = {
|
2014-05-10 03:44:08 +07:00
|
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_BAYTRAIL) },
|
2014-09-18 12:13:49 +07:00
|
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_BRASWELL) },
|
2014-05-10 03:44:08 +07:00
|
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_QUARK_X1000) },
|
2015-07-08 21:45:09 +07:00
|
|
|
{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_TANGIER) },
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
{ 0, },
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(pci, iosf_mbi_pci_ids);
|
|
|
|
|
|
|
|
static struct pci_driver iosf_mbi_pci_driver = {
|
|
|
|
.name = "iosf_mbi_pci",
|
|
|
|
.probe = iosf_mbi_probe,
|
|
|
|
.id_table = iosf_mbi_pci_ids,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init iosf_mbi_init(void)
|
|
|
|
{
|
2014-09-18 12:13:51 +07:00
|
|
|
iosf_debugfs_init();
|
|
|
|
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
return pci_register_driver(&iosf_mbi_pci_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit iosf_mbi_exit(void)
|
|
|
|
{
|
2014-09-18 12:13:51 +07:00
|
|
|
iosf_debugfs_remove();
|
2014-08-28 04:40:40 +07:00
|
|
|
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
pci_unregister_driver(&iosf_mbi_pci_driver);
|
2015-07-08 21:45:07 +07:00
|
|
|
pci_dev_put(mbi_pdev);
|
|
|
|
mbi_pdev = NULL;
|
arch: x86: New MailBox support driver for Intel SOC's
Current Intel SOC cores use a MailBox Interface (MBI) to provide access to
configuration registers on devices (called units) connected to the system
fabric. This is a support driver that implements access to this interface on
those platforms that can enumerate the device using PCI. Initial support is for
BayTrail, for which port definitons are provided. This is a requirement for
implementing platform specific features (e.g. RAPL driver requires this to
perform platform specific power management using the registers in PUNIT).
Dependant modules should select IOSF_MBI in their respective Kconfig
configuraiton. Serialized access is handled by all exported routines with
spinlocks.
The API includes 3 functions for access to unit registers:
int iosf_mbi_read(u8 port, u8 opcode, u32 offset, u32 *mdr)
int iosf_mbi_write(u8 port, u8 opcode, u32 offset, u32 mdr)
int iosf_mbi_modify(u8 port, u8 opcode, u32 offset, u32 mdr, u32 mask)
port: indicating the unit being accessed
opcode: the read or write port specific opcode
offset: the register offset within the port
mdr: the register data to be read, written, or modified
mask: bit locations in mdr to change
Returns nonzero on error
Note: GPU code handles access to the GFX unit. Therefore access to that unit
with this driver is disallowed to avoid conflicts.
Signed-off-by: David E. Box <david.e.box@linux.intel.com>
Link: http://lkml.kernel.org/r/1389216471-734-1-git-send-email-david.e.box@linux.intel.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Rafael J. Wysocki <rjw@rjwysocki.net>
Cc: Matthew Garrett <mjg59@srcf.ucam.org>
2014-01-09 04:27:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
module_init(iosf_mbi_init);
|
|
|
|
module_exit(iosf_mbi_exit);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("David E. Box <david.e.box@linux.intel.com>");
|
|
|
|
MODULE_DESCRIPTION("IOSF Mailbox Interface accessor");
|
|
|
|
MODULE_LICENSE("GPL v2");
|