4b29c4adeb
MPTS is currently not executed by the AML interpreter. Use Method (\\_SB.MPTS) instead of Scope (\\_SB) Method (MPTS) ScopeEnd Tested on Prodrive Hermes. MPTS is now executed at S5. Change-Id: I9074eb4ba55aab3f9a47ae5e3c3ddd338406a5e4 Signed-off-by: Patrick Rudolph <patrick.rudolph@9elements.com> Reviewed-on: https://review.coreboot.org/c/coreboot/+/52382 Tested-by: build bot (Jenkins) <no-reply@coreboot.org> Reviewed-by: Angel Pons <th3fanbus@gmail.com> Reviewed-by: <wouter.eckhardt@prodrive-technologies.com>
242 lines
6.1 KiB
C
242 lines
6.1 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
|
|
#include <arch/cpu.h>
|
|
#include <acpi/acpigen.h>
|
|
#include <cbmem.h>
|
|
#include <console/console.h>
|
|
#include <crc_byte.h>
|
|
#include <bootstate.h>
|
|
#include <device/device.h>
|
|
#include <device/dram/spd.h>
|
|
#include <intelblocks/pmclib.h>
|
|
#include <types.h>
|
|
#include <smbios.h>
|
|
#include "variants/baseboard/include/eeprom.h"
|
|
#include "gpio.h"
|
|
|
|
/* FIXME: Example code below */
|
|
|
|
static void mb_configure_dp1_pwr(bool enable)
|
|
{
|
|
gpio_output(GPP_K3, enable);
|
|
}
|
|
|
|
static void mb_configure_dp2_pwr(bool enable)
|
|
{
|
|
gpio_output(GPP_K4, enable);
|
|
}
|
|
|
|
static void mb_configure_dp3_pwr(bool enable)
|
|
{
|
|
gpio_output(GPP_K5, enable);
|
|
}
|
|
|
|
static void mb_hda_amp_enable(bool enable)
|
|
{
|
|
gpio_output(GPP_C19, enable);
|
|
}
|
|
|
|
static void mb_usb31_rp1_pwr_enable(bool enable)
|
|
{
|
|
gpio_output(GPP_G0, enable);
|
|
}
|
|
|
|
static void mb_usb31_rp2_pwr_enable(bool enable)
|
|
{
|
|
gpio_output(GPP_G1, enable);
|
|
}
|
|
|
|
static void mb_usb31_fp_pwr_enable(bool enable)
|
|
{
|
|
gpio_output(GPP_G2, enable);
|
|
}
|
|
|
|
static void mb_usb2_fp1_pwr_enable(bool enable)
|
|
{
|
|
gpio_output(GPP_G3, enable);
|
|
}
|
|
|
|
static void mb_usb2_fp2_pwr_enable(bool enable)
|
|
{
|
|
gpio_output(GPP_G4, enable);
|
|
}
|
|
|
|
static void copy_meminfo(const struct dimm_info *dimm, union eeprom_dimm_layout *l)
|
|
{
|
|
memset(l, 0, sizeof(*l));
|
|
if (dimm->dimm_size == 0)
|
|
return;
|
|
|
|
strncpy(l->name, (char *)dimm->module_part_number, sizeof(l->name) - 1);
|
|
l->capacity_mib = dimm->dimm_size;
|
|
l->data_width_bits = 8 * (1 << (dimm->bus_width & 0x7));
|
|
l->bus_width_bits = l->data_width_bits + 8 * ((dimm->bus_width >> 3) & 0x3);
|
|
l->ranks = dimm->rank_per_dimm;
|
|
l->controller_id = 0;
|
|
strncpy(l->manufacturer, spd_manufacturer_name(dimm->mod_id),
|
|
sizeof(l->manufacturer) - 1);
|
|
}
|
|
|
|
/*
|
|
* Collect board specific settings and update the CFG EEPROM if necessary.
|
|
* This allows the BMC webui to display the current hardware configuration.
|
|
*/
|
|
static void update_board_layout(void)
|
|
{
|
|
struct eeprom_board_layout layout = {0};
|
|
|
|
printk(BIOS_INFO, "MB: Collecting Board Layout information\n");
|
|
|
|
/* Update CPU fields */
|
|
for (struct device *cpu = all_devices; cpu; cpu = cpu->next) {
|
|
if (cpu->path.type != DEVICE_PATH_APIC)
|
|
continue;
|
|
if (cpu->bus->dev->path.type != DEVICE_PATH_CPU_CLUSTER)
|
|
continue;
|
|
if (!cpu->enabled)
|
|
continue;
|
|
layout.cpu_count++;
|
|
if (!layout.cpu_name[0])
|
|
strcpy(layout.cpu_name, cpu->name);
|
|
}
|
|
|
|
if (cpuid_get_max_func() >= 0x16)
|
|
layout.cpu_max_non_turbo_frequency = cpuid_eax(0x16);
|
|
|
|
/* PCH */
|
|
strcpy(layout.pch_name, "Cannonlake-H C246");
|
|
|
|
/* DRAM */
|
|
struct memory_info *meminfo = cbmem_find(CBMEM_ID_MEMINFO);
|
|
if (meminfo) {
|
|
const size_t meminfo_max = MIN(meminfo->dimm_cnt, ARRAY_SIZE(meminfo->dimm));
|
|
for (size_t i = 0; i < MIN(meminfo_max, ARRAY_SIZE(layout.dimm)); i++)
|
|
copy_meminfo(&meminfo->dimm[i], &layout.dimm[i]);
|
|
}
|
|
|
|
/* Update CRC */
|
|
layout.signature = CRC(layout.raw_layout, sizeof(layout.raw_layout), crc32_byte);
|
|
|
|
printk(BIOS_DEBUG, "BOARD LAYOUT:\n");
|
|
printk(BIOS_DEBUG, " Signature : 0x%x\n", layout.signature);
|
|
printk(BIOS_DEBUG, " CPU name : %s\n", layout.cpu_name);
|
|
printk(BIOS_DEBUG, " CPU count : %u\n", layout.cpu_count);
|
|
printk(BIOS_DEBUG, " CPU freq : %u\n", layout.cpu_max_non_turbo_frequency);
|
|
printk(BIOS_DEBUG, " PCH name : %s\n", layout.pch_name);
|
|
for (size_t i = 0; i < ARRAY_SIZE(layout.dimm); i++)
|
|
printk(BIOS_DEBUG, " DRAM SIZE : %u\n", layout.dimm[i].capacity_mib);
|
|
|
|
if (write_board_settings(&layout))
|
|
printk(BIOS_ERR, "MB: Failed to update Board Layout\n");
|
|
}
|
|
|
|
static void mainboard_init(void *chip_info)
|
|
{
|
|
const struct eeprom_board_settings *const board_cfg = get_board_settings();
|
|
|
|
if (!board_cfg)
|
|
return;
|
|
|
|
/* Enable internal speaker amplifier */
|
|
if (board_cfg->internal_audio_connection == 2)
|
|
mb_hda_amp_enable(1);
|
|
else
|
|
mb_hda_amp_enable(0);
|
|
}
|
|
|
|
static void mainboard_final(struct device *dev)
|
|
{
|
|
update_board_layout();
|
|
|
|
const struct eeprom_board_settings *const board_cfg = get_board_settings();
|
|
|
|
if (!board_cfg)
|
|
return;
|
|
|
|
/* Encoding: 0 -> S0, 1 -> S5 */
|
|
const bool on = !board_cfg->power_state_after_g3;
|
|
|
|
pmc_soc_set_afterg3_en(on);
|
|
}
|
|
|
|
#if CONFIG(HAVE_ACPI_TABLES)
|
|
static void mainboard_acpi_fill_ssdt(const struct device *dev)
|
|
{
|
|
const struct eeprom_board_settings *const board_cfg = get_board_settings();
|
|
|
|
if (!board_cfg)
|
|
return;
|
|
|
|
const unsigned int usb_power_gpios[] = { GPP_G0, GPP_G1, GPP_G2, GPP_G3, GPP_G4 };
|
|
|
|
/* Function pointer to write STXS or CTXS according to EEPROM board setting */
|
|
int (*acpigen_write_soc_gpio_op)(unsigned int gpio_num);
|
|
|
|
if (board_cfg->usb_powered_in_s5)
|
|
acpigen_write_soc_gpio_op = acpigen_soc_set_tx_gpio;
|
|
else
|
|
acpigen_write_soc_gpio_op = acpigen_soc_clear_tx_gpio;
|
|
|
|
acpigen_write_method("\\_SB.MPTS", 1);
|
|
{
|
|
acpigen_write_if_lequal_op_int(ARG0_OP, 5);
|
|
{
|
|
for (size_t i = 0; i < ARRAY_SIZE(usb_power_gpios); i++)
|
|
acpigen_write_soc_gpio_op(usb_power_gpios[i]);
|
|
}
|
|
acpigen_pop_len();
|
|
}
|
|
acpigen_pop_len();
|
|
}
|
|
#endif
|
|
|
|
static void mainboard_enable(struct device *dev)
|
|
{
|
|
/* FIXME: Do runtime configuration once the board is production ready */
|
|
mb_configure_dp1_pwr(1);
|
|
mb_configure_dp2_pwr(1);
|
|
mb_configure_dp3_pwr(1);
|
|
|
|
mb_usb31_rp1_pwr_enable(1);
|
|
mb_usb31_rp2_pwr_enable(1);
|
|
mb_usb31_fp_pwr_enable(1);
|
|
mb_usb2_fp1_pwr_enable(1);
|
|
mb_usb2_fp2_pwr_enable(1);
|
|
|
|
dev->ops->final = mainboard_final;
|
|
|
|
#if CONFIG(HAVE_ACPI_TABLES)
|
|
dev->ops->acpi_fill_ssdt = mainboard_acpi_fill_ssdt;
|
|
#endif
|
|
}
|
|
|
|
struct chip_operations mainboard_ops = {
|
|
.init = mainboard_init,
|
|
.enable_dev = mainboard_enable,
|
|
};
|
|
|
|
/* Must happen before MPinit */
|
|
static void mainboard_early(void *unused)
|
|
{
|
|
const struct eeprom_board_settings *const board_cfg = get_board_settings();
|
|
config_t *config = config_of_soc();
|
|
|
|
if (board_cfg) {
|
|
/* Set Deep Sx */
|
|
config->deep_s5_enable_ac = board_cfg->deep_sx_enabled;
|
|
config->deep_s5_enable_dc = board_cfg->deep_sx_enabled;
|
|
}
|
|
|
|
if (check_signature(offsetof(struct eeprom_layout, supd), FSPS_UPD_SIGNATURE)) {
|
|
struct {
|
|
struct {
|
|
u8 TurboMode;
|
|
} FspsConfig;
|
|
} supd = {0};
|
|
|
|
READ_EEPROM_FSP_S((&supd), FspsConfig.TurboMode);
|
|
config->cpu_turbo_disable = !supd.FspsConfig.TurboMode;
|
|
}
|
|
}
|
|
|
|
BOOT_STATE_INIT_ENTRY(BS_PRE_DEVICE, BS_ON_EXIT, mainboard_early, NULL);
|