166 lines
4.7 KiB
C
166 lines
4.7 KiB
C
|
/* Copyright 2018 The Chromium OS Authors. All rights reserved.
|
||
|
* Use of this source code is governed by a BSD-style license that can be
|
||
|
* found in the LICENSE file.
|
||
|
*/
|
||
|
|
||
|
/* DragonEgg board-specific configuration */
|
||
|
#include "adc.h"
|
||
|
#include "adc_chip.h"
|
||
|
#include "button.h"
|
||
|
#include "common.h"
|
||
|
#include "charger.h"
|
||
|
#include "console.h"
|
||
|
#include "driver/ppc/nx20p348x.h"
|
||
|
#include "driver/ppc/sn5s330.h"
|
||
|
#include "ec_commands.h"
|
||
|
#include "extpower.h"
|
||
|
#include "gpio.h"
|
||
|
#include "hooks.h"
|
||
|
#include "host_command.h"
|
||
|
#include "intc.h"
|
||
|
#include "lid_switch.h"
|
||
|
#include "power.h"
|
||
|
#include "power_button.h"
|
||
|
#include "pwm.h"
|
||
|
#include "pwm_chip.h"
|
||
|
#include "spi.h"
|
||
|
#include "switch.h"
|
||
|
#include "system.h"
|
||
|
#include "uart.h"
|
||
|
#include "usb_pd.h"
|
||
|
#include "util.h"
|
||
|
|
||
|
static void ppc_interrupt(enum gpio_signal signal)
|
||
|
{
|
||
|
|
||
|
switch (signal) {
|
||
|
case GPIO_USB_C0_TCPPC_INT_L:
|
||
|
sn5s330_interrupt(0);
|
||
|
break;
|
||
|
|
||
|
case GPIO_USB_C2_TCPPC_INT_ODL:
|
||
|
nx20p348x_interrupt(2);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void tcpc_alert_event(enum gpio_signal signal)
|
||
|
{
|
||
|
int port = -1;
|
||
|
|
||
|
/*
|
||
|
* Since C0/C1 TCPC are embedded within EC, we don't need the PDCMD
|
||
|
* tasks.The (embedded) TCPC status since chip driver code will
|
||
|
* handles its own interrupts and forward the correct events to
|
||
|
* the PD_C0/1 task. See it83xx/intc.c
|
||
|
*/
|
||
|
switch (signal) {
|
||
|
case GPIO_USB_C2_TCPC_INT_ODL:
|
||
|
port = 2;
|
||
|
break;
|
||
|
default:
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
schedule_deferred_pd_interrupt(port);
|
||
|
}
|
||
|
|
||
|
#include "gpio_list.h" /* Must come after other header files. */
|
||
|
|
||
|
/******************************************************************************/
|
||
|
/* ADC channels */
|
||
|
const struct adc_t adc_channels[] = {
|
||
|
/* Vbus C0 sensing (7.3x voltage divider). PPVAR_USB_C0_VBUS */
|
||
|
[ADC_VBUS_C0] = {.name = "VBUS_C0",
|
||
|
.factor_mul = (ADC_MAX_MVOLT * 73) / 10,
|
||
|
.factor_div = ADC_READ_MAX + 1,
|
||
|
.shift = 0,
|
||
|
.channel = CHIP_ADC_CH1},
|
||
|
/* Vbus C1 sensing (7.3x voltage divider). PPVAR_USB_C1_VBUS */
|
||
|
[ADC_VBUS_C1] = {.name = "VBUS_C1",
|
||
|
.factor_mul = (ADC_MAX_MVOLT * 73) / 10,
|
||
|
.factor_div = ADC_READ_MAX + 1,
|
||
|
.shift = 0,
|
||
|
.channel = CHIP_ADC_CH0},
|
||
|
};
|
||
|
BUILD_ASSERT(ARRAY_SIZE(adc_channels) == ADC_CH_COUNT);
|
||
|
|
||
|
/******************************************************************************/
|
||
|
/* SPI devices */
|
||
|
/* TODO(b/110880394): Fill out correctly (SPI FLASH) */
|
||
|
const struct spi_device_t spi_devices[] = {
|
||
|
};
|
||
|
const unsigned int spi_devices_used = ARRAY_SIZE(spi_devices);
|
||
|
|
||
|
/******************************************************************************/
|
||
|
/* PWM channels. Must be in the exactly same order as in enum pwm_channel. */
|
||
|
const struct pwm_t pwm_channels[] = {
|
||
|
[PWM_CH_KBLIGHT] = { .channel = 0, .flags = 0, .freq_hz = 100 },
|
||
|
[PWM_CH_LED_RED] = { .channel = 2, .flags = PWM_CONFIG_DSLEEP |
|
||
|
PWM_CONFIG_ACTIVE_LOW, .freq_hz = 100 },
|
||
|
[PWM_CH_LED_GREEN] = { .channel = 1, .flags = PWM_CONFIG_DSLEEP |
|
||
|
PWM_CONFIG_ACTIVE_LOW, .freq_hz = 100 },
|
||
|
[PWM_CH_LED_BLUE] = { .channel = 3, .flags = PWM_CONFIG_DSLEEP |
|
||
|
PWM_CONFIG_ACTIVE_LOW, .freq_hz = 100 },
|
||
|
};
|
||
|
BUILD_ASSERT(ARRAY_SIZE(pwm_channels) == PWM_CH_COUNT);
|
||
|
|
||
|
/* GPIO to enable/disable the USB Type-A port. */
|
||
|
const int usb_port_enable[CONFIG_USB_PORT_POWER_SMART_PORT_COUNT] = {
|
||
|
GPIO_EN_USB_A_5V,
|
||
|
};
|
||
|
|
||
|
void board_overcurrent_event(int port)
|
||
|
{
|
||
|
if (port == 0) {
|
||
|
/* TODO(b/111281797): When does this get set high again? */
|
||
|
gpio_set_level(GPIO_USB_OC_ODL, 0);
|
||
|
cprints(CC_USBPD, "p%d: overcurrent!", port);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void board_disable_learn_mode(void)
|
||
|
{
|
||
|
/* Disable learn mode after checking to make sure AC is still present */
|
||
|
if (extpower_is_present())
|
||
|
charger_discharge_on_ac(0);
|
||
|
}
|
||
|
DECLARE_DEFERRED(board_disable_learn_mode);
|
||
|
|
||
|
static void board_extpower(void)
|
||
|
{
|
||
|
/*
|
||
|
* For the bq25710 charger, we need the switching converter to remain
|
||
|
* disabled until ~130 msec from when VBUS present to allow the
|
||
|
* converter to be biased properly. Otherwise, there will be a reverse
|
||
|
* buck/boost until the converter is biased. The recommendation is to
|
||
|
* exit learn mode 200 msec after external charger is connected.
|
||
|
*
|
||
|
* TODO(b/112372451): When there are updated versions of the bq25710,
|
||
|
* this set of changes can be removed.
|
||
|
*/
|
||
|
if (extpower_is_present()) {
|
||
|
hook_call_deferred(&board_disable_learn_mode_data, 200 * MSEC);
|
||
|
} else {
|
||
|
/* Enable charger learn mode */
|
||
|
charger_discharge_on_ac(1);
|
||
|
/* Cancel any pending call to disable learn mode */
|
||
|
hook_call_deferred(&board_disable_learn_mode_data, -1);
|
||
|
}
|
||
|
}
|
||
|
DECLARE_HOOK(HOOK_AC_CHANGE, board_extpower, HOOK_PRIO_DEFAULT);
|
||
|
|
||
|
/* Initialize board. */
|
||
|
static void board_init(void)
|
||
|
{
|
||
|
/*
|
||
|
* On EC reboot, need to always set battery learn mode to the correct
|
||
|
* state based on presence of AC.
|
||
|
*/
|
||
|
board_extpower();
|
||
|
}
|
||
|
DECLARE_HOOK(HOOK_INIT, board_init, HOOK_PRIO_DEFAULT);
|