mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-11-12 16:07:30 +00:00
8d0defa44b
The dm.h header should come first. In fact it needs to, since otherwise the driver model definitions are not available to dt-structs.h Fix this, since it causes problems with OF_PLATDATA_INST. Signed-off-by: Simon Glass <sjg@chromium.org> Signed-off-by: Simon Glass <sjg@chromium.org>
224 lines
6 KiB
C
224 lines
6 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (C) 2017 Intel Corporation.
|
|
* Copyright 2019 Google LLC
|
|
*
|
|
* Modified from coreboot pmclib.c, pmc.c and pmutil.c
|
|
*/
|
|
|
|
#define LOG_CATEGORY UCLASS_ACPI_PMC
|
|
|
|
#include <common.h>
|
|
#include <dm.h>
|
|
#include <dt-structs.h>
|
|
#include <log.h>
|
|
#include <spl.h>
|
|
#include <acpi/acpi_s3.h>
|
|
#include <asm/io.h>
|
|
#include <asm/pci.h>
|
|
#include <asm/arch/pmc.h>
|
|
#include <linux/bitops.h>
|
|
#include <power/acpi_pmc.h>
|
|
|
|
#define GPIO_GPE_CFG 0x1050
|
|
|
|
/* Memory mapped IO registers behind PMC_BASE_ADDRESS */
|
|
#define PRSTS 0x1000
|
|
#define GEN_PMCON1 0x1020
|
|
#define COLD_BOOT_STS BIT(27)
|
|
#define COLD_RESET_STS BIT(26)
|
|
#define WARM_RESET_STS BIT(25)
|
|
#define GLOBAL_RESET_STS BIT(24)
|
|
#define SRS BIT(20)
|
|
#define MS4V BIT(18)
|
|
#define RPS BIT(2)
|
|
#define GEN_PMCON1_CLR1_BITS (COLD_BOOT_STS | COLD_RESET_STS | \
|
|
WARM_RESET_STS | GLOBAL_RESET_STS | \
|
|
SRS | MS4V)
|
|
#define GEN_PMCON2 0x1024
|
|
#define GEN_PMCON3 0x1028
|
|
|
|
/* Offset of TCO registers from ACPI base I/O address */
|
|
#define TCO_REG_OFFSET 0x60
|
|
#define TCO1_STS 0x64
|
|
#define DMISCI_STS BIT(9)
|
|
#define BOOT_STS BIT(18)
|
|
#define TCO2_STS 0x66
|
|
#define TCO1_CNT 0x68
|
|
#define TCO_LOCK BIT(12)
|
|
#define TCO2_CNT 0x6a
|
|
|
|
enum {
|
|
ETR = 0x1048,
|
|
CF9_LOCK = 1UL << 31,
|
|
CF9_GLB_RST = 1 << 20,
|
|
};
|
|
|
|
static int apl_pmc_fill_power_state(struct udevice *dev)
|
|
{
|
|
struct acpi_pmc_upriv *upriv = dev_get_uclass_priv(dev);
|
|
|
|
upriv->tco1_sts = inw(upriv->acpi_base + TCO1_STS);
|
|
upriv->tco2_sts = inw(upriv->acpi_base + TCO2_STS);
|
|
|
|
upriv->prsts = readl(upriv->pmc_bar0 + PRSTS);
|
|
upriv->gen_pmcon1 = readl(upriv->pmc_bar0 + GEN_PMCON1);
|
|
upriv->gen_pmcon2 = readl(upriv->pmc_bar0 + GEN_PMCON2);
|
|
upriv->gen_pmcon3 = readl(upriv->pmc_bar0 + GEN_PMCON3);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int apl_prev_sleep_state(struct udevice *dev, int prev_sleep_state)
|
|
{
|
|
struct acpi_pmc_upriv *upriv = dev_get_uclass_priv(dev);
|
|
|
|
/* WAK_STS bit will not be set when waking from G3 state */
|
|
if (!(upriv->pm1_sts & WAK_STS) &&
|
|
(upriv->gen_pmcon1 & COLD_BOOT_STS))
|
|
prev_sleep_state = ACPI_S5;
|
|
|
|
return prev_sleep_state;
|
|
}
|
|
|
|
static int apl_disable_tco(struct udevice *dev)
|
|
{
|
|
struct acpi_pmc_upriv *upriv = dev_get_uclass_priv(dev);
|
|
|
|
pmc_disable_tco_base(upriv->acpi_base + TCO_REG_OFFSET);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int apl_global_reset_set_enable(struct udevice *dev, bool enable)
|
|
{
|
|
struct acpi_pmc_upriv *upriv = dev_get_uclass_priv(dev);
|
|
|
|
if (enable)
|
|
setbits_le32(upriv->pmc_bar0 + ETR, CF9_GLB_RST);
|
|
else
|
|
clrbits_le32(upriv->pmc_bar0 + ETR, CF9_GLB_RST);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int apl_pmc_ofdata_to_uc_plat(struct udevice *dev)
|
|
{
|
|
struct acpi_pmc_upriv *upriv = dev_get_uclass_priv(dev);
|
|
struct apl_pmc_plat *plat = dev_get_plat(dev);
|
|
|
|
#if !CONFIG_IS_ENABLED(OF_PLATDATA)
|
|
u32 base[6];
|
|
int size;
|
|
int ret;
|
|
|
|
ret = dev_read_u32_array(dev, "early-regs", base,
|
|
ARRAY_SIZE(base));
|
|
if (ret)
|
|
return log_msg_ret("Missing/short early-regs", ret);
|
|
if (spl_phase() == PHASE_TPL) {
|
|
upriv->pmc_bar0 = (void *)base[0];
|
|
upriv->pmc_bar2 = (void *)base[2];
|
|
|
|
/* Since PCI is not enabled, we must get the BDF manually */
|
|
plat->bdf = pci_get_devfn(dev);
|
|
if (plat->bdf < 0)
|
|
return log_msg_ret("Cannot get PMC PCI address",
|
|
plat->bdf);
|
|
}
|
|
upriv->acpi_base = base[4];
|
|
|
|
/* Get the dwX values for pmc gpe settings */
|
|
size = dev_read_size(dev, "gpe0-dw");
|
|
if (size < 0)
|
|
return log_msg_ret("Cannot read gpe0-dm", size);
|
|
upriv->gpe0_count = size / sizeof(u32);
|
|
ret = dev_read_u32_array(dev, "gpe0-dw", upriv->gpe0_dw,
|
|
upriv->gpe0_count);
|
|
if (ret)
|
|
return log_msg_ret("Bad gpe0-dw", ret);
|
|
|
|
return pmc_ofdata_to_uc_plat(dev);
|
|
#else
|
|
struct dtd_intel_apl_pmc *dtplat = &plat->dtplat;
|
|
|
|
plat->bdf = pci_ofplat_get_devfn(dtplat->reg[0]);
|
|
upriv->pmc_bar0 = (void *)dtplat->early_regs[0];
|
|
upriv->pmc_bar2 = (void *)dtplat->early_regs[2];
|
|
upriv->acpi_base = dtplat->early_regs[4];
|
|
upriv->gpe0_dwx_mask = dtplat->gpe0_dwx_mask;
|
|
upriv->gpe0_dwx_shift_base = dtplat->gpe0_dwx_shift_base;
|
|
upriv->gpe0_sts_reg = dtplat->gpe0_sts;
|
|
upriv->gpe0_sts_reg += upriv->acpi_base;
|
|
upriv->gpe0_en_reg = dtplat->gpe0_en;
|
|
upriv->gpe0_en_reg += upriv->acpi_base;
|
|
upriv->gpe0_count = min((int)ARRAY_SIZE(dtplat->gpe0_dw), GPE0_REG_MAX);
|
|
memcpy(upriv->gpe0_dw, dtplat->gpe0_dw, sizeof(dtplat->gpe0_dw));
|
|
#endif
|
|
upriv->gpe_cfg = (u32 *)(upriv->pmc_bar0 + GPIO_GPE_CFG);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int enable_pmcbar(struct udevice *dev)
|
|
{
|
|
struct acpi_pmc_upriv *upriv = dev_get_uclass_priv(dev);
|
|
struct apl_pmc_plat *priv = dev_get_plat(dev);
|
|
pci_dev_t pmc = priv->bdf;
|
|
|
|
/*
|
|
* Set PMC base addresses and enable decoding. BARs 1 and 3 are 64-bit
|
|
* BARs.
|
|
*/
|
|
pci_x86_write_config(pmc, PCI_BASE_ADDRESS_0, (ulong)upriv->pmc_bar0,
|
|
PCI_SIZE_32);
|
|
pci_x86_write_config(pmc, PCI_BASE_ADDRESS_1, 0, PCI_SIZE_32);
|
|
pci_x86_write_config(pmc, PCI_BASE_ADDRESS_2, (ulong)upriv->pmc_bar2,
|
|
PCI_SIZE_32);
|
|
pci_x86_write_config(pmc, PCI_BASE_ADDRESS_3, 0, PCI_SIZE_32);
|
|
pci_x86_write_config(pmc, PCI_BASE_ADDRESS_4, upriv->acpi_base,
|
|
PCI_SIZE_16);
|
|
pci_x86_write_config(pmc, PCI_COMMAND, PCI_COMMAND_IO |
|
|
PCI_COMMAND_MEMORY | PCI_COMMAND_MASTER,
|
|
PCI_SIZE_16);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int apl_pmc_probe(struct udevice *dev)
|
|
{
|
|
if (spl_phase() == PHASE_TPL) {
|
|
return enable_pmcbar(dev);
|
|
} else {
|
|
struct acpi_pmc_upriv *upriv = dev_get_uclass_priv(dev);
|
|
|
|
upriv->pmc_bar0 = (void *)dm_pci_read_bar32(dev, 0);
|
|
upriv->pmc_bar2 = (void *)dm_pci_read_bar32(dev, 2);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct acpi_pmc_ops apl_pmc_ops = {
|
|
.init = apl_pmc_fill_power_state,
|
|
.prev_sleep_state = apl_prev_sleep_state,
|
|
.disable_tco = apl_disable_tco,
|
|
.global_reset_set_enable = apl_global_reset_set_enable,
|
|
};
|
|
|
|
#if !CONFIG_IS_ENABLED(OF_PLATDATA)
|
|
static const struct udevice_id apl_pmc_ids[] = {
|
|
{ .compatible = "intel,apl-pmc" },
|
|
{ }
|
|
};
|
|
#endif
|
|
|
|
U_BOOT_DRIVER(intel_apl_pmc) = {
|
|
.name = "intel_apl_pmc",
|
|
.id = UCLASS_ACPI_PMC,
|
|
.of_match = of_match_ptr(apl_pmc_ids),
|
|
.of_to_plat = apl_pmc_ofdata_to_uc_plat,
|
|
.probe = apl_pmc_probe,
|
|
.ops = &apl_pmc_ops,
|
|
.plat_auto = sizeof(struct apl_pmc_plat),
|
|
};
|