1 /* SPDX-License-Identifier: GPL-2.0-or-later */
3 #define __SIMPLE_DEVICE__
6 #include <acpi/acpi_pm.h>
8 #include <device/mmio.h>
9 #include <console/console.h>
10 #include <device/device.h>
11 #include <device/pci.h>
12 #include <intelblocks/msr.h>
13 #include <intelblocks/pmclib.h>
14 #include <intelblocks/rtc.h>
15 #include <intelblocks/tco.h>
16 #include <soc/iomap.h>
18 #include <soc/pci_devs.h>
20 #include <soc/smbus.h>
21 #include <security/vboot/vbnv.h>
26 uint8_t *pmc_mmio_regs(void)
28 return (void *)(uintptr_t)PCH_PWRM_BASE_ADDRESS
;
31 uintptr_t soc_read_pmc_base(void)
33 return (uintptr_t)pmc_mmio_regs();
36 uint32_t *soc_pmc_etr_addr(void)
38 return (uint32_t *)(soc_read_pmc_base() + ETR
);
41 const char *const *soc_smi_sts_array(size_t *a
)
43 static const char *const smi_sts_bits
[] = {
44 [BIOS_STS_BIT
] = "BIOS",
45 [LEGACY_USB_STS_BIT
] = "LEGACY USB",
46 [SMI_ON_SLP_EN_STS_BIT
] = "SLP_SMI",
47 [APM_STS_BIT
] = "APM",
48 [SWSMI_TMR_STS_BIT
] = "SWSMI_TMR",
49 [PM1_STS_BIT
] = "PM1",
50 [GPE0_STS_BIT
] = "GPE0 (reserved)",
51 [GPIO_STS_BIT
] = "GPIO_SMI",
52 [GPIO_UNLOCK_SMI_STS_BIT
] = "GPIO_UNLOCK_SSMI",
53 [MC_SMI_STS_BIT
] = "MCSMI",
54 [TCO_STS_BIT
] = "TCO",
55 [PERIODIC_STS_BIT
] = "PERIODIC",
56 [SERIRQ_SMI_STS_BIT
] = "SERIRQ",
57 [SMBUS_SMI_STS_BIT
] = "SMBUS_SMI",
58 [XHCI_SMI_STS_BIT
] = "XHCI",
59 [HSMBUS_SMI_STS_BIT
] = "HOST_SMBUS",
60 [SCS_SMI_STS_BIT
] = "SCS",
61 [PCI_EXP_SMI_STS_BIT
] = "PCI_EXP_SMI",
62 [SCC2_SMI_STS_BIT
] = "SCC2",
63 [SPI_SSMI_STS_BIT
] = "SPI_SSMI",
64 [SPI_SMI_STS_BIT
] = "SPI",
65 [PMC_OCP_SMI_STS_BIT
] = "OCP_CSE",
68 *a
= ARRAY_SIZE(smi_sts_bits
);
73 * For APL/GLK this check for power button status if nothing else
74 * is indicating an SMI and SMIs aren't turned into SCIs.
75 * Apparently, there is no PM1 status bit in the SMI status
76 * register. That makes things difficult for
77 * determining if the power button caused an SMI.
79 uint32_t soc_get_smi_status(uint32_t generic_sts
)
81 if (generic_sts
== 0 && !(pmc_read_pm1_control() & SCI_EN
)) {
82 uint16_t pm1_sts
= inw(ACPI_BASE_ADDRESS
+ PM1_STS
);
84 /* Fake PM1 status bit if power button pressed. */
85 if (pm1_sts
& PWRBTN_STS
)
86 generic_sts
|= (1 << PM1_STS_BIT
);
90 * GPE0_STS is reserved in APL/GLK datasheets. For compatibility
91 * with common code, mask it out so that it is always zero.
93 return generic_sts
& ~(1 << GPE0_STS_BIT
);
96 const char *const *soc_tco_sts_array(size_t *a
)
98 static const char *const tco_sts_bits
[] = {
103 *a
= ARRAY_SIZE(tco_sts_bits
);
107 const char *const *soc_std_gpe_sts_array(size_t *a
)
109 static const char *const gpe_sts_bits
[] = {
123 [15] = "GPIO_TIER1_SCI",
128 *a
= ARRAY_SIZE(gpe_sts_bits
);
132 void soc_clear_pm_registers(uintptr_t pmc_bar
)
136 gen_pmcon1
= read32p(pmc_bar
+ GEN_PMCON1
);
137 /* Clear the status bits. The RPS field is cleared on a 0 write. */
138 write32p(pmc_bar
+ GEN_PMCON1
, gen_pmcon1
& ~RPS
);
141 static void gpe0_different_values(const struct soc_intel_apollolake_config
*config
)
143 bool result
= (config
->gpe0_dw1
!= config
->gpe0_dw2
) &&
144 (config
->gpe0_dw1
!= config
->gpe0_dw3
) &&
145 (config
->gpe0_dw2
!= config
->gpe0_dw3
);
150 void soc_get_gpi_gpe_configs(uint8_t *dw0
, uint8_t *dw1
, uint8_t *dw2
)
152 DEVTREE_CONST
struct soc_intel_apollolake_config
*config
;
154 config
= config_of_soc();
156 gpe0_different_values(config
);
158 /* Assign to out variable */
159 *dw0
= config
->gpe0_dw1
;
160 *dw1
= config
->gpe0_dw2
;
161 *dw2
= config
->gpe0_dw3
;
164 void soc_fill_power_state(struct chipset_power_state
*ps
)
166 uintptr_t pmc_bar0
= soc_read_pmc_base();
168 ps
->tco1_sts
= tco_read_reg(TCO1_STS
);
169 ps
->tco2_sts
= tco_read_reg(TCO2_STS
);
171 ps
->prsts
= read32p(pmc_bar0
+ PRSTS
);
172 ps
->gen_pmcon1
= read32p(pmc_bar0
+ GEN_PMCON1
);
173 ps
->gen_pmcon2
= read32p(pmc_bar0
+ GEN_PMCON2
);
174 ps
->gen_pmcon3
= read32p(pmc_bar0
+ GEN_PMCON3
);
176 printk(BIOS_DEBUG
, "prsts: %08x\n",
178 printk(BIOS_DEBUG
, "tco_sts: %04x %04x\n",
179 ps
->tco1_sts
, ps
->tco2_sts
);
181 "gen_pmcon1: %08x gen_pmcon2: %08x gen_pmcon3: %08x\n",
182 ps
->gen_pmcon1
, ps
->gen_pmcon2
, ps
->gen_pmcon3
);
185 /* Return 0, 3, or 5 to indicate the previous sleep state. */
186 int soc_prev_sleep_state(const struct chipset_power_state
*ps
,
187 int prev_sleep_state
)
189 /* WAK_STS bit will not be set when waking from G3 state */
191 if (!(ps
->pm1_sts
& WAK_STS
) && (ps
->gen_pmcon1
& COLD_BOOT_STS
))
192 prev_sleep_state
= ACPI_S5
;
193 return prev_sleep_state
;
196 static int rtc_failed(uint32_t gen_pmcon1
)
198 return !!(gen_pmcon1
& RPS
);
201 int soc_get_rtc_failed(void)
203 const struct chipset_power_state
*ps
;
205 if (acpi_fetch_pm_state(&ps
, PS_CLAIMER_RTC
) < 0)
208 return rtc_failed(ps
->gen_pmcon1
);
211 int vbnv_cmos_failed(void)
213 uintptr_t pmc_bar
= soc_read_pmc_base();
214 uint32_t gen_pmcon1
= read32p(pmc_bar
+ GEN_PMCON1
);
215 int rtc_failure
= rtc_failed(gen_pmcon1
);
218 printk(BIOS_INFO
, "RTC failed!\n");
220 /* We do not want to write 1 to clear-1 bits. Set them to 0. */
221 gen_pmcon1
&= ~GEN_PMCON1_CLR1_BITS
;
223 /* RPS is write 0 to clear. */
226 write32p(pmc_bar
+ GEN_PMCON1
, gen_pmcon1
);
233 uint16_t get_pmbase(void)
235 return (uint16_t)ACPI_BASE_ADDRESS
;
238 /* Set which power state system will be after reapplying the power (from G3 State) */
239 void pmc_soc_set_afterg3_en(const bool on
)
242 uint8_t *const pmcbase
= pmc_mmio_regs();
244 reg8
= read8(pmcbase
+ GEN_PMCON_A
);
246 reg8
&= ~SLEEP_AFTER_POWER_FAIL
;
248 reg8
|= SLEEP_AFTER_POWER_FAIL
;
249 write8(pmcbase
+ GEN_PMCON_A
, reg8
);