469 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			469 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Driver for the Intel Broxton PMC
 | |
|  *
 | |
|  * (C) Copyright 2014 - 2020 Intel Corporation
 | |
|  *
 | |
|  * This driver is based on Intel SCU IPC driver (intel_scu_ipc.c) by
 | |
|  * Sreedhara DS <sreedhara.ds@intel.com>
 | |
|  *
 | |
|  * The PMC (Power Management Controller) running on the ARC processor
 | |
|  * communicates with another entity running in the IA (Intel Architecture)
 | |
|  * core through an IPC (Intel Processor Communications) mechanism which in
 | |
|  * turn sends messages between the IA and the PMC.
 | |
|  */
 | |
| 
 | |
| #include <linux/acpi.h>
 | |
| #include <linux/delay.h>
 | |
| #include <linux/errno.h>
 | |
| #include <linux/interrupt.h>
 | |
| #include <linux/io-64-nonatomic-lo-hi.h>
 | |
| #include <linux/mfd/core.h>
 | |
| #include <linux/mfd/intel_pmc_bxt.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/platform_device.h>
 | |
| #include <linux/platform_data/itco_wdt.h>
 | |
| 
 | |
| #include <asm/intel_scu_ipc.h>
 | |
| 
 | |
| /* Residency with clock rate at 19.2MHz to usecs */
 | |
| #define S0IX_RESIDENCY_IN_USECS(d, s)		\
 | |
| ({						\
 | |
| 	u64 result = 10ull * ((d) + (s));	\
 | |
| 	do_div(result, 192);			\
 | |
| 	result;					\
 | |
| })
 | |
| 
 | |
| /* Resources exported from IFWI */
 | |
| #define PLAT_RESOURCE_IPC_INDEX		0
 | |
| #define PLAT_RESOURCE_IPC_SIZE		0x1000
 | |
| #define PLAT_RESOURCE_GCR_OFFSET	0x1000
 | |
| #define PLAT_RESOURCE_GCR_SIZE		0x1000
 | |
| #define PLAT_RESOURCE_BIOS_DATA_INDEX	1
 | |
| #define PLAT_RESOURCE_BIOS_IFACE_INDEX	2
 | |
| #define PLAT_RESOURCE_TELEM_SSRAM_INDEX	3
 | |
| #define PLAT_RESOURCE_ISP_DATA_INDEX	4
 | |
| #define PLAT_RESOURCE_ISP_IFACE_INDEX	5
 | |
| #define PLAT_RESOURCE_GTD_DATA_INDEX	6
 | |
| #define PLAT_RESOURCE_GTD_IFACE_INDEX	7
 | |
| #define PLAT_RESOURCE_ACPI_IO_INDEX	0
 | |
| 
 | |
| /*
 | |
|  * BIOS does not create an ACPI device for each PMC function, but
 | |
|  * exports multiple resources from one ACPI device (IPC) for multiple
 | |
|  * functions. This driver is responsible for creating a child device and
 | |
|  * to export resources for those functions.
 | |
|  */
 | |
| #define SMI_EN_OFFSET			0x0040
 | |
| #define SMI_EN_SIZE			4
 | |
| #define TCO_BASE_OFFSET			0x0060
 | |
| #define TCO_REGS_SIZE			16
 | |
| #define TELEM_SSRAM_SIZE		240
 | |
| #define TELEM_PMC_SSRAM_OFFSET		0x1b00
 | |
| #define TELEM_PUNIT_SSRAM_OFFSET	0x1a00
 | |
| 
 | |
| /* Commands */
 | |
| #define PMC_NORTHPEAK_CTRL		0xed
 | |
| 
 | |
| static inline bool is_gcr_valid(u32 offset)
 | |
