summaryrefslogtreecommitdiff
path: root/src/mainboard/siemens/chili/ec.c
blob: bc0eb7d9ff40381f5bc4312b18f975207761884a (plain)
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));
}