blob: 7d9b6f249542ac5c8ac63a29ce145b5eb7d898b6 [file] [log] [blame]
/* SPDX-License-Identifier: GPL-2.0-only */
#include <acpi/acpi.h>
#include <console/console.h>
#include <cpu/x86/smm.h>
#include <ec/acpi/ec.h>
#include <mainboard/ec.h>
#include <timer.h>
#define EC_STATUS 0x50
#define EC_RUNNING (1 << 1)
#define EC_DEVICE_CONTROL_1 0x80
#define EC_DEVICE_CONTROL_1_PROGAS_ON (1 << 0)
#define EC_DEVICE_CONTROL_1_BOOMER_ON (1 << 1)
#define EC_DEVICE_CONTROL_1_BT_RF_ON (1 << 2)
#define EC_DEVICE_CONTROL_1_TP_ON (1 << 3)
#define EC_DEVICE_CONTROL_1_LAN2_RST (1 << 6)
#define EC_DEVICE_CONTROL_2 0x81
#define EC_DEVICE_CONTROL_2_LAN_1_ON (1 << 0)
#define EC_DEVICE_CONTROL_2_LAN_2_ON (1 << 1)
#define EC_DEVICE_CONTROL_2_WLAN_ON (1 << 2)
#define EC_DEVICE_CONTROL_2_USB_ON (1 << 3)
#define EC_DEVICE_CONTROL_2_IDE1_ON (1 << 4)
#define EC_DEVICE_CONTROL_2_IDE2_ON (1 << 5)
#define EC_DEVICE_CONTROL_2_COM1_ON (1 << 6)
#define EC_DEVICE_CONTROL_2_MPI_ON (1 << 7)
#define RUNNING_TIMEOUT_MS 3333
static bool ec_running(void)
{
struct stopwatch sw;
uint8_t ec_status;
stopwatch_init_msecs_expire(&sw, RUNNING_TIMEOUT_MS);
do
ec_status = ec_read(EC_STATUS);
while (!(ec_status & EC_RUNNING) && !stopwatch_expired(&sw));
if (!(ec_status & EC_RUNNING))
printk(BIOS_WARNING, "EC not ready after %dms\n", RUNNING_TIMEOUT_MS);
return !!(ec_status & EC_RUNNING);
}
void ec_enable_devices(bool enable_usb)
{
uint8_t control_1, control_2;
if (!ec_running())
return;
control_1 = ec_read(EC_DEVICE_CONTROL_1);
control_2 = ec_read(EC_DEVICE_CONTROL_2);
printk(BIOS_INFO, "EC previous EDC1: 0x%02x\n", control_1);
printk(BIOS_INFO, "EC previous EDC2: 0x%02x\n", control_2);
control_1 &= ~(EC_DEVICE_CONTROL_1_BT_RF_ON);
control_1 |= EC_DEVICE_CONTROL_1_BOOMER_ON;
control_2 &= ~(EC_DEVICE_CONTROL_2_WLAN_ON | EC_DEVICE_CONTROL_2_USB_ON);
control_2 |= EC_DEVICE_CONTROL_2_MPI_ON;
if (enable_usb)
control_2 |= EC_DEVICE_CONTROL_2_USB_ON;
ec_write(EC_DEVICE_CONTROL_1, control_1);
ec_write(EC_DEVICE_CONTROL_2, control_2);
printk(BIOS_INFO, "EC current EDC1: 0x%02x\n", ec_read(EC_DEVICE_CONTROL_1));
printk(BIOS_INFO, "EC current EDC2: 0x%02x\n", ec_read(EC_DEVICE_CONTROL_2));
}
void mainboard_smi_sleep(const uint8_t slp_typ)
{
uint8_t control_1, control_2;
if (slp_typ != ACPI_S5)
return;
if (!ec_running())
return;
control_1 = ec_read(EC_DEVICE_CONTROL_1);
control_2 = ec_read(EC_DEVICE_CONTROL_2);
printk(BIOS_INFO, "EC previous EDC1: 0x%02x\n", control_1);
printk(BIOS_INFO, "EC previous EDC2: 0x%02x\n", control_2);
control_1 &= ~(EC_DEVICE_CONTROL_1_BOOMER_ON);
control_2 &= ~(EC_DEVICE_CONTROL_2_USB_ON | EC_DEVICE_CONTROL_2_MPI_ON);
ec_write(EC_DEVICE_CONTROL_1, control_1);
ec_write(EC_DEVICE_CONTROL_2, control_2);
printk(BIOS_INFO, "EC current EDC1: 0x%02x\n", ec_read(EC_DEVICE_CONTROL_1));
printk(BIOS_INFO, "EC current EDC2: 0x%02x\n", ec_read(EC_DEVICE_CONTROL_2));
}