| {
 | |
| 	return offset < PLAT_RESOURCE_GCR_SIZE - 8;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * intel_pmc_gcr_read64() - Read a 64-bit PMC GCR register
 | |
|  * @pmc: PMC device pointer
 | |
|  * @offset: offset of GCR register from GCR address base
 | |
|  * @data: data pointer for storing the register output
 | |
|  *
 | |
|  * Reads the 64-bit PMC GCR register at given offset.
 | |
|  *
 | |
|  * Return: Negative value on error or 0 on success.
 | |
|  */
 | |
| int intel_pmc_gcr_read64(struct intel_pmc_dev *pmc, u32 offset, u64 *data)
 | |
| {
 | |
| 	if (!is_gcr_valid(offset))
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	spin_lock(&pmc->gcr_lock);
 | |
| 	*data = readq(pmc->gcr_mem_base + offset);
 | |
| 	spin_unlock(&pmc->gcr_lock);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(intel_pmc_gcr_read64);
 | |
| 
 | |
| /**
 | |
|  * intel_pmc_gcr_update() - Update PMC GCR register bits
 | |
|  * @pmc: PMC device pointer
 | |
|  * @offset: offset of GCR register from GCR address base
 | |
|  * @mask: bit mask for update operation
 | |
|  * @val: update value
 | |
|  *
 | |
|  * Updates the bits of given GCR register as specified by
 | |
|  * @mask and @val.
 | |
|  *
 | |
|  * Return: Negative value on error or 0 on success.
 | |
|  */
 | |
| int intel_pmc_gcr_update(struct intel_pmc_dev *pmc, u32 offset, u32 mask, u32 val)
 | |
| {
 | |
| 	u32 new_val;
 | |
| 
 | |
| 	if (!is_gcr_valid(offset))
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	spin_lock(&pmc->gcr_lock);
 | |
| 	new_val = readl(pmc->gcr_mem_base + offset);
 | |
| 
 | |
| 	new_val = (new_val & ~mask) | (val & mask);
 | |
| 	writel(new_val, pmc->gcr_mem_base + offset);
 | |
| 
 | |
| 	new_val = readl(pmc->gcr_mem_base + offset);
 | |
| 	spin_unlock(&pmc->gcr_lock);
 | |
| 
 | |
| 	/* Check whether the bit update is successful */
 | |
| 	return (new_val & mask) != (val & mask) ? -EIO : 0;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(intel_pmc_gcr_update);
 | |
| 
 | |
| /**
 | |
|  * intel_pmc_s0ix_counter_read() - Read S0ix residency
 | |
|  * @pmc: PMC device pointer
 | |
|  * @data: Out param that contains current S0ix residency count.
 | |
|  *
 | |
|  * Writes to @data how many usecs the system has been in low-power S0ix
 | |
|  * state.
 | |
|  *
 | |
|  * Return: An error code or 0 on success.
 | |
|  */
 | |
| int intel_pmc_s0ix_counter_read(struct intel_pmc_dev *pmc, u64 *data)
 | |
| {
 | |
| 	u64 deep, shlw;
 | |
| 
 | |
| 	spin_lock(&pmc->gcr_lock);
 | |
| 	deep = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_DEEP_S0IX_REG);
 | |
| 	shlw = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_SHLW_S0IX_REG);
 | |
| 	spin_unlock(&pmc->gcr_lock);
 | |
| 
 | |
| 	*data = S0IX_RESIDENCY_IN_USECS(deep, shlw);
 | |
| 	return 0;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(intel_pmc_s0ix_counter_read);
 | |
| 
 | |
| /**
 | |
|  * simplecmd_store() - Send a simple IPC command
 | |
|  * @dev: Device under the attribute is
 | |
|  * @attr: Attribute in question
 | |
|  * @buf: Buffer holding data to be stored to the attribute
 | |
|  * @count: Number of bytes in @buf
 | |
|  *
 | |
|  * Expects a string with two integers separated with space. These two
 | |
|  * values hold command and subcommand that is send to PMC.
 | |
|  *
 | |
|  * Return: Number number of bytes written (@count) or negative errno in
 | |
|  *	   case of error.
 | |
|  */
 | |
| static ssize_t simplecmd_store(struct device *dev, struct device_attribute *attr,
 | |
| 			       const char *buf, size_t count)
 | |
| {
 | |
| 	struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
 | |
| 	struct intel_scu_ipc_dev *scu = pmc->scu;
 | |
| 	int subcmd;
 | |
| 	int cmd;
 | |
| 	int ret;
 | |
| 
 | |
| 	ret = sscanf(buf, "%d %d", &cmd, &subcmd);
 | |
| 	if (ret != 2) {
 | |
| 		dev_err(dev, "Invalid values, expected: cmd subcmd\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	ret = intel_scu_ipc_dev_simple_command(scu, cmd, subcmd);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	return count;
 | |
| }
 | |
| static DEVICE_ATTR_WO(simplecmd);
 | |
| 
 | |
| /**
 | |
|  * northpeak_store() - Enable or disable Northpeak
 | |
|  * @dev: Device under the attribute is
 | |
|  * @attr: Attribute in question
 | |
|  * @buf: Buffer holding data to be stored to the attribute
 | |
|  * @count: Number of bytes in @buf
 | |
|  *
 | |
|  * Expects an unsigned integer. Non-zero enables Northpeak and zero
 | |
|  * disables it.
 | |
|  *
 | |
|  * Return: Number number of bytes written (@count) or negative errno in
 | |
|  *	   case of error.
 | |
|  */
 | |
| static ssize_t northpeak_store(struct device *dev, struct device_attribute *attr,
 | |
| 			       const char *buf, size_t count)
 | |
| {
 | |
| 	struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
 | |
| 	struct intel_scu_ipc_dev *scu = pmc->scu;
 | |
| 	unsigned long val;
 | |
| 	int subcmd;
 | |
| 	int ret;
 | |
| 
 | |
| 	ret = kstrtoul(buf, 0, &val);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	/* Northpeak is enabled if subcmd == 1 and disabled if it is 0 */
 | |
| 	if (val)
 | |
| 		subcmd = 1;
 | |
| 	else
 | |
| 		subcmd = 0;
 | |
| 
 | |
| 	ret = intel_scu_ipc_dev_simple_command(scu, PMC_NORTHPEAK_CTRL, subcmd);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	return count;
 | |
| }
 | |
| static DEVICE_ATTR_WO(northpeak);
 | |
| 
 | |
| static struct attribute *intel_pmc_attrs[] = {
 | |
| 	&dev_attr_northpeak.attr,
 | |
| 	&dev_attr_simplecmd.attr,
 | |
| 	NULL
 | |
| };
 | |
| 
 | |
| static const struct attribute_group intel_pmc_group = {
 | |
| 	.attrs = intel_pmc_attrs,
 | |
| };
 | |
| 
 | |
| static const struct attribute_group *intel_pmc_groups[] = {
 | |
| 	&intel_pmc_group,
 | |
| 	NULL
 | |
| };
 | |
| 
 | |
| static struct resource punit_res[6];
 | |
| 
 | |
| static struct mfd_cell punit = {
 | |
| 	.name = "intel_punit_ipc",
 | |
| 	.resources = punit_res,
 | |
| };
 | |
| 
 | |
| static struct itco_wdt_platform_data tco_pdata = {
 | |
| 	.name = "Apollo Lake SoC",
 | |
| 	.version = 5,
 | |
| 	.no_reboot_use_pmc = true,
 | |
| };
 | |
| 
 | |
| static struct resource tco_res[2];
 | |
| 
 | |
| static const struct mfd_cell tco = {
 | |
| 	.name = "iTCO_wdt",
 | |
| 	.ignore_resource_conflicts = true,
 | |
| 	.resources = tco_res,
 | |
| 	.num_resources = ARRAY_SIZE(tco_res),
 | |
| 	.platform_data = &tco_pdata,
 | |
| 	.pdata_size = sizeof(tco_pdata),
 | |
| };
 | |
| 
 | |
| static const struct resource telem_res[] = {
 | |
| 	DEFINE_RES_MEM(TELEM_PUNIT_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
 | |
| 	DEFINE_RES_MEM(TELEM_PMC_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
 | |
| };
 | |
| 
 | |
| static const struct mfd_cell telem = {
 | |
| 	.name = "intel_telemetry",
 | |
| 	.resources = telem_res,
 | |
| 	.num_resources = ARRAY_SIZE(telem_res),
 | |
| };
 | |
| 
 | |
| static int intel_pmc_get_tco_resources(struct platform_device *pdev)
 | |
| {
 | |
| 	struct resource *res;
 | |
| 
 | |
| 	if (acpi_has_watchdog())
 | |
| 		return 0;
 | |
| 
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_IO,
 | |
| 				    PLAT_RESOURCE_ACPI_IO_INDEX);
 | |
| 	if (!res) {
 | |
| 		dev_err(&pdev->dev, "Failed to get IO resource\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	tco_res[0].flags = IORESOURCE_IO;
 | |
| 	tco_res[0].start = res->start + TCO_BASE_OFFSET;
 | |
| 	tco_res[0].end = tco_res[0].start + TCO_REGS_SIZE - 1;
 | |
| 	tco_res[1].flags = IORESOURCE_IO;
 | |
| 	tco_res[1].start = res->start + SMI_EN_OFFSET;
 | |
| 	tco_res[1].end = tco_res[1].start + SMI_EN_SIZE - 1;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int intel_pmc_get_resources(struct platform_device *pdev,
 | |
| 				   struct intel_pmc_dev *pmc,
 | |
| 				   struct intel_scu_ipc_data *scu_data)
 | |
| {
 | |
| 	struct resource gcr_res;
 | |
| 	size_t npunit_res = 0;
 | |
| 	struct resource *res;
 | |
| 	int ret;
 | |
| 
 | |
| 	scu_data->irq = platform_get_irq_optional(pdev, 0);
 | |
| 
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_IPC_INDEX);
 | |
| 	if (!res) {
 | |
| 		dev_err(&pdev->dev, "Failed to get IPC resource\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	/* IPC registers */
 | |
| 	scu_data->mem.flags = res->flags;
 | |
| 	scu_data->mem.start = res->start;
 | |
| 	scu_data->mem.end = res->start + PLAT_RESOURCE_IPC_SIZE - 1;
 | |
| 
 | |
| 	/* GCR registers */
 | |
| 	gcr_res.flags = res->flags;
 | |
| 	gcr_res.start = res->start + PLAT_RESOURCE_GCR_OFFSET;
 | |
| 	gcr_res.end = gcr_res.start + PLAT_RESOURCE_GCR_SIZE - 1;
 | |
| 
 | |
| 	pmc->gcr_mem_base = devm_ioremap_resource(&pdev->dev, &gcr_res);
 | |
| 	if (IS_ERR(pmc->gcr_mem_base))
 | |
| 		return PTR_ERR(pmc->gcr_mem_base);
 | |
| 
 | |
| 	/* Only register iTCO watchdog if there is no WDAT ACPI table */
 | |
| 	ret = intel_pmc_get_tco_resources(pdev);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	/* BIOS data register */
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_BIOS_DATA_INDEX);
 | |
| 	if (!res) {
 | |
| 		dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS data\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 	punit_res[npunit_res++] = *res;
 | |
| 
 | |
| 	/* BIOS interface register */
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_BIOS_IFACE_INDEX);
 | |
| 	if (!res) {
 | |
| 		dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS interface\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 	punit_res[npunit_res++] = *res;
 | |
| 
 | |
| 	/* ISP data register, optional */
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_ISP_DATA_INDEX);
 | |
| 	if (res)
 | |
| 		punit_res[npunit_res++] = *res;
 | |
| 
 | |
| 	/* ISP interface register, optional */
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_ISP_IFACE_INDEX);
 | |
| 	if (res)
 | |
| 		punit_res[npunit_res++] = *res;
 | |
| 
 | |
| 	/* GTD data register, optional */
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_GTD_DATA_INDEX);
 | |
| 	if (res)
 | |
| 		punit_res[npunit_res++] = *res;
 | |
| 
 | |
| 	/* GTD interface register, optional */
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_GTD_IFACE_INDEX);
 | |
| 	if (res)
 | |
| 		punit_res[npunit_res++] = *res;
 | |
| 
 | |
| 	punit.num_resources = npunit_res;
 | |
| 
 | |
| 	/* Telemetry SSRAM is optional */
 | |
| 	res = platform_get_resource(pdev, IORESOURCE_MEM,
 | |
| 				    PLAT_RESOURCE_TELEM_SSRAM_INDEX);
 | |
| 	if (res)
 | |
| 		pmc->telem_base = res;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int intel_pmc_create_devices(struct intel_pmc_dev *pmc)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	if (!acpi_has_watchdog()) {
 | |
| 		ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &tco,
 | |
| 					   1, NULL, 0, NULL);
 | |
| 		if (ret)
 | |
| 			return ret;
 | |
| 	}
 | |
| 
 | |
| 	ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &punit, 1,
 | |
| 				   NULL, 0, NULL);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	if (pmc->telem_base) {
 | |
| 		ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO,
 | |
| 					   &telem, 1, pmc->telem_base, 0, NULL);
 | |
| 	}
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static const struct acpi_device_id intel_pmc_acpi_ids[] = {
 | |
| 	{ "INT34D2" },
 | |
| 	{ }
 | |
| };
 | |
| MODULE_DEVICE_TABLE(acpi, intel_pmc_acpi_ids);
 | |
| 
 | |
| static int intel_pmc_probe(struct platform_device *pdev)
 | |
| {
 | |
| 	struct intel_scu_ipc_data scu_data = {};
 | |
| 	struct intel_pmc_dev *pmc;
 | |
| 	int ret;
 | |
| 
 | |
| 	pmc = devm_kzalloc(&pdev->dev, sizeof(*pmc), GFP_KERNEL);
 | |
| 	if (!pmc)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	pmc->dev = &pdev->dev;
 | |
| 	spin_lock_init(&pmc->gcr_lock);
 | |
| 
 | |
| 	ret = intel_pmc_get_resources(pdev, pmc, &scu_data);
 | |
| 	if (ret) {
 | |
| 		dev_err(&pdev->dev, "Failed to request resources\n");
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	pmc->scu = devm_intel_scu_ipc_register(&pdev->dev, &scu_data);
 | |
| 	if (IS_ERR(pmc->scu))
 | |
| 		return PTR_ERR(pmc->scu);
 | |
| 
 | |
| 	platform_set_drvdata(pdev, pmc);
 | |
| 
 | |
| 	ret = intel_pmc_create_devices(pmc);
 | |
| 	if (ret)
 | |
| 		dev_err(&pdev->dev, "Failed to create PMC devices\n");
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static struct platform_driver intel_pmc_driver = {
 | |
| 	.probe = intel_pmc_probe,
 | |
| 	.driver = {
 | |
| 		.name = "intel_pmc_bxt",
 | |
| 		.acpi_match_table = intel_pmc_acpi_ids,
 | |
| 		.dev_groups = intel_pmc_groups,
 | |
| 	},
 | |
| };
 | |
| module_platform_driver(intel_pmc_driver);
 | |
| 
 | |
| MODULE_AUTHOR("Mika Westerberg <mika.westerberg@linux.intel.com>");
 | |
| MODULE_AUTHOR("Zha Qipeng <qipeng.zha@intel.com>");
 | |
| MODULE_DESCRIPTION("Intel Broxton PMC driver");
 | |
| MODULE_LICENSE("GPL v2");
 |