blob: 1964c12193751910156f67a19b6ee087fe3ab3fe [file] [log] [blame]
V Sowmyac6d71662021-07-15 08:11:08 +05301/* SPDX-License-Identifier: GPL-2.0-only */
2
3#include <device/pci_ids.h>
4#include <device/pci_ops.h>
5#include <fsp/api.h>
6#include <soc/ramstage.h>
7#include <soc/vr_config.h>
8#include <console/console.h>
9#include <intelblocks/cpulib.h>
10
11/*
12 * VR Configurations for IA and GT domains for ADL-P SKU's.
Curtis Chen0c544612021-11-19 11:38:12 +080013 * Per doc#626774 ADL_MOW_WW46_2021, update PD optimization relaxation
14 * for ADL-P 482(28W) and 442(45W).
V Sowmyac6d71662021-07-15 08:11:08 +053015 *
16 * +----------------+-----------+-------+-------+---------+-------------+----------+
17 * | SKU | Setting | AC LL | DC LL | ICC MAX | TDC Current | TDC Time |
18 * | | |(mOhms)|(mOhms)| (A) | (A) | (msec) |
19 * +----------------+-----------+-------+-------+---------+-------------+----------+
20 * | ADL-P 682(45W) | IA | 2.3 | 2.3 | 160 | 57 | 28000 |
21 * + +-----------+-------+-------+---------+-------------+----------+
22 * | | GT | 3.2 | 3.2 | 50 | 57 | 28000 |
23 * +----------------+-----------+-------+-------+---------+-------------+----------+
Curtis Chen0c544612021-11-19 11:38:12 +080024 * | ADL-P 482(28W) | IA | 2.3 | 2.3 | 85 | 40 | 28000 |
V Sowmyac6d71662021-07-15 08:11:08 +053025 * + +-----------+-------+-------+---------+-------------+----------+
26 * | | GT | 3.2 | 3.2 | 50 | 40 | 28000 |
27 * +----------------+-----------+-------+-------+---------+-------------+----------+
28 * | ADL-P 282(15W) | IA | 2.8 | 2.8 | 80 | 20 | 28000 |
29 * + +-----------+-------+-------+---------+-------------+----------+
30 * | | GT | 3.2 | 3.2 | 40 | 20 | 28000 |
31 * +----------------+-----------+-------+-------+---------+-------------+----------+
32 */
33
34struct vr_lookup {
35 uint16_t mchid;
Curtis Chenea1bb5f2021-11-25 13:17:42 +080036 uint8_t tdp;
V Sowmyac6d71662021-07-15 08:11:08 +053037 uint32_t conf[NUM_VR_DOMAINS];
38};
39
40static uint32_t load_table(const struct vr_lookup *tbl, const int tbl_entries, const int domain,
Curtis Chenea1bb5f2021-11-25 13:17:42 +080041 const uint16_t mch_id, uint8_t tdp)
V Sowmyac6d71662021-07-15 08:11:08 +053042{
43 for (size_t i = 0; i < tbl_entries; i++) {
Curtis Chenea1bb5f2021-11-25 13:17:42 +080044 if (tbl[i].mchid != mch_id || tbl[i].tdp != tdp)
V Sowmyac6d71662021-07-15 08:11:08 +053045 continue;
46 return tbl[i].conf[domain];
47 }
48
49 printk(BIOS_ERR, "ERROR: Unknown MCH (0x%x) in %s\n", mch_id, __func__);
50 return 0;
51}
52
Curtis Chen1f8563e2021-11-30 14:04:48 +080053/* Per the power map from #613643, update ADL-P 6+8+2 (28W) VR configuration */
V Sowmyac6d71662021-07-15 08:11:08 +053054static const struct vr_lookup vr_config_ll[] = {
Curtis Chenea1bb5f2021-11-25 13:17:42 +080055 { PCI_DEVICE_ID_INTEL_ADL_P_ID_1, 45, VR_CFG_ALL_DOMAINS_LOADLINE(2.3, 3.2) },
56 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 45, VR_CFG_ALL_DOMAINS_LOADLINE(2.3, 3.2) },
Curtis Chen1f8563e2021-11-30 14:04:48 +080057 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 28, VR_CFG_ALL_DOMAINS_LOADLINE(2.3, 3.2) },
Curtis Chenea1bb5f2021-11-25 13:17:42 +080058 { PCI_DEVICE_ID_INTEL_ADL_P_ID_5, 28, VR_CFG_ALL_DOMAINS_LOADLINE(2.3, 3.2) },
59 { PCI_DEVICE_ID_INTEL_ADL_P_ID_6, 15, VR_CFG_ALL_DOMAINS_LOADLINE(2.8, 3.2) },
60 { PCI_DEVICE_ID_INTEL_ADL_P_ID_7, 15, VR_CFG_ALL_DOMAINS_LOADLINE(2.8, 3.2) },
V Sowmyac6d71662021-07-15 08:11:08 +053061};
62
63static const struct vr_lookup vr_config_icc[] = {
Curtis Chenea1bb5f2021-11-25 13:17:42 +080064 { PCI_DEVICE_ID_INTEL_ADL_P_ID_1, 45, VR_CFG_ALL_DOMAINS_ICC(120, 50) },
65 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 45, VR_CFG_ALL_DOMAINS_ICC(160, 50) },
Curtis Chen1f8563e2021-11-30 14:04:48 +080066 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 28, VR_CFG_ALL_DOMAINS_ICC(111, 50) },
Curtis Chenea1bb5f2021-11-25 13:17:42 +080067 { PCI_DEVICE_ID_INTEL_ADL_P_ID_5, 28, VR_CFG_ALL_DOMAINS_ICC(85, 50) },
68 { PCI_DEVICE_ID_INTEL_ADL_P_ID_6, 15, VR_CFG_ALL_DOMAINS_ICC(80, 40) },
69 { PCI_DEVICE_ID_INTEL_ADL_P_ID_7, 15, VR_CFG_ALL_DOMAINS_ICC(80, 40) },
V Sowmyac6d71662021-07-15 08:11:08 +053070};
71
72static const struct vr_lookup vr_config_tdc_timewindow[] = {
Curtis Chenea1bb5f2021-11-25 13:17:42 +080073 { PCI_DEVICE_ID_INTEL_ADL_P_ID_1, 45, VR_CFG_ALL_DOMAINS_TDC(28000, 28000) },
74 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 45, VR_CFG_ALL_DOMAINS_TDC(28000, 28000) },
Curtis Chen1f8563e2021-11-30 14:04:48 +080075 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 28, VR_CFG_ALL_DOMAINS_TDC(28000, 28000) },
Curtis Chenea1bb5f2021-11-25 13:17:42 +080076 { PCI_DEVICE_ID_INTEL_ADL_P_ID_5, 28, VR_CFG_ALL_DOMAINS_TDC(28000, 28000) },
77 { PCI_DEVICE_ID_INTEL_ADL_P_ID_6, 15, VR_CFG_ALL_DOMAINS_TDC(28000, 28000) },
78 { PCI_DEVICE_ID_INTEL_ADL_P_ID_7, 15, VR_CFG_ALL_DOMAINS_TDC(28000, 28000) },
V Sowmyac6d71662021-07-15 08:11:08 +053079};
80
81static const struct vr_lookup vr_config_tdc_currentlimit[] = {
Curtis Chenea1bb5f2021-11-25 13:17:42 +080082 { PCI_DEVICE_ID_INTEL_ADL_P_ID_1, 45, VR_CFG_ALL_DOMAINS_TDC_CURRENT(57, 57) },
83 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 45, VR_CFG_ALL_DOMAINS_TDC_CURRENT(57, 57) },
Curtis Chen1f8563e2021-11-30 14:04:48 +080084 { PCI_DEVICE_ID_INTEL_ADL_P_ID_3, 28, VR_CFG_ALL_DOMAINS_TDC_CURRENT(40, 40) },
Curtis Chenea1bb5f2021-11-25 13:17:42 +080085 { PCI_DEVICE_ID_INTEL_ADL_P_ID_5, 28, VR_CFG_ALL_DOMAINS_TDC_CURRENT(40, 40) },
86 { PCI_DEVICE_ID_INTEL_ADL_P_ID_6, 15, VR_CFG_ALL_DOMAINS_TDC_CURRENT(20, 20) },
87 { PCI_DEVICE_ID_INTEL_ADL_P_ID_7, 15, VR_CFG_ALL_DOMAINS_TDC_CURRENT(20, 20) },
V Sowmyac6d71662021-07-15 08:11:08 +053088};
89
90void fill_vr_domain_config(FSP_S_CONFIG *s_cfg,
91 int domain, const struct vr_config *chip_cfg)
92{
93 const struct vr_config *cfg;
94
95 if (domain < 0 || domain >= NUM_VR_DOMAINS)
96 return;
97
98 /* Use device tree override if requested */
99 if (chip_cfg->vr_config_enable) {
100 cfg = chip_cfg;
101
Bora Guvendikf6f12582021-09-02 13:23:03 -0700102 if (cfg->ac_loadline)
103 s_cfg->AcLoadline[domain] = cfg->ac_loadline;
104 if (cfg->dc_loadline)
105 s_cfg->DcLoadline[domain] = cfg->dc_loadline;
106 if (cfg->icc_max)
107 s_cfg->IccMax[domain] = cfg->icc_max;
V Sowmyac6d71662021-07-15 08:11:08 +0530108 s_cfg->TdcTimeWindow[domain] = cfg->tdc_timewindow;
109 s_cfg->TdcCurrentLimit[domain] = cfg->tdc_currentlimit;
V Sowmyac6d71662021-07-15 08:11:08 +0530110 } else {
111 uint16_t mch_id = 0;
Curtis Chenea1bb5f2021-11-25 13:17:42 +0800112 uint8_t tdp = get_cpu_tdp();
V Sowmyac6d71662021-07-15 08:11:08 +0530113
114 if (!mch_id) {
115 struct device *dev = pcidev_path_on_root(SA_DEVFN_ROOT);
116 mch_id = dev ? pci_read_config16(dev, PCI_DEVICE_ID) : 0xffff;
117 }
118
119 s_cfg->AcLoadline[domain] = load_table(vr_config_ll, ARRAY_SIZE(vr_config_ll),
Curtis Chenea1bb5f2021-11-25 13:17:42 +0800120 domain, mch_id, tdp);
V Sowmyac6d71662021-07-15 08:11:08 +0530121 s_cfg->DcLoadline[domain] = load_table(vr_config_ll, ARRAY_SIZE(vr_config_ll),
Curtis Chenea1bb5f2021-11-25 13:17:42 +0800122 domain, mch_id, tdp);
V Sowmyac6d71662021-07-15 08:11:08 +0530123 s_cfg->IccMax[domain] = load_table(vr_config_icc, ARRAY_SIZE(vr_config_icc),
Curtis Chenea1bb5f2021-11-25 13:17:42 +0800124 domain, mch_id, tdp);
V Sowmyac6d71662021-07-15 08:11:08 +0530125 s_cfg->TdcTimeWindow[domain] = load_table(vr_config_tdc_timewindow,
126 ARRAY_SIZE(vr_config_tdc_timewindow),
Curtis Chenea1bb5f2021-11-25 13:17:42 +0800127 domain, mch_id, tdp);
V Sowmyac6d71662021-07-15 08:11:08 +0530128 s_cfg->TdcCurrentLimit[domain] = load_table(vr_config_tdc_currentlimit,
129 ARRAY_SIZE(vr_config_tdc_currentlimit),
Curtis Chenea1bb5f2021-11-25 13:17:42 +0800130 domain, mch_id, tdp);
Ronak Kanabar1f88a712021-07-15 19:02:22 +0530131 }
132
133 /* Check TdcTimeWindow and TdcCurrentLimit,
134 Set TdcEnable and Set VR TDC Input current to root mean square */
135 if (s_cfg->TdcTimeWindow[domain] != 0 && s_cfg->TdcCurrentLimit[domain] != 0) {
136 s_cfg->TdcEnable[domain] = 1;
137 s_cfg->Irms[domain] = 1;
V Sowmyac6d71662021-07-15 08:11:08 +0530138 }
139}