diff options
author | Alec Berg <alecaberg@chromium.org> | 2014-07-13 17:42:32 -0700 |
---|---|---|
committer | chrome-internal-fetch <chrome-internal-fetch@google.com> | 2014-07-28 21:53:13 +0000 |
commit | 537432085a3e818cff99ce82efdaa92f3fd11c1c (patch) | |
tree | 1e54d18dfd88bfb2805c614a385444ec4ea27173 /board/plankton/usb_pd_policy.c | |
parent | 3ac281f270790ab0e333ef557e4faacf12412c6b (diff) | |
download | chrome-ec-537432085a3e818cff99ce82efdaa92f3fd11c1c.tar.gz |
Add board support for Plankton
Board support for Planton, the Raiden testing board for type-C
functional testing.
BUG=none
BRANCH=none
TEST=make BOARD=plankton, load onto a plankton, and verify
buttons are read correctly, and connect raiden to samus and
verify that PD communication is successful
Change-Id: I40922d5627d62f7f3540ac6a307596428d40baf5
Signed-off-by: Alec Berg <alecaberg@chromium.org>
Reviewed-on: https://chromium-review.googlesource.com/207724
Diffstat (limited to 'board/plankton/usb_pd_policy.c')
-rw-r--r-- | board/plankton/usb_pd_policy.c | 127 |
1 files changed, 127 insertions, 0 deletions
diff --git a/board/plankton/usb_pd_policy.c b/board/plankton/usb_pd_policy.c new file mode 100644 index 0000000000..3cda22108c --- /dev/null +++ b/board/plankton/usb_pd_policy.c @@ -0,0 +1,127 @@ +/* Copyright (c) 2014 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. + */ + +#include "adc.h" +#include "board.h" +#include "common.h" +#include "console.h" +#include "gpio.h" +#include "hooks.h" +#include "registers.h" +#include "task.h" +#include "timer.h" +#include "util.h" +#include "usb_pd.h" + +#define CPRINTS(format, args...) cprints(CC_USBPD, format, ## args) + +/* Acceptable margin between requested VBUS and measured value */ +#define MARGIN_MV 400 /* mV */ + +/* we are not acting as a source */ +const uint32_t pd_src_pdo[] = { + PDO_FIXED(5000, 500, PDO_FIXED_EXTERNAL), + PDO_FIXED(12000, 3000, PDO_FIXED_EXTERNAL), +}; +const int pd_src_pdo_cnt = ARRAY_SIZE(pd_src_pdo); + +/* Fake PDOs : we just want our pre-defined voltages */ +const uint32_t pd_snk_pdo[] = { + PDO_FIXED(5000, 500, 0), + PDO_FIXED(12000, 500, 0), + PDO_FIXED(20000, 500, 0), +}; +const int pd_snk_pdo_cnt = ARRAY_SIZE(pd_snk_pdo); + +/* Desired voltage requested as a sink (in millivolts) */ +static unsigned select_mv = 5000; + +int pd_choose_voltage(int cnt, uint32_t *src_caps, uint32_t *rdo) +{ + int i; + int ma; + int set_mv = select_mv; + + /* Default to 5V */ + if (set_mv <= 0) + set_mv = 5000; + + /* Get the selected voltage */ + for (i = cnt; i >= 0; i--) { + int mv = ((src_caps[i] >> 10) & 0x3FF) * 50; + int type = src_caps[i] & PDO_TYPE_MASK; + if ((mv == set_mv) && (type == PDO_TYPE_FIXED)) + break; + } + if (i < 0) + return -EC_ERROR_UNKNOWN; + + /* request all the power ... */ + ma = 10 * (src_caps[i] & 0x3FF); + *rdo = RDO_FIXED(i + 1, ma, ma, 0); + ccprintf("Request [%d] %d V %d mA\n", i, set_mv/1000, ma); + return EC_SUCCESS; +} + +void pd_set_max_voltage(unsigned mv) +{ + select_mv = mv; +} + +int requested_voltage_idx; +int pd_request_voltage(uint32_t rdo) +{ + int op_ma = rdo & 0x3FF; + int max_ma = (rdo >> 10) & 0x3FF; + int idx = rdo >> 28; + uint32_t pdo; + uint32_t pdo_ma; + + if (!idx || idx > pd_src_pdo_cnt) + return EC_ERROR_INVAL; /* Invalid index */ + + /* check current ... */ + pdo = pd_src_pdo[idx - 1]; + pdo_ma = (pdo & 0x3ff); + if (op_ma > pdo_ma) + return EC_ERROR_INVAL; /* too much op current */ + if (max_ma > pdo_ma) + return EC_ERROR_INVAL; /* too much max current */ + + ccprintf("Switch to %d V %d mA (for %d/%d mA)\n", + ((pdo >> 10) & 0x3ff) * 50, (pdo & 0x3ff) * 10, + ((rdo >> 10) & 0x3ff) * 10, (rdo & 0x3ff) * 10); + + requested_voltage_idx = idx; + + return EC_SUCCESS; +} + +int pd_set_power_supply_ready(int port) +{ + /* Output the correct voltage */ + gpio_set_level(requested_voltage_idx ? GPIO_USBC_12V_EN : + GPIO_USBC_5V_EN, 1); + + return EC_SUCCESS; +} + +void pd_power_supply_reset(int port) +{ + /* Kill VBUS */ + requested_voltage_idx = 0; + gpio_set_level(GPIO_USBC_5V_EN, 0); + gpio_set_level(GPIO_USBC_12V_EN, 0); +} + +int pd_board_checks(void) +{ + return EC_SUCCESS; +} + +int pd_power_negotiation_allowed(void) +{ + return 1; +} |