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: David E. Box <david.e.box@linux.intel.com> Reviewed-by: Ilpo Järvinen <ilpo.jarvinen@linux.intel.com> Link: https://lore.kernel.org/r/20231129222132.2331261-14-david.e.box@linux.intel.com Signed-off-by: Hans de Goede <hdegoede@redhat.com>
This commit is contained in:
parent
9512920a6b
commit
a01486dc4b
1 changed files with 33 additions and 29 deletions
|
@ -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 @@
|
||||||
#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,
|
||||||
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) {
|
map = pmc_core_find_regmap(pmcdev->regmap_list, devid);
|
||||||
const struct pmc_reg_map *map;
|
if (!map)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
map = pmc_core_find_regmap(pmcdev->regmap_list, devid);
|
return pmc_core_pmc_add(pmcdev, pwrm_base, map, PMC_IDX_MAIN);
|
||||||
if (map)
|
|
||||||
pmc_core_pmc_add(pmcdev, pwrm_base, map, pmc_idx);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pmc_idx != PMC_IDX_SOC)
|
|
||||||
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)
|
||||||
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;
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue