242 lines
5.2 KiB
C
242 lines
5.2 KiB
C
|
// SPDX-License-Identifier: GPL-2.0-only
|
||
|
/*
|
||
|
* processor thermal device mailbox driver for Workload type hints
|
||
|
* Copyright (c) 2020, Intel Corporation.
|
||
|
*/
|
||
|
|
||
|
#include <linux/kernel.h>
|
||
|
#include <linux/module.h>
|
||
|
#include <linux/pci.h>
|
||
|
#include <linux/io-64-nonatomic-lo-hi.h>
|
||
|
#include "processor_thermal_device.h"
|
||
|
|
||
|
#define MBOX_CMD_WORKLOAD_TYPE_READ 0x0E
|
||
|
#define MBOX_CMD_WORKLOAD_TYPE_WRITE 0x0F
|
||
|
|
||
|
#define MBOX_OFFSET_DATA 0x5810
|
||
|
#define MBOX_OFFSET_INTERFACE 0x5818
|
||
|
|
||
|
#define MBOX_BUSY_BIT 31
|
||
|
#define MBOX_RETRY_COUNT 100
|
||
|
|
||
|
#define MBOX_DATA_BIT_VALID 31
|
||
|
#define MBOX_DATA_BIT_AC_DC 30
|
||
|
|
||
|
static DEFINE_MUTEX(mbox_lock);
|
||
|
|
||
|
static int wait_for_mbox_ready(struct proc_thermal_device *proc_priv)
|
||
|
{
|
||
|
u32 retries, data;
|
||
|
int ret;
|
||
|
|
||
|
/* Poll for rb bit == 0 */
|
||
|
retries = MBOX_RETRY_COUNT;
|
||
|
do {
|
||
|
data = readl(proc_priv->mmio_base + MBOX_OFFSET_INTERFACE);
|
||
|
if (data & BIT_ULL(MBOX_BUSY_BIT)) {
|
||
|
ret = -EBUSY;
|
||
|
continue;
|
||
|
}
|
||
|
ret = 0;
|
||
|
break;
|
||
|
} while (--retries);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int send_mbox_write_cmd(struct pci_dev *pdev, u16 id, u32 data)
|
||
|
{
|
||
|
struct proc_thermal_device *proc_priv;
|
||
|
u32 reg_data;
|
||
|
int ret;
|
||
|
|
||
|
proc_priv = pci_get_drvdata(pdev);
|
||
|
|
||
|
mutex_lock(&mbox_lock);
|
||
|
|
||
|
ret = wait_for_mbox_ready(proc_priv);
|
||
|
if (ret)
|
||
|
goto unlock_mbox;
|
||
|
|
||
|
writel(data, (proc_priv->mmio_base + MBOX_OFFSET_DATA));
|
||
|
/* Write command register */
|
||
|
reg_data = BIT_ULL(MBOX_BUSY_BIT) | id;
|
||
|
writel(reg_data, (proc_priv->mmio_base + MBOX_OFFSET_INTERFACE));
|
||
|
|
||
|
ret = wait_for_mbox_ready(proc_priv);
|
||
|
|
||
|
unlock_mbox:
|
||
|
mutex_unlock(&mbox_lock);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int send_mbox_read_cmd(struct pci_dev *pdev, u16 id, u64 *resp)
|
||
|
{
|
||
|
struct proc_thermal_device *proc_priv;
|
||
|
u32 reg_data;
|
||
|
int ret;
|
||
|
|
||
|
proc_priv = pci_get_drvdata(pdev);
|
||
|
|
||
|
mutex_lock(&mbox_lock);
|
||
|
|
||
|
ret = wait_for_mbox_ready(proc_priv);
|
||
|
if (ret)
|
||
|
goto unlock_mbox;
|
||
|
|
||
|
/* Write command register */
|
||
|
reg_data = BIT_ULL(MBOX_BUSY_BIT) | id;
|
||
|
writel(reg_data, (proc_priv->mmio_base + MBOX_OFFSET_INTERFACE));
|
||
|
|
||
|
ret = wait_for_mbox_ready(proc_priv);
|
||
|
if (ret)
|
||
|
goto unlock_mbox;
|
||
|
|
||
|
if (id == MBOX_CMD_WORKLOAD_TYPE_READ)
|
||
|
*resp = readl(proc_priv->mmio_base + MBOX_OFFSET_DATA);
|
||
|
else
|
||
|
*resp = readq(proc_priv->mmio_base + MBOX_OFFSET_DATA);
|
||
|
|
||
|
unlock_mbox:
|
||
|
mutex_unlock(&mbox_lock);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int processor_thermal_send_mbox_read_cmd(struct pci_dev *pdev, u16 id, u64 *resp)
|
||
|
{
|
||
|
return send_mbox_read_cmd(pdev, id, resp);
|
||
|
}
|
||
|
EXPORT_SYMBOL_NS_GPL(processor_thermal_send_mbox_read_cmd, INT340X_THERMAL);
|
||
|
|
||
|
int processor_thermal_send_mbox_write_cmd(struct pci_dev *pdev, u16 id, u32 data)
|
||
|
{
|
||
|
return send_mbox_write_cmd(pdev, id, data);
|
||
|
}
|
||
|
EXPORT_SYMBOL_NS_GPL(processor_thermal_send_mbox_write_cmd, INT340X_THERMAL);
|
||
|
|
||
|
/* List of workload types */
|
||
|
static const char * const workload_types[] = {
|
||
|
"none",
|
||
|
"idle",
|
||
|
"semi_active",
|
||
|
"bursty",
|
||
|
"sustained",
|
||
|
"battery_life",
|
||
|
NULL
|
||
|
};
|
||
|
|
||
|
static ssize_t workload_available_types_show(struct device *dev,
|
||
|
struct device_attribute *attr,
|
||
|
char *buf)
|
||
|
{
|
||
|
int i = 0;
|
||
|
int ret = 0;
|
||
|
|
||
|
while (workload_types[i] != NULL)
|
||
|
ret += sprintf(&buf[ret], "%s ", workload_types[i++]);
|
||
|
|
||
|
ret += sprintf(&buf[ret], "\n");
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static DEVICE_ATTR_RO(workload_available_types);
|
||
|
|
||
|
static ssize_t workload_type_store(struct device *dev,
|
||
|
struct device_attribute *attr,
|
||
|
const char *buf, size_t count)
|
||
|
{
|
||
|
struct pci_dev *pdev = to_pci_dev(dev);
|
||
|
char str_preference[15];
|
||
|
u32 data = 0;
|
||
|
ssize_t ret;
|
||
|
|
||
|
ret = sscanf(buf, "%14s", str_preference);
|
||
|
if (ret != 1)
|
||
|
return -EINVAL;
|
||
|
|
||
|
ret = match_string(workload_types, -1, str_preference);
|
||
|
if (ret < 0)
|
||
|
return ret;
|
||
|
|
||
|
ret &= 0xff;
|
||
|
|
||
|
if (ret)
|
||
|
data = BIT(MBOX_DATA_BIT_VALID) | BIT(MBOX_DATA_BIT_AC_DC);
|
||
|
|
||
|
data |= ret;
|
||
|
|
||
|
ret = send_mbox_write_cmd(pdev, MBOX_CMD_WORKLOAD_TYPE_WRITE, data);
|
||
|
if (ret)
|
||
|
return false;
|
||
|
|
||
|
return count;
|
||
|
}
|
||
|
|
||
|
static ssize_t workload_type_show(struct device *dev,
|
||
|
struct device_attribute *attr,
|
||
|
char *buf)
|
||
|
{
|
||
|
struct pci_dev *pdev = to_pci_dev(dev);
|
||
|
u64 cmd_resp;
|
||
|
int ret;
|
||
|
|
||
|
ret = send_mbox_read_cmd(pdev, MBOX_CMD_WORKLOAD_TYPE_READ, &cmd_resp);
|
||
|
if (ret)
|
||
|
return false;
|
||
|
|
||
|
cmd_resp &= 0xff;
|
||
|
|
||
|
if (cmd_resp > ARRAY_SIZE(workload_types) - 1)
|
||
|
return -EINVAL;
|
||
|
|
||
|
return sprintf(buf, "%s\n", workload_types[cmd_resp]);
|
||
|
}
|
||
|
|
||
|
static DEVICE_ATTR_RW(workload_type);
|
||
|
|
||
|
static struct attribute *workload_req_attrs[] = {
|
||
|
&dev_attr_workload_available_types.attr,
|
||
|
&dev_attr_workload_type.attr,
|
||
|
NULL
|
||
|
};
|
||
|
|
||
|
static const struct attribute_group workload_req_attribute_group = {
|
||
|
.attrs = workload_req_attrs,
|
||
|
.name = "workload_request"
|
||
|
};
|
||
|
|
||
|
static bool workload_req_created;
|
||
|
|
||
|
int proc_thermal_mbox_add(struct pci_dev *pdev, struct proc_thermal_device *proc_priv)
|
||
|
{
|
||
|
u64 cmd_resp;
|
||
|
int ret;
|
||
|
|
||
|
/* Check if there is a mailbox support, if fails return success */
|
||
|
ret = send_mbox_read_cmd(pdev, MBOX_CMD_WORKLOAD_TYPE_READ, &cmd_resp);
|
||
|
if (ret)
|
||
|
return 0;
|
||
|
|
||
|
ret = sysfs_create_group(&pdev->dev.kobj, &workload_req_attribute_group);
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
|
||
|
workload_req_created = true;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(proc_thermal_mbox_add);
|
||
|
|
||
|
void proc_thermal_mbox_remove(struct pci_dev *pdev)
|
||
|
{
|
||
|
if (workload_req_created)
|
||
|
sysfs_remove_group(&pdev->dev.kobj, &workload_req_attribute_group);
|
||
|
|
||
|
workload_req_created = false;
|
||
|
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(proc_thermal_mbox_remove);
|
||
|
|
||
|
MODULE_LICENSE("GPL v2");
|