/* * This file is part of the coreboot project. * * Copyright 2015 Google Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2 of the License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * */ #include #include #include #include #include #include #include /* Default values for domain configuration. PSI3 and PSI4 are disabled. */ static const struct vr_config default_configs[NUM_VR_DOMAINS] = { [VR_SYSTEM_AGENT] = { .vr_config_enable = 1, .psi1threshold = VR_CFG_AMP(20), .psi2threshold = VR_CFG_AMP(4), .psi3threshold = VR_CFG_AMP(1), .psi3enable = 0, .psi4enable = 0, .imon_slope = 0x0, .imon_offset = 0x0, .icc_max = VR_CFG_AMP(7), .voltage_limit = 1520, }, [VR_IA_CORE] = { .vr_config_enable = 1, .psi1threshold = VR_CFG_AMP(20), .psi2threshold = VR_CFG_AMP(5), .psi3threshold = VR_CFG_AMP(1), .psi3enable = 0, .psi4enable = 0, .imon_slope = 0x0, .imon_offset = 0x0, .icc_max = VR_CFG_AMP(34), .voltage_limit = 1520, }, #if CONFIG(PLATFORM_USES_FSP1_1) [VR_RING] = { .vr_config_enable = 1, .psi1threshold = VR_CFG_AMP(20), .psi2threshold = VR_CFG_AMP(5), .psi3threshold = VR_CFG_AMP(1), .psi3enable = 0, .psi4enable = 0, .imon_slope = 0x0, .imon_offset = 0x0, .icc_max = VR_CFG_AMP(34), .voltage_limit = 1520, }, #endif [VR_GT_UNSLICED] = { .vr_config_enable = 1, .psi1threshold = VR_CFG_AMP(20), .psi2threshold = VR_CFG_AMP(5), .psi3threshold = VR_CFG_AMP(1), .psi3enable = 0, .psi4enable = 0, .imon_slope = 0x0, .imon_offset = 0x0, .icc_max = VR_CFG_AMP(35), .voltage_limit = 1520, }, [VR_GT_SLICED] = { .vr_config_enable = 1, .psi1threshold = VR_CFG_AMP(20), .psi2threshold = VR_CFG_AMP(5), .psi3threshold = VR_CFG_AMP(1), .psi3enable = 0, .psi4enable = 0, .imon_slope = 0x0, .imon_offset = 0x0, .icc_max = VR_CFG_AMP(35), .voltage_limit = 1520, }, }; static uint16_t get_sku_icc_max(int domain) { const uint16_t tdp = cpu_get_power_max(); static uint16_t mch_id = 0, igd_id = 0, lpc_id = 0; if (!mch_id) { struct device *dev = pcidev_path_on_root(SA_DEVFN_ROOT); mch_id = pci_read_config16(dev, PCI_DEVICE_ID); } if (!igd_id) { struct device *dev = pcidev_path_on_root(SA_DEVFN_IGD); if (dev) igd_id = pci_read_config16(dev, PCI_DEVICE_ID); else igd_id = 0xffff; } if (!lpc_id) { struct device *dev = pcidev_path_on_root(PCH_DEVFN_LPC); lpc_id = pci_read_config16(dev, PCI_DEVICE_ID); } /* * Iccmax table from Doc #559100 Section 7.2 DC Specifications, the * Iccmax is the same among KBL-Y but KBL-U/R. * Addendum for AML-Y #594883, IccMax for IA core is 28A. * KBL-S #335195, KBL-H #335190 * +----------------+-------------+---------------+------+-----+ * | Domain/Setting | SA | IA | GTUS | GTS | * +----------------+-------------+---------------+------+-----+ * | IccMax(KBL-S) | 11.1A | 100A | 45A | 45A | * | | | ... | | | * | | | 40A | | | * +----------------+-------------+---------------+------+-----+ * | IccMax(KBL-H) | 11.1A(45W) | 68A | 55A | 55A | * | | 6.6A(Others)| 60A | | | * +----------------+-------------+---------------+------+-----+ * | IccMax(KBL-U/R)| 6A(U42) | 64A(U42) | 31A | 31A | * | | 4.5A(Others)| 29A(P/C) | | | * | | | 32A(i3/i5) | | | * +----------------+-------------+---------------+------+-----+ * | IccMax(KBL-Y) | 4.1A | 24A | 24A | 24A | * +----------------+-------------+---------------+------+-----+ * | IccMax(AML-Y) | 4.1A | 28A | 24A | 24A | * +----------------+-------------+---------------+------+-----+ */ switch (mch_id) { case PCI_DEVICE_ID_INTEL_KBL_ID_S: { uint16_t icc_max[NUM_VR_DOMAINS] = { VR_CFG_AMP(11.1), VR_CFG_AMP(40), VR_CFG_AMP(45), VR_CFG_AMP(45), }; if (tdp >= 54) icc_max[VR_IA_CORE] = VR_CFG_AMP(58); else if (tdp >= 51) icc_max[VR_IA_CORE] = VR_CFG_AMP(45); return icc_max[domain]; } case PCI_DEVICE_ID_INTEL_KBL_ID_DT_2: /* fallthrough */ case PCI_DEVICE_ID_INTEL_KBL_ID_DT: { uint16_t icc_max[NUM_VR_DOMAINS] = { VR_CFG_AMP(11.1), VR_CFG_AMP(66), VR_CFG_AMP(55), VR_CFG_AMP(55), }; if (tdp >= 91) icc_max[VR_IA_CORE] = VR_CFG_AMP(100); else if (tdp >= 65) icc_max[VR_IA_CORE] = VR_CFG_AMP(79); return icc_max[domain]; } case PCI_DEVICE_ID_INTEL_KBL_ID_H: { uint16_t icc_max[NUM_VR_DOMAINS] = { VR_CFG_AMP(6.6), VR_CFG_AMP(60), VR_CFG_AMP(55), VR_CFG_AMP(55), }; if (tdp >= 45) { icc_max[VR_SYSTEM_AGENT] = VR_CFG_AMP(11.1); icc_max[VR_IA_CORE] = VR_CFG_AMP(68); } return icc_max[domain]; } case PCI_DEVICE_ID_INTEL_KBL_U_R: { static const uint16_t icc_max[NUM_VR_DOMAINS] = { VR_CFG_AMP(6), VR_CFG_AMP(64), VR_CFG_AMP(31), VR_CFG_AMP(31), }; return icc_max[domain]; } case PCI_DEVICE_ID_INTEL_KBL_ID_Y: { uint16_t icc_max[NUM_VR_DOMAINS] = { VR_CFG_AMP(4.1), VR_CFG_AMP(24), VR_CFG_AMP(24), VR_CFG_AMP(24), }; if (igd_id == PCI_DEVICE_ID_INTEL_AML_GT2_ULX) icc_max[VR_IA_CORE] = VR_CFG_AMP(28); return icc_max[domain]; } case PCI_DEVICE_ID_INTEL_KBL_ID_U: { uint16_t icc_max[NUM_VR_DOMAINS] = { VR_CFG_AMP(4.5), VR_CFG_AMP(32), VR_CFG_AMP(31), VR_CFG_AMP(31), }; if (igd_id == PCI_DEVICE_ID_INTEL_SPT_LP_U_BASE_HDCP22) icc_max[VR_IA_CORE] = VR_CFG_AMP(29); return icc_max[domain]; } default: printk(BIOS_ERR, "ERROR: Unknown MCH in VR-config\n"); } return 0; } #if CONFIG(PLATFORM_USES_FSP2_0) static uint16_t get_sku_ac_dc_loadline(const int domain) { static uint16_t mch_id = 0, igd_id = 0; if (!mch_id) { struct device *dev = pcidev_path_on_root(SA_DEVFN_ROOT); mch_id = pci_read_config16(dev, PCI_DEVICE_ID); } if (!igd_id) { struct device *dev = pcidev_path_on_root(SA_DEVFN_IGD); if (dev) igd_id = pci_read_config16(dev, PCI_DEVICE_ID); else igd_id = 0xffff; } switch (mch_id) { case PCI_DEVICE_ID_INTEL_KBL_ID_S: /* fallthrough */ case PCI_DEVICE_ID_INTEL_KBL_ID_DT: /* fallthrough */ case PCI_DEVICE_ID_INTEL_KBL_ID_DT_2: { static const uint16_t loadline[NUM_VR_DOMAINS] = { VR_CFG_MOHMS(0), /* Not specified */ VR_CFG_MOHMS(2.1), VR_CFG_MOHMS(3.1), VR_CFG_MOHMS(3.1), }; return loadline[domain]; } case PCI_DEVICE_ID_INTEL_KBL_ID_H: { static const uint16_t loadline[NUM_VR_DOMAINS] = { VR_CFG_MOHMS(10), VR_CFG_MOHMS(1.8), VR_CFG_MOHMS(2.65), VR_CFG_MOHMS(2.65), }; return loadline[domain]; } case PCI_DEVICE_ID_INTEL_KBL_ID_Y: { uint16_t loadline[NUM_VR_DOMAINS] = { VR_CFG_MOHMS(18), VR_CFG_MOHMS(5.9), VR_CFG_MOHMS(5.7), VR_CFG_MOHMS(5.7), }; if (igd_id == PCI_DEVICE_ID_INTEL_AML_GT2_ULX) loadline[VR_IA_CORE] = VR_CFG_MOHMS(4); return loadline[domain]; } case PCI_DEVICE_ID_INTEL_KBL_U_R: /* fallthrough */ case PCI_DEVICE_ID_INTEL_KBL_ID_U: { uint16_t loadline[NUM_VR_DOMAINS] = { VR_CFG_MOHMS(10.3), VR_CFG_MOHMS(2.4), VR_CFG_MOHMS(3.1), VR_CFG_MOHMS(3.1), }; if (igd_id == PCI_DEVICE_ID_INTEL_SPT_LP_U_PREMIUM_HDCP22) { loadline[VR_GT_UNSLICED] = VR_CFG_MOHMS(6); loadline[VR_GT_SLICED] = VR_CFG_MOHMS(6); } return loadline[domain]; } default: printk(BIOS_ERR, "ERROR: Unknown MCH in VR-config\n"); } return 0; } #endif void fill_vr_domain_config(void *params, int domain, const struct vr_config *chip_cfg) { FSP_SIL_UPD *vr_params = (FSP_SIL_UPD *)params; const struct vr_config *cfg; if (domain < 0 || domain >= NUM_VR_DOMAINS) return; /* Use device tree override if requested. */ if (chip_cfg->vr_config_enable) cfg = chip_cfg; else cfg = &default_configs[domain]; vr_params->VrConfigEnable[domain] = cfg->vr_config_enable; vr_params->Psi1Threshold[domain] = cfg->psi1threshold; vr_params->Psi2Threshold[domain] = cfg->psi2threshold; vr_params->Psi3Threshold[domain] = cfg->psi3threshold; vr_params->Psi3Enable[domain] = cfg->psi3enable; vr_params->Psi4Enable[domain] = cfg->psi4enable; vr_params->ImonSlope[domain] = cfg->imon_slope; vr_params->ImonOffset[domain] = cfg->imon_offset; /* If board provided non-zero value, use it. */ if (cfg->icc_max) vr_params->IccMax[domain] = cfg->icc_max; else vr_params->IccMax[domain] = get_sku_icc_max(domain); vr_params->VrVoltageLimit[domain] = cfg->voltage_limit; #if CONFIG(PLATFORM_USES_FSP2_0) if (cfg->ac_loadline) vr_params->AcLoadline[domain] = cfg->ac_loadline; else vr_params->AcLoadline[domain] = get_sku_ac_dc_loadline(domain); if (cfg->dc_loadline) vr_params->DcLoadline[domain] = cfg->dc_loadline; else vr_params->DcLoadline[domain] = get_sku_ac_dc_loadline(domain); #endif }