cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

intel_pmc_bxt.c (12261B)


      1// SPDX-License-Identifier: GPL-2.0
      2/*
      3 * Driver for the Intel Broxton PMC
      4 *
      5 * (C) Copyright 2014 - 2020 Intel Corporation
      6 *
      7 * This driver is based on Intel SCU IPC driver (intel_scu_ipc.c) by
      8 * Sreedhara DS <sreedhara.ds@intel.com>
      9 *
     10 * The PMC (Power Management Controller) running on the ARC processor
     11 * communicates with another entity running in the IA (Intel Architecture)
     12 * core through an IPC (Intel Processor Communications) mechanism which in
     13 * turn sends messages between the IA and the PMC.
     14 */
     15
     16#include <linux/acpi.h>
     17#include <linux/delay.h>
     18#include <linux/errno.h>
     19#include <linux/interrupt.h>
     20#include <linux/io-64-nonatomic-lo-hi.h>
     21#include <linux/mfd/core.h>
     22#include <linux/mfd/intel_pmc_bxt.h>
     23#include <linux/module.h>
     24#include <linux/platform_device.h>
     25#include <linux/platform_data/itco_wdt.h>
     26
     27#include <asm/intel_scu_ipc.h>
     28
     29/* Residency with clock rate at 19.2MHz to usecs */
     30#define S0IX_RESIDENCY_IN_USECS(d, s)		\
     31({						\
     32	u64 result = 10ull * ((d) + (s));	\
     33	do_div(result, 192);			\
     34	result;					\
     35})
     36
     37/* Resources exported from IFWI */
     38#define PLAT_RESOURCE_IPC_INDEX		0
     39#define PLAT_RESOURCE_IPC_SIZE		0x1000
     40#define PLAT_RESOURCE_GCR_OFFSET	0x1000
     41#define PLAT_RESOURCE_GCR_SIZE		0x1000
     42#define PLAT_RESOURCE_BIOS_DATA_INDEX	1
     43#define PLAT_RESOURCE_BIOS_IFACE_INDEX	2
     44#define PLAT_RESOURCE_TELEM_SSRAM_INDEX	3
     45#define PLAT_RESOURCE_ISP_DATA_INDEX	4
     46#define PLAT_RESOURCE_ISP_IFACE_INDEX	5
     47#define PLAT_RESOURCE_GTD_DATA_INDEX	6
     48#define PLAT_RESOURCE_GTD_IFACE_INDEX	7
     49#define PLAT_RESOURCE_ACPI_IO_INDEX	0
     50
     51/*
     52 * BIOS does not create an ACPI device for each PMC function, but
     53 * exports multiple resources from one ACPI device (IPC) for multiple
     54 * functions. This driver is responsible for creating a child device and
     55 * to export resources for those functions.
     56 */
     57#define SMI_EN_OFFSET			0x0040
     58#define SMI_EN_SIZE			4
     59#define TCO_BASE_OFFSET			0x0060
     60#define TCO_REGS_SIZE			16
     61#define TELEM_SSRAM_SIZE		240
     62#define TELEM_PMC_SSRAM_OFFSET		0x1b00
     63#define TELEM_PUNIT_SSRAM_OFFSET	0x1a00
     64
     65/* Commands */
     66#define PMC_NORTHPEAK_CTRL		0xed
     67
     68static inline bool is_gcr_valid(u32 offset)
     69{
     70	return offset < PLAT_RESOURCE_GCR_SIZE - 8;
     71}
     72
     73/**
     74 * intel_pmc_gcr_read64() - Read a 64-bit PMC GCR register
     75 * @pmc: PMC device pointer
     76 * @offset: offset of GCR register from GCR address base
     77 * @data: data pointer for storing the register output
     78 *
     79 * Reads the 64-bit PMC GCR register at given offset.
     80 *
     81 * Return: Negative value on error or 0 on success.
     82 */
     83int intel_pmc_gcr_read64(struct intel_pmc_dev *pmc, u32 offset, u64 *data)
     84{
     85	if (!is_gcr_valid(offset))
     86		return -EINVAL;
     87
     88	spin_lock(&pmc->gcr_lock);
     89	*data = readq(pmc->gcr_mem_base + offset);
     90	spin_unlock(&pmc->gcr_lock);
     91
     92	return 0;
     93}
     94EXPORT_SYMBOL_GPL(intel_pmc_gcr_read64);
     95
     96/**
     97 * intel_pmc_gcr_update() - Update PMC GCR register bits
     98 * @pmc: PMC device pointer
     99 * @offset: offset of GCR register from GCR address base
    100 * @mask: bit mask for update operation
    101 * @val: update value
    102 *
    103 * Updates the bits of given GCR register as specified by
    104 * @mask and @val.
    105 *
    106 * Return: Negative value on error or 0 on success.
    107 */
    108int intel_pmc_gcr_update(struct intel_pmc_dev *pmc, u32 offset, u32 mask, u32 val)
    109{
    110	u32 new_val;
    111
    112	if (!is_gcr_valid(offset))
    113		return -EINVAL;
    114
    115	spin_lock(&pmc->gcr_lock);
    116	new_val = readl(pmc->gcr_mem_base + offset);
    117
    118	new_val = (new_val & ~mask) | (val & mask);
    119	writel(new_val, pmc->gcr_mem_base + offset);
    120
    121	new_val = readl(pmc->gcr_mem_base + offset);
    122	spin_unlock(&pmc->gcr_lock);
    123
    124	/* Check whether the bit update is successful */
    125	return (new_val & mask) != (val & mask) ? -EIO : 0;
    126}
    127EXPORT_SYMBOL_GPL(intel_pmc_gcr_update);
    128
    129/**
    130 * intel_pmc_s0ix_counter_read() - Read S0ix residency
    131 * @pmc: PMC device pointer
    132 * @data: Out param that contains current S0ix residency count.
    133 *
    134 * Writes to @data how many usecs the system has been in low-power S0ix
    135 * state.
    136 *
    137 * Return: An error code or 0 on success.
    138 */
    139int intel_pmc_s0ix_counter_read(struct intel_pmc_dev *pmc, u64 *data)
    140{
    141	u64 deep, shlw;
    142
    143	spin_lock(&pmc->gcr_lock);
    144	deep = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_DEEP_S0IX_REG);
    145	shlw = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_SHLW_S0IX_REG);
    146	spin_unlock(&pmc->gcr_lock);
    147
    148	*data = S0IX_RESIDENCY_IN_USECS(deep, shlw);
    149	return 0;
    150}
    151EXPORT_SYMBOL_GPL(intel_pmc_s0ix_counter_read);
    152
    153/**
    154 * simplecmd_store() - Send a simple IPC command
    155 * @dev: Device under the attribute is
    156 * @attr: Attribute in question
    157 * @buf: Buffer holding data to be stored to the attribute
    158 * @count: Number of bytes in @buf
    159 *
    160 * Expects a string with two integers separated with space. These two
    161 * values hold command and subcommand that is send to PMC.
    162 *
    163 * Return: Number number of bytes written (@count) or negative errno in
    164 *	   case of error.
    165 */
    166static ssize_t simplecmd_store(struct device *dev, struct device_attribute *attr,
    167			       const char *buf, size_t count)
    168{
    169	struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
    170	struct intel_scu_ipc_dev *scu = pmc->scu;
    171	int subcmd;
    172	int cmd;
    173	int ret;
    174
    175	ret = sscanf(buf, "%d %d", &cmd, &subcmd);
    176	if (ret != 2) {
    177		dev_err(dev, "Invalid values, expected: cmd subcmd\n");
    178		return -EINVAL;
    179	}
    180
    181	ret = intel_scu_ipc_dev_simple_command(scu, cmd, subcmd);
    182	if (ret)
    183		return ret;
    184
    185	return count;
    186}
    187static DEVICE_ATTR_WO(simplecmd);
    188
    189/**
    190 * northpeak_store() - Enable or disable Northpeak
    191 * @dev: Device under the attribute is
    192 * @attr: Attribute in question
    193 * @buf: Buffer holding data to be stored to the attribute
    194 * @count: Number of bytes in @buf
    195 *
    196 * Expects an unsigned integer. Non-zero enables Northpeak and zero
    197 * disables it.
    198 *
    199 * Return: Number number of bytes written (@count) or negative errno in
    200 *	   case of error.
    201 */
    202static ssize_t northpeak_store(struct device *dev, struct device_attribute *attr,
    203			       const char *buf, size_t count)
    204{
    205	struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
    206	struct intel_scu_ipc_dev *scu = pmc->scu;
    207	unsigned long val;
    208	int subcmd;
    209	int ret;
    210
    211	ret = kstrtoul(buf, 0, &val);
    212	if (ret)
    213		return ret;
    214
    215	/* Northpeak is enabled if subcmd == 1 and disabled if it is 0 */
    216	if (val)
    217		subcmd = 1;
    218	else
    219		subcmd = 0;
    220
    221	ret = intel_scu_ipc_dev_simple_command(scu, PMC_NORTHPEAK_CTRL, subcmd);
    222	if (ret)
    223		return ret;
    224
    225	return count;
    226}
    227static DEVICE_ATTR_WO(northpeak);
    228
    229static struct attribute *intel_pmc_attrs[] = {
    230	&dev_attr_northpeak.attr,
    231	&dev_attr_simplecmd.attr,
    232	NULL
    233};
    234
    235static const struct attribute_group intel_pmc_group = {
    236	.attrs = intel_pmc_attrs,
    237};
    238
    239static const struct attribute_group *intel_pmc_groups[] = {
    240	&intel_pmc_group,
    241	NULL
    242};
    243
    244static struct resource punit_res[6];
    245
    246static struct mfd_cell punit = {
    247	.name = "intel_punit_ipc",
    248	.resources = punit_res,
    249};
    250
    251static struct itco_wdt_platform_data tco_pdata = {
    252	.name = "Apollo Lake SoC",
    253	.version = 5,
    254	.no_reboot_use_pmc = true,
    255};
    256
    257static struct resource tco_res[2];
    258
    259static const struct mfd_cell tco = {
    260	.name = "iTCO_wdt",
    261	.ignore_resource_conflicts = true,
    262	.resources = tco_res,
    263	.num_resources = ARRAY_SIZE(tco_res),
    264	.platform_data = &tco_pdata,
    265	.pdata_size = sizeof(tco_pdata),
    266};
    267
    268static const struct resource telem_res[] = {
    269	DEFINE_RES_MEM(TELEM_PUNIT_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
    270	DEFINE_RES_MEM(TELEM_PMC_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
    271};
    272
    273static const struct mfd_cell telem = {
    274	.name = "intel_telemetry",
    275	.resources = telem_res,
    276	.num_resources = ARRAY_SIZE(telem_res),
    277};
    278
    279static int intel_pmc_get_tco_resources(struct platform_device *pdev)
    280{
    281	struct resource *res;
    282
    283	if (acpi_has_watchdog())
    284		return 0;
    285
    286	res = platform_get_resource(pdev, IORESOURCE_IO,
    287				    PLAT_RESOURCE_ACPI_IO_INDEX);
    288	if (!res) {
    289		dev_err(&pdev->dev, "Failed to get IO resource\n");
    290		return -EINVAL;
    291	}
    292
    293	tco_res[0].flags = IORESOURCE_IO;
    294	tco_res[0].start = res->start + TCO_BASE_OFFSET;
    295	tco_res[0].end = tco_res[0].start + TCO_REGS_SIZE - 1;
    296	tco_res[1].flags = IORESOURCE_IO;
    297	tco_res[1].start = res->start + SMI_EN_OFFSET;
    298	tco_res[1].end = tco_res[1].start + SMI_EN_SIZE - 1;
    299
    300	return 0;
    301}
    302
    303static int intel_pmc_get_resources(struct platform_device *pdev,
    304				   struct intel_pmc_dev *pmc,
    305				   struct intel_scu_ipc_data *scu_data)
    306{
    307	struct resource gcr_res;
    308	size_t npunit_res = 0;
    309	struct resource *res;
    310	int ret;
    311
    312	scu_data->irq = platform_get_irq_optional(pdev, 0);
    313
    314	res = platform_get_resource(pdev, IORESOURCE_MEM,
    315				    PLAT_RESOURCE_IPC_INDEX);
    316	if (!res) {
    317		dev_err(&pdev->dev, "Failed to get IPC resource\n");
    318		return -EINVAL;
    319	}
    320
    321	/* IPC registers */
    322	scu_data->mem.flags = res->flags;
    323	scu_data->mem.start = res->start;
    324	scu_data->mem.end = res->start + PLAT_RESOURCE_IPC_SIZE - 1;
    325
    326	/* GCR registers */
    327	gcr_res.flags = res->flags;
    328	gcr_res.start = res->start + PLAT_RESOURCE_GCR_OFFSET;
    329	gcr_res.end = gcr_res.start + PLAT_RESOURCE_GCR_SIZE - 1;
    330
    331	pmc->gcr_mem_base = devm_ioremap_resource(&pdev->dev, &gcr_res);
    332	if (IS_ERR(pmc->gcr_mem_base))
    333		return PTR_ERR(pmc->gcr_mem_base);
    334
    335	/* Only register iTCO watchdog if there is no WDAT ACPI table */
    336	ret = intel_pmc_get_tco_resources(pdev);
    337	if (ret)
    338		return ret;
    339
    340	/* BIOS data register */
    341	res = platform_get_resource(pdev, IORESOURCE_MEM,
    342				    PLAT_RESOURCE_BIOS_DATA_INDEX);
    343	if (!res) {
    344		dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS data\n");
    345		return -EINVAL;
    346	}
    347	punit_res[npunit_res++] = *res;
    348
    349	/* BIOS interface register */
    350	res = platform_get_resource(pdev, IORESOURCE_MEM,
    351				    PLAT_RESOURCE_BIOS_IFACE_INDEX);
    352	if (!res) {
    353		dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS interface\n");
    354		return -EINVAL;
    355	}
    356	punit_res[npunit_res++] = *res;
    357
    358	/* ISP data register, optional */
    359	res = platform_get_resource(pdev, IORESOURCE_MEM,
    360				    PLAT_RESOURCE_ISP_DATA_INDEX);
    361	if (res)
    362		punit_res[npunit_res++] = *res;
    363
    364	/* ISP interface register, optional */
    365	res = platform_get_resource(pdev, IORESOURCE_MEM,
    366				    PLAT_RESOURCE_ISP_IFACE_INDEX);
    367	if (res)
    368		punit_res[npunit_res++] = *res;
    369
    370	/* GTD data register, optional */
    371	res = platform_get_resource(pdev, IORESOURCE_MEM,
    372				    PLAT_RESOURCE_GTD_DATA_INDEX);
    373	if (res)
    374		punit_res[npunit_res++] = *res;
    375
    376	/* GTD interface register, optional */
    377	res = platform_get_resource(pdev, IORESOURCE_MEM,
    378				    PLAT_RESOURCE_GTD_IFACE_INDEX);
    379	if (res)
    380		punit_res[npunit_res++] = *res;
    381
    382	punit.num_resources = npunit_res;
    383
    384	/* Telemetry SSRAM is optional */
    385	res = platform_get_resource(pdev, IORESOURCE_MEM,
    386				    PLAT_RESOURCE_TELEM_SSRAM_INDEX);
    387	if (res)
    388		pmc->telem_base = res;
    389
    390	return 0;
    391}
    392
    393static int intel_pmc_create_devices(struct intel_pmc_dev *pmc)
    394{
    395	int ret;
    396
    397	if (!acpi_has_watchdog()) {
    398		ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &tco,
    399					   1, NULL, 0, NULL);
    400		if (ret)
    401			return ret;
    402	}
    403
    404	ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &punit, 1,
    405				   NULL, 0, NULL);
    406	if (ret)
    407		return ret;
    408
    409	if (pmc->telem_base) {
    410		ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO,
    411					   &telem, 1, pmc->telem_base, 0, NULL);
    412	}
    413
    414	return ret;
    415}
    416
    417static const struct acpi_device_id intel_pmc_acpi_ids[] = {
    418	{ "INT34D2" },
    419	{ }
    420};
    421MODULE_DEVICE_TABLE(acpi, intel_pmc_acpi_ids);
    422
    423static int intel_pmc_probe(struct platform_device *pdev)
    424{
    425	struct intel_scu_ipc_data scu_data = {};
    426	struct intel_pmc_dev *pmc;
    427	int ret;
    428
    429	pmc = devm_kzalloc(&pdev->dev, sizeof(*pmc), GFP_KERNEL);
    430	if (!pmc)
    431		return -ENOMEM;
    432
    433	pmc->dev = &pdev->dev;
    434	spin_lock_init(&pmc->gcr_lock);
    435
    436	ret = intel_pmc_get_resources(pdev, pmc, &scu_data);
    437	if (ret) {
    438		dev_err(&pdev->dev, "Failed to request resources\n");
    439		return ret;
    440	}
    441
    442	pmc->scu = devm_intel_scu_ipc_register(&pdev->dev, &scu_data);
    443	if (IS_ERR(pmc->scu))
    444		return PTR_ERR(pmc->scu);
    445
    446	platform_set_drvdata(pdev, pmc);
    447
    448	ret = intel_pmc_create_devices(pmc);
    449	if (ret)
    450		dev_err(&pdev->dev, "Failed to create PMC devices\n");
    451
    452	return ret;
    453}
    454
    455static struct platform_driver intel_pmc_driver = {
    456	.probe = intel_pmc_probe,
    457	.driver = {
    458		.name = "intel_pmc_bxt",
    459		.acpi_match_table = intel_pmc_acpi_ids,
    460		.dev_groups = intel_pmc_groups,
    461	},
    462};
    463module_platform_driver(intel_pmc_driver);
    464
    465MODULE_AUTHOR("Mika Westerberg <mika.westerberg@linux.intel.com>");
    466MODULE_AUTHOR("Zha Qipeng <qipeng.zha@intel.com>");
    467MODULE_DESCRIPTION("Intel Broxton PMC driver");
    468MODULE_LICENSE("GPL v2");