Instead of creating platform devices manually we can take advantage of MFD APIs that are designed exactly for this. Unfortunately the ACPI device includes separate resources for most of the subdevices so we cannot simply call mfd_add_devices() to create all of them but instead we need to call it separately for each device. Signed-off-by: Mika Westerberg <mika.westerberg@xxxxxxxxxxxxxxx> --- drivers/mfd/Kconfig | 1 + drivers/mfd/intel_pmc_bxt.c | 254 +++++++++++------------------------- 2 files changed, 75 insertions(+), 180 deletions(-) diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig index 93ab1b6a9733..04542feffe25 100644 --- a/drivers/mfd/Kconfig +++ b/drivers/mfd/Kconfig @@ -636,6 +636,7 @@ config MFD_INTEL_PMC_BXT tristate "Intel PMC Driver for Broxton" depends on X86 && X86_PLATFORM_DEVICES && ACPI select INTEL_SCU_IPC + select MFD_CORE help This driver provides support for PMC (Power Management Controller) on Intel Broxton and Apollo Lake. PMC is a diff --git a/drivers/mfd/intel_pmc_bxt.c b/drivers/mfd/intel_pmc_bxt.c index 8974fe46d6b5..850ed55deef0 100644 --- a/drivers/mfd/intel_pmc_bxt.c +++ b/drivers/mfd/intel_pmc_bxt.c @@ -16,6 +16,7 @@ #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> @@ -62,8 +63,6 @@ #define TELEM_SSRAM_SIZE 240 #define TELEM_PMC_SSRAM_OFFSET 0x1B00 #define TELEM_PUNIT_SSRAM_OFFSET 0x1A00 -#define TCO_PMC_OFFSET 0x08 -#define TCO_PMC_SIZE 0x04 /* PMC register bit definitions */ @@ -75,27 +74,19 @@ static struct intel_pmc_dev { struct device *dev; - /* The following PMC BARs share the same ACPI device with the IPC */ - resource_size_t acpi_io_base; - int acpi_io_size; - struct platform_device *tco_dev; + /* iTCO */ + struct resource tco_res[2]; /* gcr */ void __iomem *gcr_mem_base; - bool has_gcr_regs; spinlock_t gcr_lock; /* punit */ - struct platform_device *punit_dev; + struct resource punit_res[6]; unsigned int punit_res_count; /* Telemetry */ - resource_size_t telem_pmc_ssram_base; - resource_size_t telem_punit_ssram_base; - int telem_pmc_ssram_size; - int telem_punit_ssram_size; - u8 telem_res_inval; - struct platform_device *telemetry_dev; + struct resource *telem_base; } pmcdev; static inline u64 gcr_data_readq(u32 offset) @@ -105,7 +96,7 @@ static inline u64 gcr_data_readq(u32 offset) static inline int is_gcr_valid(u32 offset) { - if (!pmcdev.has_gcr_regs) + if (!pmcdev.gcr_mem_base) return -EACCES; if (offset > PLAT_RESOURCE_GCR_SIZE) @@ -254,141 +245,62 @@ static const struct attribute_group intel_pmc_group = { .attrs = intel_pmc_attrs, }; -static struct resource punit_res_array[] = { - /* Punit BIOS */ - { - .flags = IORESOURCE_MEM, - }, - { - .flags = IORESOURCE_MEM, - }, - /* Punit ISP */ - { - .flags = IORESOURCE_MEM, - }, - { - .flags = IORESOURCE_MEM, - }, - /* Punit GTD */ - { - .flags = IORESOURCE_MEM, - }, - { - .flags = IORESOURCE_MEM, - }, -}; - -#define TCO_RESOURCE_ACPI_IO 0 -#define TCO_RESOURCE_SMI_EN_IO 1 -#define TCO_RESOURCE_GCR_MEM 2 -static struct resource tco_res[] = { - /* ACPI - TCO */ - { - .flags = IORESOURCE_IO, - }, - /* ACPI - SMI */ - { - .flags = IORESOURCE_IO, - }, -}; - -static struct itco_wdt_platform_data tco_info = { - .name = "Apollo Lake SoC", - .version = 5, - .no_reboot_priv = &pmcdev, - .update_no_reboot_bit = update_no_reboot_bit, -}; - -#define TELEMETRY_RESOURCE_PUNIT_SSRAM 0 -#define TELEMETRY_RESOURCE_PMC_SSRAM 1 -static struct resource telemetry_res[] = { - /*Telemetry*/ - { - .flags = IORESOURCE_MEM, - }, - { - .flags = IORESOURCE_MEM, - }, -}; - static int pmc_create_punit_device(void) { - struct platform_device *pdev; - const struct platform_device_info pdevinfo = { - .parent = pmcdev.dev, + struct mfd_cell punit = { .name = PUNIT_DEVICE_NAME, - .id = -1, - .res = punit_res_array, - .num_res = pmcdev.punit_res_count, - }; + .resources = pmcdev.punit_res, + .num_resources = pmcdev.punit_res_count, + }; - pdev = platform_device_register_full(&pdevinfo); - if (IS_ERR(pdev)) - return PTR_ERR(pdev); + if (!pmcdev.punit_res_count) + return 0; - pmcdev.punit_dev = pdev; - - return 0; + return devm_mfd_add_devices(pmcdev.dev, PLATFORM_DEVID_AUTO, &punit, 1, + NULL, 0, NULL); } static int pmc_create_tco_device(void) { - struct platform_device *pdev; - struct resource *res; - const struct platform_device_info pdevinfo = { - .parent = pmcdev.dev, + struct itco_wdt_platform_data tco_info = { + .name = "Apollo Lake SoC", + .version = 5, + .no_reboot_priv = &pmcdev, + .update_no_reboot_bit = update_no_reboot_bit, + }; + struct mfd_cell tco = { .name = TCO_DEVICE_NAME, - .id = -1, - .res = tco_res, - .num_res = ARRAY_SIZE(tco_res), - .data = &tco_info, - .size_data = sizeof(tco_info), - }; - - res = tco_res + TCO_RESOURCE_ACPI_IO; - res->start = pmcdev.acpi_io_base + TCO_BASE_OFFSET; - res->end = res->start + TCO_REGS_SIZE - 1; - - res = tco_res + TCO_RESOURCE_SMI_EN_IO; - res->start = pmcdev.acpi_io_base + SMI_EN_OFFSET; - res->end = res->start + SMI_EN_SIZE - 1; - - pdev = platform_device_register_full(&pdevinfo); - if (IS_ERR(pdev)) - return PTR_ERR(pdev); - - pmcdev.tco_dev = pdev; - - return 0; + .ignore_resource_conflicts = true, + .platform_data = &tco_info, + .pdata_size = sizeof(tco_info), + .resources = pmcdev.tco_res, + .num_resources = ARRAY_SIZE(pmcdev.tco_res), + }; + + if (!pmcdev.tco_res[0].start) + return 0; + + return devm_mfd_add_devices(pmcdev.dev, PLATFORM_DEVID_AUTO, &tco, 1, + NULL, 0, NULL); } static int pmc_create_telemetry_device(void) { - struct platform_device *pdev; - struct resource *res; - const struct platform_device_info pdevinfo = { - .parent = pmcdev.dev, + struct resource telem_res[] = { + DEFINE_RES_MEM(TELEM_PUNIT_SSRAM_OFFSET, TELEM_SSRAM_SIZE), + DEFINE_RES_MEM(TELEM_PMC_SSRAM_OFFSET, TELEM_SSRAM_SIZE), + }; + struct mfd_cell telem = { .name = TELEMETRY_DEVICE_NAME, - .id = -1, - .res = telemetry_res, - .num_res = ARRAY_SIZE(telemetry_res), - }; - - res = telemetry_res + TELEMETRY_RESOURCE_PUNIT_SSRAM; - res->start = pmcdev.telem_punit_ssram_base; - res->end = res->start + pmcdev.telem_punit_ssram_size - 1; + .resources = telem_res, + .num_resources = ARRAY_SIZE(telem_res), + }; - res = telemetry_res + TELEMETRY_RESOURCE_PMC_SSRAM; - res->start = pmcdev.telem_pmc_ssram_base; - res->end = res->start + pmcdev.telem_pmc_ssram_size - 1; + if (!pmcdev.telem_base) + return 0; - pdev = platform_device_register_full(&pdevinfo); - if (IS_ERR(pdev)) - return PTR_ERR(pdev); - - pmcdev.telemetry_dev = pdev; - - return 0; + return devm_mfd_add_devices(pmcdev.dev, PLATFORM_DEVID_AUTO, &telem, 1, + pmcdev.telem_base, 0, NULL); } static int pmc_create_devices(void) @@ -407,19 +319,12 @@ static int pmc_create_devices(void) ret = pmc_create_punit_device(); if (ret) { dev_err(pmcdev.dev, "Failed to add punit platform device\n"); - platform_device_unregister(pmcdev.tco_dev); return ret; } - if (!pmcdev.telem_res_inval) { - ret = pmc_create_telemetry_device(); - if (ret) { - dev_warn(pmcdev.dev, - "Failed to add telemetry platform device\n"); - platform_device_unregister(pmcdev.punit_dev); - platform_device_unregister(pmcdev.tco_dev); - } - } + ret = pmc_create_telemetry_device(); + if (ret) + dev_warn(pmcdev.dev, "Failed to add telemetry platform device\n"); return ret; } @@ -427,20 +332,25 @@ static int pmc_create_devices(void) static int pmc_plat_get_res(struct platform_device *pdev, struct intel_scu_ipc_pdata *pdata) { - struct resource *res, *punit_res = punit_res_array; + struct resource *res, *punit_res = pmcdev.punit_res; + struct resource *tco_res = pmcdev.tco_res; void __iomem *addr; int size; res = platform_get_resource(pdev, IORESOURCE_IO, PLAT_RESOURCE_ACPI_IO_INDEX); if (!res) { - dev_err(&pdev->dev, "Failed to get io resource\n"); + dev_err(&pdev->dev, "Failed to get IO resource\n"); return -ENXIO; + } - size = resource_size(res); - pmcdev.acpi_io_base = res->start; - pmcdev.acpi_io_size = size; - dev_info(&pdev->dev, "io res: %pR\n", res); + 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; + dev_dbg(&pdev->dev, "IO: %pR\n", res); pmcdev.punit_res_count = 0; @@ -448,28 +358,28 @@ static int pmc_plat_get_res(struct platform_device *pdev, res = platform_get_resource(pdev, IORESOURCE_MEM, PLAT_RESOURCE_BIOS_DATA_INDEX); if (!res) { - dev_err(&pdev->dev, "Failed to get res of punit BIOS data\n"); + dev_err(&pdev->dev, "Failed to get res of P-unit BIOS data\n"); return -ENXIO; } punit_res[pmcdev.punit_res_count++] = *res; - dev_info(&pdev->dev, "punit BIOS data res: %pR\n", res); + dev_dbg(&pdev->dev, "P-unit BIOS data: %pR\n", res); /* This is index 1 to cover BIOS interface register */ res = platform_get_resource(pdev, IORESOURCE_MEM, PLAT_RESOURCE_BIOS_IFACE_INDEX); if (!res) { - dev_err(&pdev->dev, "Failed to get res of punit BIOS iface\n"); + dev_err(&pdev->dev, "Failed to get res of P-unit BIOS iface\n"); return -ENXIO; } punit_res[pmcdev.punit_res_count++] = *res; - dev_info(&pdev->dev, "punit BIOS interface res: %pR\n", res); + dev_dbg(&pdev->dev, "P-unit BIOS interface: %pR\n", res); /* This is index 2 to cover ISP data register, optional */ res = platform_get_resource(pdev, IORESOURCE_MEM, PLAT_RESOURCE_ISP_DATA_INDEX); if (res) { punit_res[pmcdev.punit_res_count++] = *res; - dev_info(&pdev->dev, "punit ISP data res: %pR\n", res); + dev_dbg(&pdev->dev, "P-unit ISP data: %pR\n", res); } /* This is index 3 to cover ISP interface register, optional */ @@ -477,7 +387,7 @@ static int pmc_plat_get_res(struct platform_device *pdev, PLAT_RESOURCE_ISP_IFACE_INDEX); if (res) { punit_res[pmcdev.punit_res_count++] = *res; - dev_info(&pdev->dev, "punit ISP interface res: %pR\n", res); + dev_dbg(&pdev->dev, "P-unit ISP interface: %pR\n", res); } /* This is index 4 to cover GTD data register, optional */ @@ -485,7 +395,7 @@ static int pmc_plat_get_res(struct platform_device *pdev, PLAT_RESOURCE_GTD_DATA_INDEX); if (res) { punit_res[pmcdev.punit_res_count++] = *res; - dev_info(&pdev->dev, "punit GTD data res: %pR\n", res); + dev_dbg(&pdev->dev, "P-unit GTD data: %pR\n", res); } /* This is index 5 to cover GTD interface register, optional */ @@ -493,13 +403,13 @@ static int pmc_plat_get_res(struct platform_device *pdev, PLAT_RESOURCE_GTD_IFACE_INDEX); if (res) { punit_res[pmcdev.punit_res_count++] = *res; - dev_info(&pdev->dev, "punit GTD interface res: %pR\n", res); + dev_dbg(&pdev->dev, "P-unit GTD interface: %pR\n", res); } res = platform_get_resource(pdev, IORESOURCE_MEM, PLAT_RESOURCE_IPC_INDEX); if (!res) { - dev_err(&pdev->dev, "Failed to get ipc resource\n"); + dev_err(&pdev->dev, "Failed to get IPC resource\n"); return -ENXIO; } size = PLAT_RESOURCE_IPC_SIZE + PLAT_RESOURCE_GCR_SIZE; @@ -512,22 +422,15 @@ static int pmc_plat_get_res(struct platform_device *pdev, pdata->ipc_regs = addr; pmcdev.gcr_mem_base = addr + PLAT_RESOURCE_GCR_OFFSET; - dev_info(&pdev->dev, "ipc res: %pR\n", res); + dev_dbg(&pdev->dev, "IPC: %pR\n", res); - pmcdev.telem_res_inval = 0; res = platform_get_resource(pdev, IORESOURCE_MEM, PLAT_RESOURCE_TELEM_SSRAM_INDEX); if (!res) { - dev_err(&pdev->dev, "Failed to get telemetry ssram resource\n"); - pmcdev.telem_res_inval = 1; + dev_err(&pdev->dev, "Failed to get telemetry SSRAM resource\n"); } else { - pmcdev.telem_punit_ssram_base = res->start + - TELEM_PUNIT_SSRAM_OFFSET; - pmcdev.telem_punit_ssram_size = TELEM_SSRAM_SIZE; - pmcdev.telem_pmc_ssram_base = res->start + - TELEM_PMC_SSRAM_OFFSET; - pmcdev.telem_pmc_ssram_size = TELEM_SSRAM_SIZE; - dev_info(&pdev->dev, "telemetry ssram res: %pR\n", res); + dev_dbg(&pdev->dev, "Telemetry SSRAM: %pR\n", res); + pmcdev.telem_base = res; } return 0; @@ -543,7 +446,7 @@ int intel_pmc_s0ix_counter_read(u64 *data) { u64 deep, shlw; - if (!pmcdev.has_gcr_regs) + if (!pmcdev.gcr_mem_base) return -EACCES; deep = gcr_data_readq(PMC_GCR_TELEM_DEEP_S0IX_REG); @@ -597,19 +500,13 @@ static int intel_pmc_probe(struct platform_device *pdev) if (ret) { dev_err(&pdev->dev, "Failed to create sysfs group %d\n", ret); - goto err_devs; + goto err_ipc; } - pmcdev.has_gcr_regs = true; - return 0; err_ipc: intel_scu_ipc_remove(scu); -err_devs: - platform_device_unregister(pmcdev.tco_dev); - platform_device_unregister(pmcdev.punit_dev); - platform_device_unregister(pmcdev.telemetry_dev); return ret; } @@ -617,9 +514,6 @@ static int intel_pmc_probe(struct platform_device *pdev) static int intel_pmc_remove(struct platform_device *pdev) { sysfs_remove_group(&pdev->dev.kobj, &intel_pmc_group); - platform_device_unregister(pmcdev.tco_dev); - platform_device_unregister(pmcdev.punit_dev); - platform_device_unregister(pmcdev.telemetry_dev); intel_scu_ipc_remove(platform_get_drvdata(pdev)); pmcdev.dev = NULL; return 0; -- 2.24.0