Commit a01486dc authored by David E. Box's avatar David E. Box Committed by Hans de Goede

platform/x86/intel/pmc: Cleanup SSRAM discovery

Clean up the code handling SSRAM discovery. Handle all resource allocation
and cleanup in pmc_core_ssram_get_pmc(). Return the error status from this
function but only fail the init if we fail to discover the primary PMC.
Signed-off-by: default avatarDavid E. Box <david.e.box@linux.intel.com>
Reviewed-by: default avatarIlpo Järvinen <ilpo.jarvinen@linux.intel.com>
Link: https://lore.kernel.org/r/20231129222132.2331261-14-david.e.box@linux.intel.comSigned-off-by: default avatarHans de Goede <hdegoede@redhat.com>
parent 9512920a
...@@ -8,6 +8,7 @@ ...@@ -8,6 +8,7 @@
* *
*/ */
#include <linux/cleanup.h>
#include <linux/pci.h> #include <linux/pci.h>
#include <linux/io-64-nonatomic-lo-hi.h> #include <linux/io-64-nonatomic-lo-hi.h>
...@@ -21,6 +22,8 @@ ...@@ -21,6 +22,8 @@
#define SSRAM_IOE_OFFSET 0x68 #define SSRAM_IOE_OFFSET 0x68
#define SSRAM_DEVID_OFFSET 0x70 #define SSRAM_DEVID_OFFSET 0x70
DEFINE_FREE(pmc_core_iounmap, void __iomem *, iounmap(_T));
static const struct pmc_reg_map *pmc_core_find_regmap(struct pmc_info *list, u16 devid) static const struct pmc_reg_map *pmc_core_find_regmap(struct pmc_info *list, u16 devid)
{ {
for (; list->map; ++list) for (; list->map; ++list)
...@@ -65,44 +68,49 @@ pmc_core_pmc_add(struct pmc_dev *pmcdev, u64 pwrm_base, ...@@ -65,44 +68,49 @@ pmc_core_pmc_add(struct pmc_dev *pmcdev, u64 pwrm_base,
return 0; return 0;
} }
static void static int
pmc_core_ssram_get_pmc(struct pmc_dev *pmcdev, void __iomem *ssram, u32 offset, pmc_core_ssram_get_pmc(struct pmc_dev *pmcdev, int pmc_idx, u32 offset)
int pmc_idx)
{ {
u64 pwrm_base; struct pci_dev *ssram_pcidev = pmcdev->ssram_pcidev;
void __iomem __free(pmc_core_iounmap) *tmp_ssram = NULL;
void __iomem __free(pmc_core_iounmap) *ssram = NULL;
const struct pmc_reg_map *map;
u64 ssram_base, pwrm_base;
u16 devid; u16 devid;
if (pmc_idx != PMC_IDX_SOC) { if (!pmcdev->regmap_list)
u64 ssram_base = get_base(ssram, offset); return -ENOENT;
if (!ssram_base) ssram_base = ssram_pcidev->resource[0].start;
return; tmp_ssram = ioremap(ssram_base, SSRAM_HDR_SIZE);
if (pmc_idx != PMC_IDX_MAIN) {
/*
* The secondary PMC BARS (which are behind hidden PCI devices)
* are read from fixed offsets in MMIO of the primary PMC BAR.
*/
ssram_base = get_base(tmp_ssram, offset);
ssram = ioremap(ssram_base, SSRAM_HDR_SIZE); ssram = ioremap(ssram_base, SSRAM_HDR_SIZE);
if (!ssram) if (!ssram)
return; return -ENOMEM;
} else {
ssram = no_free_ptr(tmp_ssram);
} }
pwrm_base = get_base(ssram, SSRAM_PWRM_OFFSET); pwrm_base = get_base(ssram, SSRAM_PWRM_OFFSET);
devid = readw(ssram + SSRAM_DEVID_OFFSET); devid = readw(ssram + SSRAM_DEVID_OFFSET);
if (pmcdev->regmap_list) {
const struct pmc_reg_map *map;
map = pmc_core_find_regmap(pmcdev->regmap_list, devid); map = pmc_core_find_regmap(pmcdev->regmap_list, devid);
if (map) if (!map)
pmc_core_pmc_add(pmcdev, pwrm_base, map, pmc_idx); return -ENODEV;
}
if (pmc_idx != PMC_IDX_SOC) return pmc_core_pmc_add(pmcdev, pwrm_base, map, PMC_IDX_MAIN);
iounmap(ssram);
} }
int pmc_core_ssram_init(struct pmc_dev *pmcdev) int pmc_core_ssram_init(struct pmc_dev *pmcdev)
{ {
void __iomem *ssram;
struct pci_dev *pcidev; struct pci_dev *pcidev;
u64 ssram_base;
int ret; int ret;
pcidev = pci_get_domain_bus_and_slot(0, 0, PCI_DEVFN(20, 2)); pcidev = pci_get_domain_bus_and_slot(0, 0, PCI_DEVFN(20, 2));
...@@ -113,18 +121,14 @@ int pmc_core_ssram_init(struct pmc_dev *pmcdev) ...@@ -113,18 +121,14 @@ int pmc_core_ssram_init(struct pmc_dev *pmcdev)
if (ret) if (ret)
goto release_dev; goto release_dev;
ssram_base = pcidev->resource[0].start;
ssram = ioremap(ssram_base, SSRAM_HDR_SIZE);
if (!ssram)
goto disable_dev;
pmcdev->ssram_pcidev = pcidev; pmcdev->ssram_pcidev = pcidev;
pmc_core_ssram_get_pmc(pmcdev, ssram, 0, PMC_IDX_SOC); ret = pmc_core_ssram_get_pmc(pmcdev, PMC_IDX_MAIN, 0);
pmc_core_ssram_get_pmc(pmcdev, ssram, SSRAM_IOE_OFFSET, PMC_IDX_IOE); if (ret)
pmc_core_ssram_get_pmc(pmcdev, ssram, SSRAM_PCH_OFFSET, PMC_IDX_PCH); goto disable_dev;
iounmap(ssram); pmc_core_ssram_get_pmc(pmcdev, PMC_IDX_IOE, SSRAM_IOE_OFFSET);
pmc_core_ssram_get_pmc(pmcdev, PMC_IDX_PCH, SSRAM_PCH_OFFSET);
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment