1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
|
/* 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 <timer.h>
#include "ec.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));
}
|