2020-04-05 13:21:31 +02:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-only */
|
2016-03-02 08:45:47 +01:00
|
|
|
|
|
|
|
#include <boardid.h>
|
|
|
|
#include <boot/coreboot_tables.h>
|
|
|
|
#include <console/console.h>
|
|
|
|
#include <delay.h>
|
|
|
|
#include <drivers/i2c/ww_ring/ww_ring.h>
|
|
|
|
#include <gpio.h>
|
|
|
|
#include <soc/cdp.h>
|
2016-03-08 10:32:56 +01:00
|
|
|
#include <soc/blsp.h>
|
2016-03-02 08:45:47 +01:00
|
|
|
#include <timer.h>
|
|
|
|
#include <vendorcode/google/chromeos/chromeos.h>
|
|
|
|
|
2019-03-20 17:30:38 +01:00
|
|
|
#define PP_SW 41
|
2016-03-02 08:45:47 +01:00
|
|
|
|
2016-07-15 03:05:20 +02:00
|
|
|
static int get_rec_sw_gpio_pin(void)
|
|
|
|
{
|
|
|
|
uint8_t board_rev = board_id();
|
2016-07-29 03:55:26 +02:00
|
|
|
|
2016-07-15 03:05:20 +02:00
|
|
|
switch (board_rev) {
|
2016-07-21 19:51:47 +02:00
|
|
|
case BOARD_ID_GALE_PROTO:
|
2016-07-15 03:05:20 +02:00
|
|
|
case BOARD_ID_GALE_EVT:
|
2016-07-29 03:55:26 +02:00
|
|
|
case BOARD_ID_GALE_EVT2_0:
|
|
|
|
case BOARD_ID_GALE_EVT2_1:
|
2016-07-15 03:05:20 +02:00
|
|
|
return 7;
|
|
|
|
case BOARD_ID_GALE_EVT3:
|
|
|
|
default:
|
|
|
|
return 57;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int get_wp_status_gpio_pin(void)
|
|
|
|
{
|
|
|
|
uint8_t board_rev = board_id();
|
|
|
|
switch (board_rev) {
|
2016-07-21 19:51:47 +02:00
|
|
|
case BOARD_ID_GALE_PROTO:
|
2016-07-15 03:05:20 +02:00
|
|
|
case BOARD_ID_GALE_EVT:
|
2016-07-29 03:55:26 +02:00
|
|
|
case BOARD_ID_GALE_EVT2_0:
|
|
|
|
case BOARD_ID_GALE_EVT2_1:
|
2016-07-15 03:05:20 +02:00
|
|
|
return 6;
|
|
|
|
case BOARD_ID_GALE_EVT3:
|
|
|
|
default:
|
|
|
|
return 53;
|
|
|
|
}
|
|
|
|
}
|
2016-03-02 08:45:47 +01:00
|
|
|
static int read_gpio(gpio_t gpio_num)
|
|
|
|
{
|
|
|
|
gpio_tlmm_config_set(gpio_num, GPIO_FUNC_DISABLE,
|
|
|
|
GPIO_NO_PULL, GPIO_2MA, GPIO_DISABLE);
|
|
|
|
udelay(10); /* Should be enough to settle. */
|
|
|
|
return gpio_get(gpio_num);
|
|
|
|
}
|
|
|
|
|
|
|
|
void fill_lb_gpios(struct lb_gpios *gpios)
|
|
|
|
{
|
|
|
|
struct lb_gpio chromeos_gpios[] = {
|
2019-04-06 18:37:14 +02:00
|
|
|
{PP_SW, ACTIVE_LOW, read_gpio(PP_SW), "presence"},
|
2016-03-02 08:45:47 +01:00
|
|
|
{-1, ACTIVE_LOW, 1, "power"},
|
|
|
|
{-1, ACTIVE_LOW, 0, "lid"},
|
|
|
|
};
|
|
|
|
lb_add_gpios(gpios, chromeos_gpios, ARRAY_SIZE(chromeos_gpios));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The recovery switch on storm is overloaded: it needs to be pressed for a
|
|
|
|
* certain duration at startup to signal different requests:
|
|
|
|
*
|
|
|
|
* - keeping it pressed for 8 to 16 seconds after startup signals the need for
|
|
|
|
* factory reset (wipeout);
|
|
|
|
* - keeping it pressed for longer than 16 seconds signals the need for Chrome
|
|
|
|
* OS recovery.
|
|
|
|
*
|
|
|
|
* The state is read once and cached for following inquiries. The below enum
|
|
|
|
* lists possible states.
|
|
|
|
*/
|
|
|
|
enum switch_state {
|
|
|
|
not_probed = -1,
|
|
|
|
no_req,
|
|
|
|
recovery_req,
|
|
|
|
wipeout_req
|
|
|
|
};
|
|
|
|
|
|
|
|
static void display_pattern(int pattern)
|
|
|
|
{
|
2016-05-19 02:48:33 +02:00
|
|
|
ww_ring_display_pattern(BLSP_QUP_ID_3, pattern);
|
2016-03-02 08:45:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#define WIPEOUT_MODE_DELAY_MS (8 * 1000)
|
|
|
|
#define RECOVERY_MODE_EXTRA_DELAY_MS (8 * 1000)
|
|
|
|
|
|
|
|
static enum switch_state get_switch_state(void)
|
|
|
|
{
|
|
|
|
struct stopwatch sw;
|
|
|
|
int sampled_value;
|
2016-07-15 03:05:20 +02:00
|
|
|
uint8_t rec_sw;
|
2016-03-02 08:45:47 +01:00
|
|
|
static enum switch_state saved_state = not_probed;
|
|
|
|
|
|
|
|
if (saved_state != not_probed)
|
|
|
|
return saved_state;
|
|
|
|
|
2016-07-15 03:05:20 +02:00
|
|
|
rec_sw = get_rec_sw_gpio_pin();
|
2019-04-06 18:37:14 +02:00
|
|
|
sampled_value = !read_gpio(rec_sw);
|
2016-03-02 08:45:47 +01:00
|
|
|
|
|
|
|
if (!sampled_value) {
|
|
|
|
saved_state = no_req;
|
|
|
|
display_pattern(WWR_NORMAL_BOOT);
|
|
|
|
return saved_state;
|
|
|
|
}
|
|
|
|
|
|
|
|
display_pattern(WWR_RECOVERY_PUSHED);
|
|
|
|
printk(BIOS_INFO, "recovery button pressed\n");
|
|
|
|
|
|
|
|
stopwatch_init_msecs_expire(&sw, WIPEOUT_MODE_DELAY_MS);
|
|
|
|
|
|
|
|
do {
|
2019-04-06 18:37:14 +02:00
|
|
|
sampled_value = !read_gpio(rec_sw);
|
2016-03-02 08:45:47 +01:00
|
|
|
if (!sampled_value)
|
|
|
|
break;
|
|
|
|
} while (!stopwatch_expired(&sw));
|
|
|
|
|
|
|
|
if (sampled_value) {
|
|
|
|
display_pattern(WWR_WIPEOUT_REQUEST);
|
|
|
|
printk(BIOS_INFO, "wipeout requested, checking recovery\n");
|
|
|
|
stopwatch_init_msecs_expire(&sw, RECOVERY_MODE_EXTRA_DELAY_MS);
|
|
|
|
do {
|
2019-04-06 18:37:14 +02:00
|
|
|
sampled_value = !read_gpio(rec_sw);
|
2016-03-02 08:45:47 +01:00
|
|
|
if (!sampled_value)
|
|
|
|
break;
|
|
|
|
} while (!stopwatch_expired(&sw));
|
|
|
|
|
|
|
|
if (sampled_value) {
|
|
|
|
saved_state = recovery_req;
|
|
|
|
display_pattern(WWR_RECOVERY_REQUEST);
|
|
|
|
printk(BIOS_INFO, "recovery requested\n");
|
|
|
|
} else {
|
|
|
|
saved_state = wipeout_req;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
saved_state = no_req;
|
|
|
|
display_pattern(WWR_NORMAL_BOOT);
|
|
|
|
}
|
|
|
|
|
|
|
|
return saved_state;
|
|
|
|
}
|
|
|
|
|
|
|
|
int get_recovery_mode_switch(void)
|
|
|
|
{
|
|
|
|
return get_switch_state() == recovery_req;
|
|
|
|
}
|
|
|
|
|
|
|
|
int get_wipeout_mode_switch(void)
|
|
|
|
{
|
|
|
|
return get_switch_state() == wipeout_req;
|
|
|
|
}
|
|
|
|
|
|
|
|
int get_write_protect_state(void)
|
|
|
|
{
|
2019-04-06 18:37:14 +02:00
|
|
|
return !read_gpio(get_wp_status_gpio_pin());
|
2016-03-02 08:45:47 +01:00
|
|
|
}
|
2021-08-12 09:47:06 +02:00
|
|
|
|
|
|
|
int get_ec_is_trusted(void)
|
|
|
|
{
|
|
|
|
/* Do not have a Chrome EC involved in entering recovery mode;
|
|
|
|
Always return trusted. */
|
|
|
|
return 1;
|
|
|
|
}
|