2019-12-09 00:40:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
/*
|
|
|
|
* Copyright 2019 Google LLC
|
|
|
|
* Written by Simon Glass <sjg@chromium.org>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
|
|
|
#include <binman.h>
|
2020-05-10 17:40:00 +00:00
|
|
|
#include <bootstage.h>
|
2019-12-09 00:40:19 +00:00
|
|
|
#include <dm.h>
|
2020-05-10 17:40:02 +00:00
|
|
|
#include <init.h>
|
2019-12-09 00:40:19 +00:00
|
|
|
#include <irq.h>
|
2020-05-10 17:40:05 +00:00
|
|
|
#include <log.h>
|
2020-02-03 14:36:16 +00:00
|
|
|
#include <malloc.h>
|
2020-07-08 03:32:31 +00:00
|
|
|
#include <p2sb.h>
|
2020-04-08 22:57:35 +00:00
|
|
|
#include <acpi/acpi_s3.h>
|
2019-12-09 00:40:19 +00:00
|
|
|
#include <asm/intel_pinctrl.h>
|
|
|
|
#include <asm/io.h>
|
|
|
|
#include <asm/intel_regs.h>
|
|
|
|
#include <asm/msr.h>
|
|
|
|
#include <asm/msr-index.h>
|
|
|
|
#include <asm/pci.h>
|
|
|
|
#include <asm/arch/cpu.h>
|
|
|
|
#include <asm/arch/systemagent.h>
|
2020-07-08 03:32:31 +00:00
|
|
|
#include <asm/arch/fsp_bindings.h>
|
2019-12-09 00:40:19 +00:00
|
|
|
#include <asm/arch/fsp/fsp_configs.h>
|
|
|
|
#include <asm/arch/fsp/fsp_s_upd.h>
|
2020-07-08 03:32:31 +00:00
|
|
|
#include <dm/uclass-internal.h>
|
2020-05-10 17:40:13 +00:00
|
|
|
#include <linux/bitops.h>
|
2019-12-09 00:40:19 +00:00
|
|
|
|
|
|
|
#define PCH_P2SB_E0 0xe0
|
|
|
|
#define HIDE_BIT BIT(0)
|
|
|
|
|
|
|
|
int fsps_update_config(struct udevice *dev, ulong rom_offset,
|
|
|
|
struct fsps_upd *upd)
|
|
|
|
{
|
|
|
|
struct fsp_s_config *cfg = &upd->config;
|
2020-05-18 10:33:35 +00:00
|
|
|
ofnode node;
|
2019-12-09 00:40:19 +00:00
|
|
|
|
2020-05-18 10:33:33 +00:00
|
|
|
if (IS_ENABLED(CONFIG_HAVE_VBT)) {
|
2020-07-08 03:32:25 +00:00
|
|
|
void *buf;
|
2020-05-18 10:33:33 +00:00
|
|
|
int ret;
|
|
|
|
|
2020-07-08 03:32:25 +00:00
|
|
|
ret = binman_entry_map(ofnode_null(), "intel-vbt", &buf, NULL);
|
2020-05-18 10:33:33 +00:00
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("Cannot find VBT", ret);
|
2020-07-08 03:32:25 +00:00
|
|
|
if (*(u32 *)buf != VBT_SIGNATURE)
|
|
|
|
return log_msg_ret("VBT signature", -EINVAL);
|
2020-05-18 10:33:33 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Load VBT before devicetree-specific config. This only
|
|
|
|
* supports memory-mapped SPI at present.
|
|
|
|
*/
|
2020-07-08 03:32:25 +00:00
|
|
|
cfg->graphics_config_ptr = (ulong)buf;
|
2020-05-18 10:33:33 +00:00
|
|
|
}
|
2019-12-09 00:40:19 +00:00
|
|
|
|
2020-05-18 10:33:35 +00:00
|
|
|
node = dev_read_subnode(dev, "fsp-s");
|
|
|
|
if (!ofnode_valid(node))
|
|
|
|
return log_msg_ret("fsp-s settings", -ENOENT);
|
2019-12-09 00:40:19 +00:00
|
|
|
|
2020-05-18 10:33:35 +00:00
|
|
|
return fsp_s_update_config_from_dtb(node, cfg);
|
2019-12-09 00:40:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Configure package power limits */
|
|
|
|
static int set_power_limits(struct udevice *dev)
|
|
|
|
{
|
|
|
|
msr_t rapl_msr_reg, limit;
|
|
|
|
u32 power_unit;
|
|
|
|
u32 tdp, min_power, max_power;
|
|
|
|
u32 pl2_val;
|
|
|
|
u32 override_tdp[2];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Get units */
|
|
|
|
rapl_msr_reg = msr_read(MSR_PKG_POWER_SKU_UNIT);
|
|
|
|
power_unit = 1 << (rapl_msr_reg.lo & 0xf);
|
|
|
|
|
|
|
|
/* Get power defaults for this SKU */
|
|
|
|
rapl_msr_reg = msr_read(MSR_PKG_POWER_SKU);
|
|
|
|
tdp = rapl_msr_reg.lo & PKG_POWER_LIMIT_MASK;
|
|
|
|
pl2_val = rapl_msr_reg.hi & PKG_POWER_LIMIT_MASK;
|
|
|
|
min_power = (rapl_msr_reg.lo >> 16) & PKG_POWER_LIMIT_MASK;
|
|
|
|
max_power = rapl_msr_reg.hi & PKG_POWER_LIMIT_MASK;
|
|
|
|
|
|
|
|
if (min_power > 0 && tdp < min_power)
|
|
|
|
tdp = min_power;
|
|
|
|
|
|
|
|
if (max_power > 0 && tdp > max_power)
|
|
|
|
tdp = max_power;
|
|
|
|
|
|
|
|
ret = dev_read_u32_array(dev, "tdp-pl-override-mw", override_tdp,
|
|
|
|
ARRAY_SIZE(override_tdp));
|
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("tdp-pl-override-mw", ret);
|
|
|
|
|
|
|
|
/* Set PL1 override value */
|
|
|
|
if (override_tdp[0])
|
|
|
|
tdp = override_tdp[0] * power_unit / 1000;
|
|
|
|
|
|
|
|
/* Set PL2 override value */
|
|
|
|
if (override_tdp[1])
|
|
|
|
pl2_val = override_tdp[1] * power_unit / 1000;
|
|
|
|
|
|
|
|
/* Set long term power limit to TDP */
|
|
|
|
limit.lo = tdp & PKG_POWER_LIMIT_MASK;
|
|
|
|
/* Set PL1 Pkg Power clamp bit */
|
|
|
|
limit.lo |= PKG_POWER_LIMIT_CLAMP;
|
|
|
|
|
|
|
|
limit.lo |= PKG_POWER_LIMIT_EN;
|
|
|
|
limit.lo |= (MB_POWER_LIMIT1_TIME_DEFAULT &
|
|
|
|
PKG_POWER_LIMIT_TIME_MASK) << PKG_POWER_LIMIT_TIME_SHIFT;
|
|
|
|
|
|
|
|
/* Set short term power limit PL2 */
|
|
|
|
limit.hi = pl2_val & PKG_POWER_LIMIT_MASK;
|
|
|
|
limit.hi |= PKG_POWER_LIMIT_EN;
|
|
|
|
|
|
|
|
/* Program package power limits in RAPL MSR */
|
|
|
|
msr_write(MSR_PKG_POWER_LIMIT, limit);
|
|
|
|
log_info("RAPL PL1 %d.%dW\n", tdp / power_unit,
|
|
|
|
100 * (tdp % power_unit) / power_unit);
|
|
|
|
log_info("RAPL PL2 %d.%dW\n", pl2_val / power_unit,
|
|
|
|
100 * (pl2_val % power_unit) / power_unit);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Sett RAPL MMIO register for Power limits. RAPL driver is using MSR
|
|
|
|
* instead of MMIO, so disable LIMIT_EN bit for MMIO
|
|
|
|
*/
|
|
|
|
writel(limit.lo & ~PKG_POWER_LIMIT_EN, MCHBAR_REG(MCHBAR_RAPL_PPL));
|
|
|
|
writel(limit.hi & ~PKG_POWER_LIMIT_EN, MCHBAR_REG(MCHBAR_RAPL_PPL + 4));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int p2sb_unhide(void)
|
|
|
|
{
|
2020-07-08 03:32:31 +00:00
|
|
|
struct udevice *dev;
|
|
|
|
int ret;
|
2019-12-09 00:40:19 +00:00
|
|
|
|
2020-07-08 03:32:31 +00:00
|
|
|
ret = uclass_find_first_device(UCLASS_P2SB, &dev);
|
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("p2sb", ret);
|
|
|
|
ret = p2sb_set_hide(dev, false);
|
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("hide", ret);
|
2019-12-09 00:40:19 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Overwrites the SCI IRQ if another IRQ number is given by device tree */
|
|
|
|
static void set_sci_irq(void)
|
|
|
|
{
|
|
|
|
/* Skip this for now */
|
|
|
|
}
|
|
|
|
|
|
|
|
int arch_fsps_preinit(void)
|
|
|
|
{
|
|
|
|
struct udevice *itss;
|
|
|
|
int ret;
|
|
|
|
|
2020-02-06 16:54:58 +00:00
|
|
|
ret = irq_first_device_type(X86_IRQT_ITSS, &itss);
|
2019-12-09 00:40:19 +00:00
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("no itss", ret);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Clear the GPI interrupt status and enable registers. These
|
|
|
|
* registers do not get reset to default state when booting from S5.
|
|
|
|
*/
|
|
|
|
ret = pinctrl_gpi_clear_int_cfg();
|
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("gpi_clear", ret);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int arch_fsp_init_r(void)
|
|
|
|
{
|
2020-07-10 00:43:16 +00:00
|
|
|
bool s3wake;
|
2019-12-09 00:40:19 +00:00
|
|
|
struct udevice *dev, *itss;
|
|
|
|
int ret;
|
|
|
|
|
2020-04-26 15:12:54 +00:00
|
|
|
if (!ll_boot_init())
|
|
|
|
return 0;
|
2020-07-10 00:43:16 +00:00
|
|
|
|
|
|
|
s3wake = IS_ENABLED(CONFIG_HAVE_ACPI_RESUME) &&
|
|
|
|
gd->arch.prev_sleep_state == ACPI_S3;
|
|
|
|
|
2019-12-09 00:40:19 +00:00
|
|
|
/*
|
|
|
|
* This must be called before any devices are probed. Put any probing
|
|
|
|
* into arch_fsps_preinit() above.
|
|
|
|
*
|
|
|
|
* We don't use CONFIG_APL_BOOT_FROM_FAST_SPI_FLASH here since it will
|
|
|
|
* force PCI to be probed.
|
|
|
|
*/
|
|
|
|
ret = fsp_silicon_init(s3wake, false);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2020-02-06 16:54:58 +00:00
|
|
|
ret = irq_first_device_type(X86_IRQT_ITSS, &itss);
|
2019-12-09 00:40:19 +00:00
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("no itss", ret);
|
2020-07-17 03:22:30 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Restore GPIO IRQ polarities back to previous settings. This was
|
|
|
|
* stored in reserve_arch() - see X86_IRQT_ITSS
|
|
|
|
*/
|
2019-12-09 00:40:19 +00:00
|
|
|
irq_restore_polarities(itss);
|
|
|
|
|
|
|
|
/* soc_init() */
|
|
|
|
ret = p2sb_unhide();
|
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("unhide p2sb", ret);
|
|
|
|
|
|
|
|
/* Set RAPL MSR for Package power limits*/
|
|
|
|
ret = uclass_first_device_err(UCLASS_NORTHBRIDGE, &dev);
|
|
|
|
if (ret)
|
|
|
|
return log_msg_ret("Cannot get northbridge", ret);
|
|
|
|
set_power_limits(dev);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* FSP-S routes SCI to IRQ 9. With the help of this function you can
|
|
|
|
* select another IRQ for SCI.
|
|
|
|
*/
|
|
|
|
set_sci_irq();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|