mirror of
https://github.com/revyos/th1520-vendor-uboot.git
synced 2026-06-21 09:02:25 +02:00
sync: huiwei: charge: add support for pd charging
commit: 475700a11696fe2dc45c01cdfbf5f3cc470e3a60
This commit is contained in:
@@ -1,6 +1,7 @@
|
||||
/dts-v1/;
|
||||
|
||||
#include <dt-bindings/pmic/light_pmic.h>
|
||||
#include <dt-bindings/usb/pd.h>
|
||||
|
||||
/ {
|
||||
model = "T-HEAD c910 light";
|
||||
@@ -150,6 +151,27 @@
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
usbc0: husb311_0@4e {
|
||||
compatible = "hynetek,husb311";
|
||||
int-n-gpios = <&gpio3_porta 10 1>;
|
||||
reg = <0x4e>;
|
||||
status = "okay";
|
||||
|
||||
usb_con0: connector {
|
||||
compatible = "usb-c-connector";
|
||||
label = "USB-C";
|
||||
data-role = "dual";
|
||||
power-role = "dual";
|
||||
try-power-role = "sink";
|
||||
source-pdos = <PDO_FIXED(5000, 2000, PDO_FIXED_USB_COMM)>;
|
||||
sink-pdos =
|
||||
<PDO_FIXED(5000, 2000, PDO_FIXED_USB_COMM)
|
||||
PDO_FIXED(9000, 3000, PDO_FIXED_USB_COMM)
|
||||
PDO_FIXED(12000, 3000, PDO_FIXED_USB_COMM)>;
|
||||
op-sink-microwatt = <10000000>;
|
||||
};
|
||||
};
|
||||
|
||||
cw2015@62 {
|
||||
clock-frequency = <100000>;
|
||||
status = "okay";
|
||||
@@ -174,6 +196,8 @@
|
||||
status = "okay";
|
||||
compatible = "ti,bq25703";
|
||||
reg = <0x6b>;
|
||||
typec0-enable-gpios = <&gpio3_porta 13 0>; //CHG_PATH_SEL0_180
|
||||
typec1-enable-gpios = <&gpio3_porta 12 0>; //CHG_PATH_SEL1_180
|
||||
ti,charge-current = <2500000>;
|
||||
ti,max-input-voltage = <5000000>;
|
||||
ti,input-current = <2000000>;
|
||||
@@ -194,6 +218,27 @@
|
||||
compatible = "mcu_hc32fx";
|
||||
reg = <0x4c>;
|
||||
};
|
||||
|
||||
usbc1: husb311_1@4e {
|
||||
compatible = "hynetek,husb311";
|
||||
int-n-gpios = <&gpio1_porta 5 1>;
|
||||
reg = <0x4e>;
|
||||
status = "okay";
|
||||
|
||||
usb_con1: connector {
|
||||
compatible = "usb-c-connector";
|
||||
label = "USB-C";
|
||||
data-role = "dual";
|
||||
power-role = "dual";
|
||||
try-power-role = "sink";
|
||||
source-pdos = <PDO_FIXED(5000, 2000, PDO_FIXED_USB_COMM)>;
|
||||
sink-pdos =
|
||||
<PDO_FIXED(5000, 3000, PDO_FIXED_USB_COMM)
|
||||
PDO_FIXED(9000, 3000, PDO_FIXED_USB_COMM)
|
||||
PDO_FIXED(12000, 3000, PDO_FIXED_USB_COMM)>;
|
||||
op-sink-microwatt = <10000000>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
i2c2: i2c@ffec00c000{
|
||||
@@ -395,6 +440,21 @@
|
||||
};
|
||||
};
|
||||
|
||||
gpio3: gpio@ffe7f38000 {
|
||||
compatible = "snps,dw-apb-gpio";
|
||||
reg = <0xff 0xe7f38000 0x0 0x1000>;
|
||||
clocks = <&dummy_apb>;
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
gpio3_porta: gpio-controller@0 {
|
||||
compatible = "snps,dw-apb-gpio-port";
|
||||
gpio-controller;
|
||||
#gpio-cells = <2>;
|
||||
snps,nr-gpios = <32>;
|
||||
reg = <0>;
|
||||
};
|
||||
};
|
||||
|
||||
gpio0: gpio@ffec005000 {
|
||||
compatible = "snps,dw-apb-gpio";
|
||||
reg = <0xff 0xec005000 0x0 0x1000>;
|
||||
@@ -570,6 +630,7 @@
|
||||
};
|
||||
|
||||
ili9881c_panel {
|
||||
status = "disabled";
|
||||
compatible = "ilitek,ili9881c";
|
||||
backlight = <&lcd_backlight>;
|
||||
reset-gpios = <&gpio1_porta 5 1>; /* active low */
|
||||
|
||||
@@ -131,3 +131,7 @@ CONFIG_LED=y
|
||||
CONFIG_LED_GPIO=y
|
||||
CONFIG_DM_MCU=y
|
||||
CONFIG_MCU_HC32fX=y
|
||||
CONFIG_DM_POWER_DELIVERY=y
|
||||
CONFIG_TYPEC_TCPM=y
|
||||
CONFIG_TYPEC_TCPCI=y
|
||||
CONFIG_TYPEC_HUSB311=y
|
||||
|
||||
@@ -85,6 +85,7 @@ obj-y += misc/
|
||||
obj-$(CONFIG_MMC) += mmc/
|
||||
obj-$(CONFIG_NVME) += nvme/
|
||||
obj-$(CONFIG_PCI_ENDPOINT) += pci_endpoint/
|
||||
obj-$(CONFIG_DM_POWER_DELIVERY) += power/power_delivery/
|
||||
obj-y += dfu/
|
||||
obj-$(CONFIG_PCH) += pch/
|
||||
obj-y += phy/allwinner/
|
||||
|
||||
@@ -10,6 +10,8 @@ source "drivers/power/regulator/Kconfig"
|
||||
|
||||
source "drivers/power/charge/Kconfig"
|
||||
|
||||
source "drivers/power/power_delivery/Kconfig"
|
||||
|
||||
config DM_CHARGE_DISPLAY
|
||||
bool "Enable driver model for charge display support"
|
||||
depends on DM
|
||||
|
||||
@@ -10,6 +10,7 @@
|
||||
#include <dm/uclass.h>
|
||||
#include <power/fuel_gauge.h>
|
||||
#include <power/pmic.h>
|
||||
#include <power/power_delivery/power_delivery.h>
|
||||
|
||||
DECLARE_GLOBAL_DATA_PTR;
|
||||
|
||||
@@ -39,6 +40,9 @@ DECLARE_GLOBAL_DATA_PTR;
|
||||
#define BQ25703_CHARGERSTAUS_REG 0x20
|
||||
#define BQ25703_INPUTVOLTAGE_REG 0x0A
|
||||
#define BQ25703_INPUTCURREN_REG 0x0E
|
||||
#define PD_MUN 2
|
||||
#define TYPEC0_I2C "i2c@ffe7f20000"
|
||||
#define TYPEC1_I2C "i2c@ffe7f24000"
|
||||
|
||||
enum bq25700_table_ids {
|
||||
/* range tables */
|
||||
@@ -56,6 +60,7 @@ struct bq25700 {
|
||||
struct udevice *dev;
|
||||
u32 ichg;
|
||||
u32 chip_id;
|
||||
struct udevice *pd[PD_MUN];
|
||||
};
|
||||
|
||||
struct bq25700_range {
|
||||
@@ -72,7 +77,7 @@ static int bq25700_read(struct bq25700 *charger, uint reg)
|
||||
ret = dm_i2c_read(charger->dev, reg, (u8 *)&val, 2);
|
||||
if (ret) {
|
||||
printf("write error to device: %p register: %#x!",
|
||||
charger->dev, reg);
|
||||
charger->dev, reg);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -86,7 +91,7 @@ static int bq25700_write(struct bq25700 *charger, uint reg, u16 val)
|
||||
ret = dm_i2c_write(charger->dev, reg, (u8 *)&val, 2);
|
||||
if (ret) {
|
||||
printf("write error to device: %p register: %#x!",
|
||||
charger->dev, reg);
|
||||
charger->dev, reg);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -94,22 +99,22 @@ static int bq25700_write(struct bq25700 *charger, uint reg, u16 val)
|
||||
}
|
||||
|
||||
static const union {
|
||||
struct bq25700_range rt;
|
||||
struct bq25700_range rt;
|
||||
} bq25700_tables[] = {
|
||||
/* range tables */
|
||||
[TBL_ICHG] = { .rt = {0, 8128000, 64000} },
|
||||
[TBL_ICHG] = {.rt = {0, 8128000, 64000}},
|
||||
/* uV */
|
||||
[TBL_CHGMAX] = { .rt = {0, 19200000, 16000} },
|
||||
[TBL_CHGMAX] = {.rt = {0, 19200000, 16000}},
|
||||
/* uV max charge voltage*/
|
||||
[TBL_INPUTVOL] = { .rt = {3200000, 19520000, 64000} },
|
||||
[TBL_INPUTVOL] = {.rt = {3200000, 19520000, 64000}},
|
||||
/* uV input charge voltage*/
|
||||
[TBL_INPUTCUR] = {.rt = {0, 6350000, 50000} },
|
||||
[TBL_INPUTCUR] = {.rt = {0, 6350000, 50000}},
|
||||
/*uA input current*/
|
||||
[TBL_SYSVMIN] = { .rt = {1024000, 16182000, 256000} },
|
||||
[TBL_SYSVMIN] = {.rt = {1024000, 16182000, 256000}},
|
||||
/* uV min system voltage*/
|
||||
[TBL_OTGVOL] = {.rt = {4480000, 20800000, 64000} },
|
||||
[TBL_OTGVOL] = {.rt = {4480000, 20800000, 64000}},
|
||||
/*uV OTG volage*/
|
||||
[TBL_OTGCUR] = {.rt = {0, 6350000, 50000} },
|
||||
[TBL_OTGCUR] = {.rt = {0, 6350000, 50000}},
|
||||
};
|
||||
|
||||
static u32 bq25700_find_idx(u32 value, enum bq25700_table_ids id)
|
||||
@@ -121,8 +126,8 @@ static u32 bq25700_find_idx(u32 value, enum bq25700_table_ids id)
|
||||
rtbl_size = (rtbl->max - rtbl->min) / rtbl->step + 1;
|
||||
|
||||
for (idx = 1;
|
||||
idx < rtbl_size && (idx * rtbl->step + rtbl->min <= value);
|
||||
idx++)
|
||||
idx < rtbl_size && (idx * rtbl->step + rtbl->min <= value);
|
||||
idx++)
|
||||
;
|
||||
|
||||
return idx - 1;
|
||||
@@ -174,26 +179,84 @@ static int bq25700_get_usb_type(void)
|
||||
#endif
|
||||
}
|
||||
|
||||
static int bq25700_get_pd_output_val(struct bq25700 *charger,
|
||||
int *vol, int *cur)
|
||||
{
|
||||
struct power_delivery_data pd_data;
|
||||
int ret;
|
||||
|
||||
if (!charger->pd[0] && !charger->pd[1]) {
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
memset(&pd_data, 0, sizeof(pd_data));
|
||||
int i = 0;
|
||||
for (i = 0; i < PD_MUN; i++) {
|
||||
if (!charger->pd[i]) {
|
||||
continue;
|
||||
}
|
||||
ret = power_delivery_get_data(charger->pd[i], &pd_data);
|
||||
if (ret) {
|
||||
continue;
|
||||
}
|
||||
if (!pd_data.online || !pd_data.voltage || !pd_data.current) {
|
||||
continue;
|
||||
}
|
||||
|
||||
*vol = pd_data.voltage;
|
||||
*cur = pd_data.current;
|
||||
printf("voltage is %d current is %d\n", *vol, *cur);
|
||||
goto end;
|
||||
}
|
||||
return -EINVAL;
|
||||
|
||||
end:
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bq25703_charger_current_init(struct bq25700 *charger)
|
||||
{
|
||||
u16 vol_idx = 0, cur_idx, chr_idx;
|
||||
u16 charge_current = BQ25700_CHARGE_CURRENT_1500MA;
|
||||
u16 sdp_inputcurrent = BQ25700_SDP_INPUT_CURRENT_500MA;
|
||||
u16 dcp_inputcurrent = BQ25700_DCP_INPUT_CURRENT_1500MA;
|
||||
int pd_inputvol, pd_inputcurrent;
|
||||
u16 vol_idx = 0, cur_idx;
|
||||
u16 temp;
|
||||
|
||||
temp = bq25700_read(charger, BQ25703_CHARGEOPTION0_REG);
|
||||
temp &= (~WATCHDOG_ENSABLE);
|
||||
bq25700_write(charger, BQ25703_CHARGEOPTION0_REG, temp);
|
||||
|
||||
vol_idx = bq25700_find_idx((5000000 - 1280000 - 3200000), TBL_INPUTVOL);
|
||||
vol_idx = vol_idx << 6;
|
||||
cur_idx = bq25700_find_idx(2500000, TBL_INPUTCUR);
|
||||
cur_idx = cur_idx << 8;
|
||||
chr_idx = bq25700_find_idx(2500000, TBL_ICHG);
|
||||
chr_idx = chr_idx << 6;
|
||||
|
||||
bq25700_write(charger, BQ25703_INPUTCURREN_REG, cur_idx); //0x0E
|
||||
bq25700_write(charger, BQ25703_INPUTVOLTAGE_REG, vol_idx); //0x0A
|
||||
bq25700_write(charger, BQ25703_CHARGECURREN_REG, chr_idx); //0x02
|
||||
if (!bq25700_get_pd_output_val(charger, &pd_inputvol,
|
||||
&pd_inputcurrent)) {
|
||||
if (pd_inputvol > 5000000) {
|
||||
vol_idx = bq25700_find_idx(pd_inputvol - 1280000 - 3200000,
|
||||
TBL_INPUTVOL);
|
||||
vol_idx = vol_idx << 6;
|
||||
}
|
||||
cur_idx = bq25700_find_idx(pd_inputcurrent,
|
||||
TBL_INPUTCUR);
|
||||
cur_idx = cur_idx << 8;
|
||||
if (pd_inputcurrent != 0)
|
||||
{
|
||||
bq25700_write(charger, BQ25703_INPUTCURREN_REG,
|
||||
cur_idx);
|
||||
if (vol_idx)
|
||||
bq25700_write(charger, BQ25703_INPUTVOLTAGE_REG,
|
||||
vol_idx);
|
||||
charge_current = bq25700_find_idx(pd_inputcurrent,
|
||||
TBL_ICHG);
|
||||
charge_current = charge_current << 6;
|
||||
}
|
||||
} else {
|
||||
bq25700_write(charger, BQ25703_INPUTCURREN_REG,
|
||||
dcp_inputcurrent);
|
||||
}
|
||||
|
||||
if (bq25703_charger_status(charger)) {
|
||||
bq25700_write(charger, BQ25703_CHARGECURREN_REG,
|
||||
charge_current);
|
||||
}
|
||||
}
|
||||
|
||||
static int bq25700_ofdata_to_platdata(struct udevice *dev)
|
||||
@@ -218,8 +281,6 @@ static int bq25700_ofdata_to_platdata(struct udevice *dev)
|
||||
charger->chip_id = BQ25700_ID;
|
||||
}
|
||||
|
||||
charger->ichg = fdtdec_get_int(blob, node, "ti,charge-current", 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -228,15 +289,32 @@ static int bq25700_probe(struct udevice *dev)
|
||||
struct bq25700 *charger = dev_get_priv(dev);
|
||||
int ret;
|
||||
|
||||
if (charger->chip_id == BQ25703_ID)
|
||||
struct udevice *pd_tmp;
|
||||
struct udevice *dev_tmp;
|
||||
|
||||
for (uclass_first_device(UCLASS_PD, &pd_tmp);
|
||||
pd_tmp;
|
||||
uclass_next_device(&pd_tmp))
|
||||
{
|
||||
dev_tmp = dev_get_parent(pd_tmp);
|
||||
if (!strncmp(TYPEC0_I2C, dev_tmp->name, strlen(TYPEC0_I2C))) { // Ensure that typec0 has the highest priority
|
||||
charger->pd[0] = pd_tmp;
|
||||
} else if (!strncmp(TYPEC1_I2C, dev_tmp->name, strlen(TYPEC1_I2C))) {
|
||||
charger->pd[1] = pd_tmp;
|
||||
}
|
||||
}
|
||||
|
||||
if (charger->chip_id == BQ25703_ID) {
|
||||
bq25703_charger_current_init(charger);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct udevice_id charger_ids[] = {
|
||||
{ .compatible = "ti,bq25700" },
|
||||
{ .compatible = "ti,bq25703" },
|
||||
{.compatible = "ti,bq25700"},
|
||||
{.compatible = "ti,bq25703"},
|
||||
{},
|
||||
{ },
|
||||
};
|
||||
|
||||
|
||||
40
drivers/power/power_delivery/Kconfig
Executable file
40
drivers/power/power_delivery/Kconfig
Executable file
@@ -0,0 +1,40 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
config DM_POWER_DELIVERY
|
||||
bool "Enable driver model power delivery support"
|
||||
depends on DM
|
||||
help
|
||||
This adds a simple uclass for power delivery.
|
||||
|
||||
config TYPEC_TCPM
|
||||
tristate "USB Type-C Port Controller Manager"
|
||||
depends on DM && DM_POWER_DELIVERY
|
||||
help
|
||||
The Type-C Port Controller Manager provides a USB PD and USB Type-C
|
||||
state machine for use with Type-C Port Controllers.
|
||||
|
||||
config TYPEC_TCPCI
|
||||
tristate "Type-C Port Controller Interface driver"
|
||||
depends on DM && DM_POWER_DELIVERY && DM_I2C
|
||||
help
|
||||
Type-C Port Controller driver for TCPCI-compliant controller.
|
||||
|
||||
if TYPEC_TCPCI
|
||||
|
||||
config TYPEC_HUSB311
|
||||
tristate "Hynetek HUSB311 Type-C chip driver"
|
||||
depends on DM && DM_POWER_DELIVERY && DM_I2C
|
||||
help
|
||||
Hynetek HUSB311 Type-C chip driver that works with
|
||||
Type-C Port Controller Manager to provide USB PD and USB
|
||||
Type-C functionalities.
|
||||
|
||||
endif # TYPEC_TCPCI
|
||||
|
||||
config TYPEC_FUSB302
|
||||
tristate "Fairchild FUSB302 Type-C chip driver"
|
||||
depends on DM && DM_POWER_DELIVERY && DM_I2C
|
||||
help
|
||||
The Fairchild FUSB302 Type-C chip driver that works with
|
||||
Type-C Port Controller Manager to provide USB PD and USB
|
||||
Type-C functionalities.
|
||||
7
drivers/power/power_delivery/Makefile
Executable file
7
drivers/power/power_delivery/Makefile
Executable file
@@ -0,0 +1,7 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
obj-$(CONFIG_$(SPL_)DM_POWER_DELIVERY) += power_delivery_uclass.o
|
||||
obj-$(CONFIG_TYPEC_TCPM) += tcpm.o
|
||||
obj-$(CONFIG_TYPEC_FUSB302) += fusb302.o
|
||||
obj-$(CONFIG_TYPEC_TCPCI) += tcpci.o
|
||||
obj-$(CONFIG_TYPEC_HUSB311) += tcpci_husb311.o
|
||||
31
drivers/power/power_delivery/power_delivery_uclass.c
Executable file
31
drivers/power/power_delivery/power_delivery_uclass.c
Executable file
@@ -0,0 +1,31 @@
|
||||
/*
|
||||
* (C) Copyright 2022 Rockchip Electronics Co., Ltd
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-2.0+
|
||||
*/
|
||||
|
||||
#include <errno.h>
|
||||
#include <dm.h>
|
||||
#include <dm/device-internal.h>
|
||||
#include <power/power_delivery/power_delivery.h>
|
||||
|
||||
DECLARE_GLOBAL_DATA_PTR;
|
||||
|
||||
int power_delivery_get_data(struct udevice *dev, struct power_delivery_data *pd_data)
|
||||
{
|
||||
const struct dm_power_delivery_ops *ops = dev_get_driver_ops(dev);
|
||||
|
||||
if (!ops || !ops->get_current || !ops->get_voltage || !ops->get_online)
|
||||
return -ENOSYS;
|
||||
|
||||
pd_data->voltage = ops->get_voltage(dev);
|
||||
pd_data->current = ops->get_current(dev);
|
||||
pd_data->online = ops->get_online(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
UCLASS_DRIVER(power_delivery) = {
|
||||
.id = UCLASS_PD,
|
||||
.name = "power_delivery",
|
||||
};
|
||||
778
drivers/power/power_delivery/tcpci.c
Executable file
778
drivers/power/power_delivery/tcpci.c
Executable file
@@ -0,0 +1,778 @@
|
||||
// SPDX-License-Identifier: GPL-2.0+
|
||||
/*
|
||||
* Copyright 2015-2017 Google, Inc
|
||||
*
|
||||
* USB Type-C Port Controller Interface.
|
||||
*/
|
||||
|
||||
#include <dm.h>
|
||||
#include <i2c.h>
|
||||
#include <asm/gpio.h>
|
||||
#include <power/power_delivery/pd.h>
|
||||
#include <power/power_delivery/tcpm.h>
|
||||
#include <power/power_delivery/typec.h>
|
||||
#include <power/power_delivery/power_delivery.h>
|
||||
|
||||
#include "tcpci.h"
|
||||
|
||||
#define PD_RETRY_COUNT 3
|
||||
|
||||
#define tcpc_presenting_cc1_rd(reg) \
|
||||
(!(TCPC_ROLE_CTRL_DRP & (reg)) && \
|
||||
(((reg) & (TCPC_ROLE_CTRL_CC1_MASK << TCPC_ROLE_CTRL_CC1_SHIFT)) == \
|
||||
(TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC1_SHIFT)))
|
||||
#define tcpc_presenting_cc2_rd(reg) \
|
||||
(!(TCPC_ROLE_CTRL_DRP & (reg)) && \
|
||||
(((reg) & (TCPC_ROLE_CTRL_CC2_MASK << TCPC_ROLE_CTRL_CC2_SHIFT)) == \
|
||||
(TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC2_SHIFT)))
|
||||
|
||||
struct tcpci {
|
||||
struct udevice *dev;
|
||||
|
||||
struct tcpm_port *port;
|
||||
|
||||
bool controls_vbus;
|
||||
bool gpio_cc_int_present;
|
||||
|
||||
struct tcpc_dev tcpc;
|
||||
struct tcpci_data *data;
|
||||
struct gpio_desc gpio_cc_int;
|
||||
};
|
||||
|
||||
struct tcpci_chip {
|
||||
struct udevice *udev;
|
||||
struct tcpci *tcpci;
|
||||
struct tcpci_data data;
|
||||
};
|
||||
|
||||
static inline struct tcpci *tcpc_to_tcpci(struct tcpc_dev *tcpc)
|
||||
{
|
||||
return container_of(tcpc, struct tcpci, tcpc);
|
||||
}
|
||||
|
||||
static int tcpci_read16(struct tcpci *tcpci, unsigned int reg, u16 *val)
|
||||
{
|
||||
int ret = 0;
|
||||
u8 buffer[2];
|
||||
|
||||
ret = dm_i2c_read(tcpci->dev, reg, buffer, 2);
|
||||
if (ret) {
|
||||
printf("%s: cannot read %02x, ret=%d\n",
|
||||
__func__, reg, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
*val = ((buffer[1] << 8) & 0xFF00) | (buffer[0] & 0xFF);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int tcpci_block_read(struct tcpci *tcpci, unsigned int reg,
|
||||
u8 *data, u8 length)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
ret = dm_i2c_read(tcpci->dev, reg, data, length);
|
||||
if (ret)
|
||||
printf("%s: cannot block read 0x%02x, len=%d, ret=%d\n",
|
||||
__func__, reg, length, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int tcpci_write16(struct tcpci *tcpci, unsigned int reg, u16 val)
|
||||
{
|
||||
int ret = 0;
|
||||
u8 buffer[2];
|
||||
|
||||
buffer[0] = val & 0xFF;
|
||||
buffer[1] = (val >> 8) & 0xFF;
|
||||
ret = dm_i2c_write(tcpci->dev, reg, buffer, 2);
|
||||
if (ret)
|
||||
printf("%s: cannot write 0x%02x, ret=%d\n",
|
||||
__func__, reg, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int tcpci_block_write(struct tcpci *tcpci, unsigned int reg,
|
||||
u8 *data, u8 length)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
ret = dm_i2c_write(tcpci->dev, reg, data, length);
|
||||
if (ret)
|
||||
printf("%s: cannot block write 0x%02x, len=%d, ret=%d\n",
|
||||
__func__, reg, length, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int tcpci_set_cc(struct tcpc_dev *tcpc, enum typec_cc_status cc)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg;
|
||||
int ret;
|
||||
|
||||
switch (cc) {
|
||||
case TYPEC_CC_RA:
|
||||
reg = (TCPC_ROLE_CTRL_CC_RA << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_RA << TCPC_ROLE_CTRL_CC2_SHIFT);
|
||||
break;
|
||||
case TYPEC_CC_RD:
|
||||
reg = (TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC2_SHIFT);
|
||||
break;
|
||||
case TYPEC_CC_RP_DEF:
|
||||
reg = (TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC2_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_RP_VAL_DEF <<
|
||||
TCPC_ROLE_CTRL_RP_VAL_SHIFT);
|
||||
break;
|
||||
case TYPEC_CC_RP_1_5:
|
||||
reg = (TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC2_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_RP_VAL_1_5 <<
|
||||
TCPC_ROLE_CTRL_RP_VAL_SHIFT);
|
||||
break;
|
||||
case TYPEC_CC_RP_3_0:
|
||||
reg = (TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC2_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_RP_VAL_3_0 <<
|
||||
TCPC_ROLE_CTRL_RP_VAL_SHIFT);
|
||||
break;
|
||||
case TYPEC_CC_OPEN:
|
||||
default:
|
||||
reg = (TCPC_ROLE_CTRL_CC_OPEN << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_OPEN << TCPC_ROLE_CTRL_CC2_SHIFT);
|
||||
break;
|
||||
}
|
||||
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_ROLE_CTRL, reg);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_start_toggling(struct tcpc_dev *tcpc,
|
||||
enum typec_port_type port_type,
|
||||
enum typec_cc_status cc)
|
||||
{
|
||||
int ret;
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg = TCPC_ROLE_CTRL_DRP;
|
||||
|
||||
if (port_type != TYPEC_PORT_DRP)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
/* Handle vendor drp toggling */
|
||||
if (tcpci->data->start_drp_toggling) {
|
||||
ret = tcpci->data->start_drp_toggling(tcpci, tcpci->data, cc);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
switch (cc) {
|
||||
default:
|
||||
case TYPEC_CC_RP_DEF:
|
||||
reg |= (TCPC_ROLE_CTRL_RP_VAL_DEF <<
|
||||
TCPC_ROLE_CTRL_RP_VAL_SHIFT);
|
||||
break;
|
||||
case TYPEC_CC_RP_1_5:
|
||||
reg |= (TCPC_ROLE_CTRL_RP_VAL_1_5 <<
|
||||
TCPC_ROLE_CTRL_RP_VAL_SHIFT);
|
||||
break;
|
||||
case TYPEC_CC_RP_3_0:
|
||||
reg |= (TCPC_ROLE_CTRL_RP_VAL_3_0 <<
|
||||
TCPC_ROLE_CTRL_RP_VAL_SHIFT);
|
||||
break;
|
||||
}
|
||||
|
||||
if (cc == TYPEC_CC_RD)
|
||||
reg |= (TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC2_SHIFT);
|
||||
else
|
||||
reg |= (TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC1_SHIFT) |
|
||||
(TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC2_SHIFT);
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_ROLE_CTRL, reg);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
return dm_i2c_reg_write(tcpci->dev, TCPC_COMMAND,
|
||||
TCPC_CMD_LOOK4CONNECTION);
|
||||
}
|
||||
|
||||
static enum typec_cc_status tcpci_to_typec_cc(unsigned int cc, bool sink)
|
||||
{
|
||||
switch (cc) {
|
||||
case 0x1:
|
||||
return sink ? TYPEC_CC_RP_DEF : TYPEC_CC_RA;
|
||||
case 0x2:
|
||||
return sink ? TYPEC_CC_RP_1_5 : TYPEC_CC_RD;
|
||||
case 0x3:
|
||||
if (sink)
|
||||
return TYPEC_CC_RP_3_0;
|
||||
/* fall through */
|
||||
case 0x0:
|
||||
default:
|
||||
return TYPEC_CC_OPEN;
|
||||
}
|
||||
}
|
||||
|
||||
static int tcpci_get_cc(struct tcpc_dev *tcpc,
|
||||
enum typec_cc_status *cc1, enum typec_cc_status *cc2)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg, role_control;
|
||||
|
||||
role_control = dm_i2c_reg_read(tcpci->dev, TCPC_ROLE_CTRL);
|
||||
if (role_control < 0)
|
||||
return role_control;
|
||||
|
||||
reg = dm_i2c_reg_read(tcpci->dev, TCPC_CC_STATUS);
|
||||
if (reg < 0)
|
||||
return reg;
|
||||
|
||||
*cc1 = tcpci_to_typec_cc((reg >> TCPC_CC_STATUS_CC1_SHIFT) &
|
||||
TCPC_CC_STATUS_CC1_MASK,
|
||||
reg & TCPC_CC_STATUS_TERM ||
|
||||
tcpc_presenting_cc1_rd(role_control));
|
||||
*cc2 = tcpci_to_typec_cc((reg >> TCPC_CC_STATUS_CC2_SHIFT) &
|
||||
TCPC_CC_STATUS_CC2_MASK,
|
||||
reg & TCPC_CC_STATUS_TERM ||
|
||||
tcpc_presenting_cc2_rd(role_control));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_set_polarity(struct tcpc_dev *tcpc,
|
||||
enum typec_cc_polarity polarity)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg;
|
||||
int ret;
|
||||
enum typec_cc_status cc1, cc2;
|
||||
|
||||
/* Obtain Rp setting from role control */
|
||||
reg = dm_i2c_reg_read(tcpci->dev, TCPC_ROLE_CTRL);
|
||||
if (reg < 0)
|
||||
return reg;
|
||||
|
||||
ret = tcpci_get_cc(tcpc, &cc1, &cc2);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
/*
|
||||
* When port has drp toggling enabled, ROLE_CONTROL would only have the initial
|
||||
* terminations for the toggling and does not indicate the final cc
|
||||
* terminations when ConnectionResult is 0 i.e. drp toggling stops and
|
||||
* the connection is resolbed. Infer port role from TCPC_CC_STATUS based on the
|
||||
* terminations seen. The port role is then used to set the cc terminations.
|
||||
*/
|
||||
if (reg & TCPC_ROLE_CTRL_DRP) {
|
||||
/* Disable DRP for the OPEN setting to take effect */
|
||||
reg = reg & ~TCPC_ROLE_CTRL_DRP;
|
||||
|
||||
if (polarity == TYPEC_POLARITY_CC2) {
|
||||
reg &= ~(TCPC_ROLE_CTRL_CC2_MASK << TCPC_ROLE_CTRL_CC2_SHIFT);
|
||||
/* Local port is source */
|
||||
if (cc2 == TYPEC_CC_RD)
|
||||
/* Role control would have the Rp setting when DRP was enabled */
|
||||
reg |= TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC2_SHIFT;
|
||||
else
|
||||
reg |= TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC2_SHIFT;
|
||||
} else {
|
||||
reg &= ~(TCPC_ROLE_CTRL_CC1_MASK << TCPC_ROLE_CTRL_CC1_SHIFT);
|
||||
/* Local port is source */
|
||||
if (cc1 == TYPEC_CC_RD)
|
||||
/* Role control would have the Rp setting when DRP was enabled */
|
||||
reg |= TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC1_SHIFT;
|
||||
else
|
||||
reg |= TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC1_SHIFT;
|
||||
}
|
||||
}
|
||||
|
||||
if (polarity == TYPEC_POLARITY_CC2)
|
||||
reg |= TCPC_ROLE_CTRL_CC_OPEN << TCPC_ROLE_CTRL_CC1_SHIFT;
|
||||
else
|
||||
reg |= TCPC_ROLE_CTRL_CC_OPEN << TCPC_ROLE_CTRL_CC2_SHIFT;
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_ROLE_CTRL, reg);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return dm_i2c_reg_write(tcpci->dev, TCPC_TCPC_CTRL,
|
||||
(polarity == TYPEC_POLARITY_CC2) ?
|
||||
TCPC_TCPC_CTRL_ORIENTATION : 0);
|
||||
}
|
||||
|
||||
static int tcpci_set_vconn(struct tcpc_dev *tcpc, bool enable)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
int ret;
|
||||
unsigned int reg;
|
||||
|
||||
/* Handle vendor set vconn */
|
||||
if (tcpci->data->set_vconn) {
|
||||
ret = tcpci->data->set_vconn(tcpci, tcpci->data, enable);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
reg = dm_i2c_reg_read(tcpci->dev, TCPC_POWER_CTRL);
|
||||
if (reg)
|
||||
return reg;
|
||||
reg &= ~TCPC_POWER_CTRL_VCONN_ENABLE;
|
||||
reg |= enable ? TCPC_POWER_CTRL_VCONN_ENABLE : 0;
|
||||
return dm_i2c_reg_write(tcpci->dev, TCPC_POWER_CTRL, reg);
|
||||
}
|
||||
|
||||
static int tcpci_set_roles(struct tcpc_dev *tcpc, bool attached,
|
||||
enum typec_role role, enum typec_data_role data)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg;
|
||||
int ret;
|
||||
|
||||
reg = PD_REV20 << TCPC_MSG_HDR_INFO_REV_SHIFT;
|
||||
if (role == TYPEC_SOURCE)
|
||||
reg |= TCPC_MSG_HDR_INFO_PWR_ROLE;
|
||||
if (data == TYPEC_HOST)
|
||||
reg |= TCPC_MSG_HDR_INFO_DATA_ROLE;
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_MSG_HDR_INFO, reg);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_set_pd_rx(struct tcpc_dev *tcpc, bool enable)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg = 0;
|
||||
int ret;
|
||||
|
||||
if (enable)
|
||||
reg = TCPC_RX_DETECT_SOP | TCPC_RX_DETECT_HARD_RESET;
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_RX_DETECT, reg);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_get_vbus(struct tcpc_dev *tcpc)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg;
|
||||
|
||||
reg = dm_i2c_reg_read(tcpci->dev, TCPC_POWER_STATUS);
|
||||
if (reg < 0)
|
||||
return reg;
|
||||
|
||||
return !!(reg & TCPC_POWER_STATUS_VBUS_PRES);
|
||||
}
|
||||
|
||||
static int tcpci_set_vbus(struct tcpc_dev *tcpc, bool source, bool sink)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
int ret;
|
||||
|
||||
/* Disable both source and sink first before enabling anything */
|
||||
|
||||
if (!source) {
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_COMMAND,
|
||||
TCPC_CMD_DISABLE_SRC_VBUS);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (!sink) {
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_COMMAND,
|
||||
TCPC_CMD_DISABLE_SINK_VBUS);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (source) {
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_COMMAND,
|
||||
TCPC_CMD_SRC_VBUS_DEFAULT);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (sink) {
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_COMMAND,
|
||||
TCPC_CMD_SINK_VBUS);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_pd_transmit(struct tcpc_dev *tcpc,
|
||||
enum tcpm_transmit_type type,
|
||||
const struct pd_message *msg,
|
||||
unsigned int negotiated_rev)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
u16 header = msg ? le16_to_cpu(msg->header) : 0;
|
||||
unsigned int reg, cnt;
|
||||
int ret;
|
||||
|
||||
cnt = msg ? pd_header_cnt(header) * 4 : 0;
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_TX_BYTE_CNT, cnt + 2);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = tcpci_write16(tcpci, TCPC_TX_HDR, header);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
if (cnt > 0) {
|
||||
ret = tcpci_block_write(tcpci, TCPC_TX_DATA,
|
||||
(u8 *)&msg->payload, cnt);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
reg = (PD_RETRY_COUNT << TCPC_TRANSMIT_RETRY_SHIFT) |
|
||||
(type << TCPC_TRANSMIT_TYPE_SHIFT);
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_TRANSMIT, reg);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_init(struct tcpc_dev *tcpc)
|
||||
{
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int timeout = 0; /* XXX */
|
||||
unsigned int reg;
|
||||
int ret;
|
||||
|
||||
while (timeout < 100) {
|
||||
reg = dm_i2c_reg_read(tcpci->dev, TCPC_POWER_STATUS);
|
||||
if (reg < 0)
|
||||
return reg;
|
||||
if (!(reg & TCPC_POWER_STATUS_UNINIT))
|
||||
break;
|
||||
timeout++;
|
||||
udelay(200);
|
||||
}
|
||||
if (timeout >= 100)
|
||||
return -ETIMEDOUT;
|
||||
|
||||
/* Handle vendor init */
|
||||
if (tcpci->data->init) {
|
||||
ret = tcpci->data->init(tcpci, tcpci->data);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Clear all events */
|
||||
ret = tcpci_write16(tcpci, TCPC_ALERT, 0xffff);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
if (tcpci->controls_vbus)
|
||||
reg = TCPC_POWER_STATUS_VBUS_PRES;
|
||||
else
|
||||
reg = 0;
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_POWER_STATUS_MASK, reg);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
/* Enable Vbus detection */
|
||||
ret = dm_i2c_reg_write(tcpci->dev, TCPC_COMMAND,
|
||||
TCPC_CMD_ENABLE_VBUS_DETECT);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
reg = TCPC_ALERT_TX_SUCCESS | TCPC_ALERT_TX_FAILED |
|
||||
TCPC_ALERT_TX_DISCARDED | TCPC_ALERT_RX_STATUS |
|
||||
TCPC_ALERT_RX_HARD_RST | TCPC_ALERT_CC_STATUS;
|
||||
if (tcpci->controls_vbus)
|
||||
reg |= TCPC_ALERT_POWER_STATUS;
|
||||
return tcpci_write16(tcpci, TCPC_ALERT_MASK, reg);
|
||||
}
|
||||
|
||||
static void tcpci_poll_event(struct tcpc_dev *tcpc)
|
||||
{
|
||||
u16 status;
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
|
||||
tcpci_read16(tcpci, TCPC_ALERT, &status);
|
||||
|
||||
/*
|
||||
* Clear alert status for everything except RX_STATUS, which shouldn't
|
||||
* be cleared until we have successfully retrieved message.
|
||||
*/
|
||||
if (status & ~TCPC_ALERT_RX_STATUS)
|
||||
tcpci_write16(tcpci, TCPC_ALERT,
|
||||
status & ~TCPC_ALERT_RX_STATUS);
|
||||
|
||||
if (status & TCPC_ALERT_CC_STATUS)
|
||||
tcpm_cc_change(tcpci->port);
|
||||
|
||||
if (status & TCPC_ALERT_POWER_STATUS) {
|
||||
unsigned int reg;
|
||||
|
||||
reg = dm_i2c_reg_read(tcpci->dev, TCPC_POWER_STATUS_MASK);
|
||||
if (reg < 0)
|
||||
return;
|
||||
|
||||
/*
|
||||
* If power status mask has been reset, then the TCPC
|
||||
* has reset.
|
||||
*/
|
||||
if (reg == 0xff)
|
||||
tcpm_tcpc_reset(tcpci->port);
|
||||
else
|
||||
tcpm_vbus_change(tcpci->port);
|
||||
}
|
||||
|
||||
if (status & TCPC_ALERT_RX_STATUS) {
|
||||
struct pd_message msg;
|
||||
unsigned int cnt, payload_cnt;
|
||||
u16 header;
|
||||
|
||||
cnt = dm_i2c_reg_read(tcpci->dev, TCPC_RX_BYTE_CNT);
|
||||
if (cnt < 0)
|
||||
return;
|
||||
/*
|
||||
* 'cnt' corresponds to READABLE_BYTE_COUNT in section 4.4.14
|
||||
* of the TCPCI spec [Rev 2.0 Ver 1.0 October 2017] and is
|
||||
* defined in table 4-36 as one greater than the number of
|
||||
* bytes received. And that number includes the header. So:
|
||||
*/
|
||||
if (cnt > 3)
|
||||
payload_cnt = cnt - (1 + sizeof(msg.header));
|
||||
else
|
||||
payload_cnt = 0;
|
||||
|
||||
tcpci_read16(tcpci, TCPC_RX_HDR, &header);
|
||||
msg.header = cpu_to_le16(header);
|
||||
|
||||
if ((payload_cnt > sizeof(msg.payload)))
|
||||
payload_cnt = sizeof(msg.payload);
|
||||
|
||||
if (payload_cnt > 0)
|
||||
tcpci_block_read(tcpci, TCPC_RX_DATA,
|
||||
(u8 *)&msg.payload, payload_cnt);
|
||||
|
||||
/* Read complete, clear RX status alert bit */
|
||||
tcpci_write16(tcpci, TCPC_ALERT, TCPC_ALERT_RX_STATUS);
|
||||
|
||||
tcpm_pd_receive(tcpci->port, &msg);
|
||||
}
|
||||
|
||||
if (status & TCPC_ALERT_RX_HARD_RST)
|
||||
tcpm_pd_hard_reset(tcpci->port);
|
||||
|
||||
if (status & TCPC_ALERT_TX_SUCCESS)
|
||||
tcpm_pd_transmit_complete(tcpci->port, TCPC_TX_SUCCESS);
|
||||
else if (status & TCPC_ALERT_TX_DISCARDED)
|
||||
tcpm_pd_transmit_complete(tcpci->port, TCPC_TX_DISCARDED);
|
||||
else if (status & TCPC_ALERT_TX_FAILED)
|
||||
tcpm_pd_transmit_complete(tcpci->port, TCPC_TX_FAILED);
|
||||
}
|
||||
|
||||
static int tcpci_enter_low_power_mode(struct tcpc_dev *tcpc,
|
||||
bool attached, bool pd_capable)
|
||||
{
|
||||
int ret;
|
||||
struct tcpci *tcpci = tcpc_to_tcpci(tcpc);
|
||||
unsigned int reg;
|
||||
|
||||
/* Disable chip interrupts before unregistering port */
|
||||
ret = tcpci_write16(tcpci, TCPC_ALERT_MASK, 0);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
reg = dm_i2c_reg_read(tcpci->dev, TCPC_BMCIO_CTRL);
|
||||
if (reg < 0)
|
||||
return reg;
|
||||
/*
|
||||
* For Type-C devices with PD capability, Only disable VBUS detect,
|
||||
* do not diable 24M oscillator for BMC communication. Otherwise,
|
||||
* data packets cannot be received.
|
||||
*/
|
||||
if (attached && pd_capable)
|
||||
reg &= ~TCPC_BMCIO_VBUS_DETECT_MASK;
|
||||
else
|
||||
reg &= ~(TCPC_BMCIO_VBUS_DETECT_MASK | TCPC_BMCIO_24M_OSC_MASK);
|
||||
return dm_i2c_reg_write(tcpci->dev, TCPC_BMCIO_CTRL, reg);
|
||||
}
|
||||
|
||||
static int tcpci_parse_config(struct tcpci *tcpci)
|
||||
{
|
||||
tcpci->controls_vbus = true; /* XXX */
|
||||
|
||||
tcpci->tcpc.connector_node = dev_read_subnode(tcpci->dev, "connector");
|
||||
if (!ofnode_valid(tcpci->tcpc.connector_node)) {
|
||||
printf("%s: 'connector' node is not found\n", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct tcpci *tcpci_register_port(struct udevice *dev, struct tcpci_data *data)
|
||||
{
|
||||
struct tcpci *tcpci;
|
||||
int err;
|
||||
|
||||
tcpci = devm_kzalloc(dev, sizeof(*tcpci), GFP_KERNEL);
|
||||
if (!tcpci)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
err = gpio_request_by_name(dev, "int-n-gpios", 0, &tcpci->gpio_cc_int, GPIOD_IS_IN);
|
||||
if (err) {
|
||||
printf("%s: fail to get int GPIO: err=%d\n", __func__, err);
|
||||
tcpci->gpio_cc_int_present = false;
|
||||
} else {
|
||||
printf("%s: success to get int GPIO: err=%d\n", __func__, err);
|
||||
tcpci->gpio_cc_int_present = true;
|
||||
}
|
||||
|
||||
tcpci->dev = dev;
|
||||
tcpci->data = data;
|
||||
|
||||
tcpci->tcpc.init = tcpci_init;
|
||||
tcpci->tcpc.get_vbus = tcpci_get_vbus;
|
||||
tcpci->tcpc.set_vbus = tcpci_set_vbus;
|
||||
tcpci->tcpc.set_cc = tcpci_set_cc;
|
||||
tcpci->tcpc.get_cc = tcpci_get_cc;
|
||||
tcpci->tcpc.set_polarity = tcpci_set_polarity;
|
||||
tcpci->tcpc.set_vconn = tcpci_set_vconn;
|
||||
tcpci->tcpc.start_toggling = tcpci_start_toggling;
|
||||
|
||||
tcpci->tcpc.set_pd_rx = tcpci_set_pd_rx;
|
||||
tcpci->tcpc.set_roles = tcpci_set_roles;
|
||||
tcpci->tcpc.pd_transmit = tcpci_pd_transmit;
|
||||
tcpci->tcpc.poll_event = tcpci_poll_event;
|
||||
tcpci->tcpc.enter_low_power_mode = tcpci_enter_low_power_mode;
|
||||
|
||||
err = tcpci_parse_config(tcpci);
|
||||
if (err < 0)
|
||||
return ERR_PTR(err);
|
||||
|
||||
tcpci->port = tcpm_port_init(tcpci->dev, &tcpci->tcpc);
|
||||
if (IS_ERR(tcpci->port)) {
|
||||
printf("%s: failed to tcpm port init\n", __func__);
|
||||
return ERR_CAST(tcpci->port);
|
||||
}
|
||||
|
||||
// tcpm_tcpc_reset(tcpci->port);
|
||||
|
||||
tcpm_poll_event(tcpci->port);
|
||||
|
||||
return tcpci;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(tcpci_register_port);
|
||||
|
||||
void tcpci_unregister_port(struct tcpci *tcpci)
|
||||
{
|
||||
tcpm_uninit_port(tcpci->port);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(tcpci_unregister_port);
|
||||
|
||||
int tcpci_get_voltage_fun(struct tcpci *tcpci)
|
||||
{
|
||||
return tcpm_get_voltage(tcpci->port);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(tcpci_get_voltage_fun);
|
||||
|
||||
int tcpci_get_current_fun(struct tcpci *tcpci)
|
||||
{
|
||||
return tcpm_get_current(tcpci->port);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(tcpci_get_current_fun);
|
||||
|
||||
int tcpci_get_online_fun(struct tcpci *tcpci)
|
||||
{
|
||||
return tcpm_get_online(tcpci->port);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(tcpci_get_online_fun);
|
||||
|
||||
static int tcpci_probe(struct udevice *dev)
|
||||
{
|
||||
struct tcpci_chip *chip = dev_get_priv(dev);
|
||||
int err;
|
||||
u16 val = 0;
|
||||
|
||||
chip->udev = dev;
|
||||
|
||||
/* Disable chip interrupts before requesting irq */
|
||||
err = tcpci_write16(chip->tcpci, TCPC_ALERT_MASK, val);
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
chip->tcpci = tcpci_register_port(chip->udev, &chip->data);
|
||||
if (IS_ERR(chip->tcpci))
|
||||
return PTR_ERR(chip->tcpci);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_remove(struct udevice *dev)
|
||||
{
|
||||
struct tcpci_chip *chip = dev_get_priv(dev);
|
||||
int err;
|
||||
|
||||
/* Disable chip interrupts before unregistering port */
|
||||
err = tcpci_write16(chip->tcpci, TCPC_ALERT_MASK, 0);
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
tcpci_unregister_port(chip->tcpci);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tcpci_get_voltage(struct udevice *dev)
|
||||
{
|
||||
struct tcpci_chip *chip = dev_get_priv(dev);
|
||||
|
||||
return tcpm_get_voltage(chip->tcpci->port);
|
||||
}
|
||||
|
||||
static int tcpci_get_current(struct udevice *dev)
|
||||
{
|
||||
struct tcpci_chip *chip = dev_get_priv(dev);
|
||||
|
||||
return tcpm_get_current(chip->tcpci->port);
|
||||
}
|
||||
|
||||
static int tcpci_get_online(struct udevice *dev)
|
||||
{
|
||||
struct tcpci_chip *chip = dev_get_priv(dev);
|
||||
|
||||
return tcpm_get_online(chip->tcpci->port);
|
||||
}
|
||||
|
||||
static struct dm_power_delivery_ops tcpci_ops = {
|
||||
.get_voltage = tcpci_get_voltage,
|
||||
.get_current = tcpci_get_current,
|
||||
.get_online = tcpci_get_online,
|
||||
};
|
||||
|
||||
static const struct udevice_id tcpci_ids[] = {
|
||||
{ .compatible = "nxp,ptn5110", },
|
||||
{},
|
||||
};
|
||||
|
||||
U_BOOT_DRIVER(tcpci) = {
|
||||
.name = "tcpci",
|
||||
.id = UCLASS_PD,
|
||||
.of_match = tcpci_ids,
|
||||
.ops = &tcpci_ops,
|
||||
.probe = tcpci_probe,
|
||||
.remove = tcpci_remove,
|
||||
.priv_auto_alloc_size = sizeof(struct tcpci_chip),
|
||||
}
|
||||
|
||||
MODULE_DESCRIPTION("USB Type-C Port Controller Interface driver");
|
||||
MODULE_LICENSE("GPL");
|
||||
150
drivers/power/power_delivery/tcpci.h
Executable file
150
drivers/power/power_delivery/tcpci.h
Executable file
@@ -0,0 +1,150 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||
/*
|
||||
* Copyright 2015-2017 Google, Inc
|
||||
*
|
||||
* USB Type-C Port Controller Interface.
|
||||
*/
|
||||
|
||||
#ifndef __LINUX_USB_TCPCI_H
|
||||
#define __LINUX_USB_TCPCI_H
|
||||
|
||||
#define TCPC_VENDOR_ID 0x0
|
||||
#define TCPC_PRODUCT_ID 0x2
|
||||
#define TCPC_BCD_DEV 0x4
|
||||
#define TCPC_TC_REV 0x6
|
||||
#define TCPC_PD_REV 0x8
|
||||
#define TCPC_PD_INT_REV 0xa
|
||||
|
||||
#define TCPC_ALERT 0x10
|
||||
#define TCPC_ALERT_VBUS_DISCNCT BIT(11)
|
||||
#define TCPC_ALERT_RX_BUF_OVF BIT(10)
|
||||
#define TCPC_ALERT_FAULT BIT(9)
|
||||
#define TCPC_ALERT_V_ALARM_LO BIT(8)
|
||||
#define TCPC_ALERT_V_ALARM_HI BIT(7)
|
||||
#define TCPC_ALERT_TX_SUCCESS BIT(6)
|
||||
#define TCPC_ALERT_TX_DISCARDED BIT(5)
|
||||
#define TCPC_ALERT_TX_FAILED BIT(4)
|
||||
#define TCPC_ALERT_RX_HARD_RST BIT(3)
|
||||
#define TCPC_ALERT_RX_STATUS BIT(2)
|
||||
#define TCPC_ALERT_POWER_STATUS BIT(1)
|
||||
#define TCPC_ALERT_CC_STATUS BIT(0)
|
||||
|
||||
#define TCPC_ALERT_MASK 0x12
|
||||
#define TCPC_POWER_STATUS_MASK 0x14
|
||||
#define TCPC_FAULT_STATUS_MASK 0x15
|
||||
#define TCPC_CONFIG_STD_OUTPUT 0x18
|
||||
|
||||
#define TCPC_TCPC_CTRL 0x19
|
||||
#define TCPC_TCPC_CTRL_ORIENTATION BIT(0)
|
||||
|
||||
#define TCPC_ROLE_CTRL 0x1a
|
||||
#define TCPC_ROLE_CTRL_DRP BIT(6)
|
||||
#define TCPC_ROLE_CTRL_RP_VAL_SHIFT 4
|
||||
#define TCPC_ROLE_CTRL_RP_VAL_MASK 0x3
|
||||
#define TCPC_ROLE_CTRL_RP_VAL_DEF 0x0
|
||||
#define TCPC_ROLE_CTRL_RP_VAL_1_5 0x1
|
||||
#define TCPC_ROLE_CTRL_RP_VAL_3_0 0x2
|
||||
#define TCPC_ROLE_CTRL_CC2_SHIFT 2
|
||||
#define TCPC_ROLE_CTRL_CC2_MASK 0x3
|
||||
#define TCPC_ROLE_CTRL_CC1_SHIFT 0
|
||||
#define TCPC_ROLE_CTRL_CC1_MASK 0x3
|
||||
#define TCPC_ROLE_CTRL_CC_RA 0x0
|
||||
#define TCPC_ROLE_CTRL_CC_RP 0x1
|
||||
#define TCPC_ROLE_CTRL_CC_RD 0x2
|
||||
#define TCPC_ROLE_CTRL_CC_OPEN 0x3
|
||||
|
||||
#define TCPC_FAULT_CTRL 0x1b
|
||||
|
||||
#define TCPC_POWER_CTRL 0x1c
|
||||
#define TCPC_POWER_CTRL_VCONN_ENABLE BIT(0)
|
||||
|
||||
#define TCPC_CC_STATUS 0x1d
|
||||
#define TCPC_CC_STATUS_TOGGLING BIT(5)
|
||||
#define TCPC_CC_STATUS_TERM BIT(4)
|
||||
#define TCPC_CC_STATUS_CC2_SHIFT 2
|
||||
#define TCPC_CC_STATUS_CC2_MASK 0x3
|
||||
#define TCPC_CC_STATUS_CC1_SHIFT 0
|
||||
#define TCPC_CC_STATUS_CC1_MASK 0x3
|
||||
|
||||
#define TCPC_POWER_STATUS 0x1e
|
||||
#define TCPC_POWER_STATUS_UNINIT BIT(6)
|
||||
#define TCPC_POWER_STATUS_VBUS_DET BIT(3)
|
||||
#define TCPC_POWER_STATUS_VBUS_PRES BIT(2)
|
||||
|
||||
#define TCPC_FAULT_STATUS 0x1f
|
||||
|
||||
#define TCPC_COMMAND 0x23
|
||||
#define TCPC_CMD_WAKE_I2C 0x11
|
||||
#define TCPC_CMD_DISABLE_VBUS_DETECT 0x22
|
||||
#define TCPC_CMD_ENABLE_VBUS_DETECT 0x33
|
||||
#define TCPC_CMD_DISABLE_SINK_VBUS 0x44
|
||||
#define TCPC_CMD_SINK_VBUS 0x55
|
||||
#define TCPC_CMD_DISABLE_SRC_VBUS 0x66
|
||||
#define TCPC_CMD_SRC_VBUS_DEFAULT 0x77
|
||||
#define TCPC_CMD_SRC_VBUS_HIGH 0x88
|
||||
#define TCPC_CMD_LOOK4CONNECTION 0x99
|
||||
#define TCPC_CMD_RXONEMORE 0xAA
|
||||
#define TCPC_CMD_I2C_IDLE 0xFF
|
||||
|
||||
#define TCPC_DEV_CAP_1 0x24
|
||||
#define TCPC_DEV_CAP_2 0x26
|
||||
#define TCPC_STD_INPUT_CAP 0x28
|
||||
#define TCPC_STD_OUTPUT_CAP 0x29
|
||||
|
||||
#define TCPC_MSG_HDR_INFO 0x2e
|
||||
#define TCPC_MSG_HDR_INFO_DATA_ROLE BIT(3)
|
||||
#define TCPC_MSG_HDR_INFO_PWR_ROLE BIT(0)
|
||||
#define TCPC_MSG_HDR_INFO_REV_SHIFT 1
|
||||
#define TCPC_MSG_HDR_INFO_REV_MASK 0x3
|
||||
|
||||
#define TCPC_RX_DETECT 0x2f
|
||||
#define TCPC_RX_DETECT_HARD_RESET BIT(5)
|
||||
#define TCPC_RX_DETECT_SOP BIT(0)
|
||||
|
||||
#define TCPC_RX_BYTE_CNT 0x30
|
||||
#define TCPC_RX_BUF_FRAME_TYPE 0x31
|
||||
#define TCPC_RX_HDR 0x32
|
||||
#define TCPC_RX_DATA 0x34 /* through 0x4f */
|
||||
|
||||
#define TCPC_TRANSMIT 0x50
|
||||
#define TCPC_TRANSMIT_RETRY_SHIFT 4
|
||||
#define TCPC_TRANSMIT_RETRY_MASK 0x3
|
||||
#define TCPC_TRANSMIT_TYPE_SHIFT 0
|
||||
#define TCPC_TRANSMIT_TYPE_MASK 0x7
|
||||
|
||||
#define TCPC_TX_BYTE_CNT 0x51
|
||||
#define TCPC_TX_HDR 0x52
|
||||
#define TCPC_TX_DATA 0x54 /* through 0x6f */
|
||||
|
||||
#define TCPC_VBUS_VOLTAGE 0x70
|
||||
#define TCPC_VBUS_SINK_DISCONNECT_THRESH 0x72
|
||||
#define TCPC_VBUS_STOP_DISCHARGE_THRESH 0x74
|
||||
#define TCPC_VBUS_VOLTAGE_ALARM_HI_CFG 0x76
|
||||
#define TCPC_VBUS_VOLTAGE_ALARM_LO_CFG 0x78
|
||||
|
||||
#define TCPC_BMCIO_CTRL 0x90
|
||||
#define TCPC_BMCIO_VBUS_DETECT_MASK BIT(1)
|
||||
#define TCPC_BMCIO_VBUS_DETECT_ENABLE BIT(1)
|
||||
#define TCPC_BMCIO_VBUS_DETECT_DISABLE 0
|
||||
#define TCPC_BMCIO_24M_OSC_MASK BIT(0)
|
||||
#define TCPC_BMCIO_ENABLE_24M_OSC BIT(0)
|
||||
#define TCPC_BMCIO_DISABLE_24M_OSC 0
|
||||
|
||||
struct tcpci;
|
||||
struct tcpci_data {
|
||||
struct regmap *regmap;
|
||||
int (*init)(struct tcpci *tcpci, struct tcpci_data *data);
|
||||
int (*set_vconn)(struct tcpci *tcpci, struct tcpci_data *data,
|
||||
bool enable);
|
||||
int (*start_drp_toggling)(struct tcpci *tcpci, struct tcpci_data *data,
|
||||
enum typec_cc_status cc);
|
||||
};
|
||||
|
||||
struct tcpci *tcpci_register_port(struct udevice *dev, struct tcpci_data *data);
|
||||
void tcpci_unregister_port(struct tcpci *tcpci);
|
||||
int tcpci_get_voltage_fun(struct tcpci *tcpci);
|
||||
int tcpci_get_current_fun(struct tcpci *tcpci);
|
||||
int tcpci_get_online_fun(struct tcpci *tcpci);
|
||||
irqreturn_t tcpci_irq(struct tcpci *tcpci);
|
||||
|
||||
#endif /* __LINUX_USB_TCPCI_H */
|
||||
229
drivers/power/power_delivery/tcpci_husb311.c
Executable file
229
drivers/power/power_delivery/tcpci_husb311.c
Executable file
@@ -0,0 +1,229 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
/*
|
||||
* Copyright (C) 2021 Rockchip Co.,Ltd.
|
||||
* Author: Wang Jie <dave.wang@rock-chips.com>
|
||||
*
|
||||
* Hynetek Husb311 Type-C Chip Driver
|
||||
*/
|
||||
|
||||
#include <dm.h>
|
||||
#include <i2c.h>
|
||||
#include <asm/gpio.h>
|
||||
#include <power/power_delivery/tcpm.h>
|
||||
#include <power/power_delivery/power_delivery.h>
|
||||
#include "tcpci.h"
|
||||
|
||||
#define HUSB311_VID 0x2E99
|
||||
#define HUSB311_PID 0x0311
|
||||
#define HUSB311_TCPC_I2C_RESET 0x9E
|
||||
#define HUSB311_TCPC_SOFTRESET 0xA0
|
||||
#define HUSB311_TCPC_FILTER 0xA1
|
||||
#define HUSB311_TCPC_TDRP 0xA2
|
||||
#define HUSB311_TCPC_DCSRCDRP 0xA3
|
||||
#define HUSB311_I2C_RETRY_MAX_CNT 3
|
||||
|
||||
struct husb311_chip {
|
||||
struct udevice *udev;
|
||||
struct tcpci_data data;
|
||||
struct tcpci *tcpci;
|
||||
};
|
||||
|
||||
static int husb311_read16(struct husb311_chip *chip, unsigned int reg)
|
||||
{
|
||||
int ret = 0;
|
||||
u8 buffer[2];
|
||||
|
||||
ret = dm_i2c_read(chip->udev, reg, buffer, 2);
|
||||
if (ret < 0) {
|
||||
printf("%s: cannot read %02x, ret=%d\n",
|
||||
__func__, reg, ret);
|
||||
return ret;
|
||||
}
|
||||
ret = ((buffer[1] << 8) & 0xFF00) + (buffer[0] & 0xFF);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int husb311_write8(struct husb311_chip *chip, unsigned int reg, u8 val)
|
||||
{
|
||||
int ret = 0;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < HUSB311_I2C_RETRY_MAX_CNT; i++) {
|
||||
ret = dm_i2c_write(chip->udev, reg, &val, 1);
|
||||
if (!ret)
|
||||
break;
|
||||
else
|
||||
udelay(200);
|
||||
}
|
||||
|
||||
if (ret)
|
||||
printf("%s: cannot write 0x%02x to 0x%02x, ret=%d\n",
|
||||
__func__, val, reg, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int husb311_write16(struct husb311_chip *chip, unsigned int reg, u16 val)
|
||||
{
|
||||
int ret = 0;
|
||||
u8 buffer[2];
|
||||
|
||||
buffer[0] = val & 0xFF;
|
||||
buffer[1] = (val >> 8) & 0xFF;
|
||||
ret = dm_i2c_write(chip->udev, reg, buffer, 2);
|
||||
if (ret)
|
||||
printf("%s: cannot write 0x%02x, len=%d, ret=%d\n",
|
||||
__func__, reg, 2, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static struct husb311_chip *tdata_to_husb311(struct tcpci_data *tdata)
|
||||
{
|
||||
return container_of(tdata, struct husb311_chip, data);
|
||||
}
|
||||
|
||||
static int husb311_sw_reset(struct husb311_chip *chip)
|
||||
{
|
||||
/* soft reset */
|
||||
return husb311_write8(chip, HUSB311_TCPC_SOFTRESET, 0x01);
|
||||
}
|
||||
|
||||
static int husb311_init(struct tcpci *tcpci, struct tcpci_data *tdata)
|
||||
{
|
||||
int ret;
|
||||
struct husb311_chip *chip = tdata_to_husb311(tdata);
|
||||
|
||||
/* I2C reset : (val + 1) * 12.5ms */
|
||||
ret = husb311_write8(chip, HUSB311_TCPC_I2C_RESET, 0x8F);
|
||||
/* tTCPCfilter : (26.7 * val) us */
|
||||
ret |= husb311_write8(chip, HUSB311_TCPC_FILTER, 0x0F);
|
||||
/* tDRP : (51.2 + 6.4 * val) ms */
|
||||
ret |= husb311_write8(chip, HUSB311_TCPC_TDRP, 0x04);
|
||||
/* dcSRC.DRP : 33% */
|
||||
ret |= husb311_write16(chip, HUSB311_TCPC_DCSRCDRP, 330);
|
||||
|
||||
if (ret)
|
||||
printf("%s: fail to init registers(%d)\n", __func__, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int husb311_check_revision(struct husb311_chip *chip)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = husb311_read16(chip, TCPC_VENDOR_ID);
|
||||
if (ret < 0) {
|
||||
printf("%s: fail to read Vendor id(%d)\n", __func__, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (ret != HUSB311_VID) {
|
||||
printf("%s: vid is not correct, 0x%04x\n", __func__, ret);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
ret = husb311_read16(chip, TCPC_PRODUCT_ID);
|
||||
if (ret < 0) {
|
||||
printf("%s: fail to read Product id(%d)\n", __func__, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (ret != HUSB311_PID) {
|
||||
printf("%s: pid is not correct, 0x%04x\n", __func__, ret);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int husb311_probe(struct udevice *dev)
|
||||
{
|
||||
int ret;
|
||||
struct husb311_chip *chip = dev_get_priv(dev);
|
||||
|
||||
chip->udev = dev;
|
||||
|
||||
ret = husb311_check_revision(chip);
|
||||
if (ret < 0) {
|
||||
printf("%s: check vid/pid fail(%d)\n", __func__, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = husb311_sw_reset(chip);
|
||||
if (ret) {
|
||||
printf("%s: fail to soft reset, ret = %d\n", __func__, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
chip->data.init = husb311_init;
|
||||
chip->tcpci = tcpci_register_port(chip->udev, &chip->data);
|
||||
if (IS_ERR(chip->tcpci))
|
||||
return PTR_ERR(chip->tcpci);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int husb311_remove(struct udevice *dev)
|
||||
{
|
||||
struct husb311_chip *chip = dev_get_priv(dev);
|
||||
int ret = 0;
|
||||
|
||||
printf("PD chip husb311 remove\n");
|
||||
/* Disable chip interrupts before unregistering port */
|
||||
ret = husb311_write16(chip, TCPC_ALERT_MASK, 0);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
tcpci_unregister_port(chip->tcpci);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int husb311_get_voltage(struct udevice *dev)
|
||||
{
|
||||
struct husb311_chip *chip = dev_get_priv(dev);
|
||||
|
||||
return tcpci_get_voltage_fun(chip->tcpci);
|
||||
}
|
||||
|
||||
static int husb311_get_current(struct udevice *dev)
|
||||
{
|
||||
struct husb311_chip *chip = dev_get_priv(dev);
|
||||
|
||||
return tcpci_get_current_fun(chip->tcpci);
|
||||
}
|
||||
|
||||
static int husb311_get_online(struct udevice *dev)
|
||||
{
|
||||
struct husb311_chip *chip = dev_get_priv(dev);
|
||||
|
||||
return tcpci_get_online_fun(chip->tcpci);
|
||||
}
|
||||
|
||||
static struct dm_power_delivery_ops husb311_ops = {
|
||||
.get_voltage = husb311_get_voltage,
|
||||
.get_current = husb311_get_current,
|
||||
.get_online = husb311_get_online,
|
||||
};
|
||||
|
||||
static const struct udevice_id husb311_ids[] = {
|
||||
{ .compatible = "hynetek,husb311" },
|
||||
{},
|
||||
};
|
||||
|
||||
U_BOOT_DRIVER(husb311) = {
|
||||
.name = "husb311",
|
||||
.id = UCLASS_PD,
|
||||
.of_match = husb311_ids,
|
||||
.ops = &husb311_ops,
|
||||
.probe = husb311_probe,
|
||||
.remove = husb311_remove,
|
||||
.priv_auto_alloc_size = sizeof(struct husb311_chip),
|
||||
};
|
||||
|
||||
MODULE_AUTHOR("Wang Jie <dave.wang@rock-chips.com>");
|
||||
MODULE_DESCRIPTION("Husb311 USB Type-C Port Controller Interface Driver");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
3520
drivers/power/power_delivery/tcpm.c
Executable file
3520
drivers/power/power_delivery/tcpm.c
Executable file
File diff suppressed because it is too large
Load Diff
@@ -118,6 +118,7 @@ enum uclass_id {
|
||||
UCLASS_FG, /* Fuel gauge */
|
||||
UCLASS_CHARGE_DISPLAY, /* Charge display */
|
||||
UCLASS_MCU, /* MCU device */
|
||||
UCLASS_PD, /* PD device */
|
||||
UCLASS_COUNT,
|
||||
UCLASS_INVALID = -1,
|
||||
};
|
||||
|
||||
498
include/power/power_delivery/pd.h
Executable file
498
include/power/power_delivery/pd.h
Executable file
@@ -0,0 +1,498 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
/*
|
||||
* Copyright 2015-2017 Google, Inc
|
||||
*/
|
||||
|
||||
#ifndef __LINUX_USB_PD_H
|
||||
#define __LINUX_USB_PD_H
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/types.h>
|
||||
#include "typec.h"
|
||||
|
||||
/* USB PD Messages */
|
||||
enum pd_ctrl_msg_type {
|
||||
/* 0 Reserved */
|
||||
PD_CTRL_GOOD_CRC = 1,
|
||||
PD_CTRL_GOTO_MIN = 2,
|
||||
PD_CTRL_ACCEPT = 3,
|
||||
PD_CTRL_REJECT = 4,
|
||||
PD_CTRL_PING = 5,
|
||||
PD_CTRL_PS_RDY = 6,
|
||||
PD_CTRL_GET_SOURCE_CAP = 7,
|
||||
PD_CTRL_GET_SINK_CAP = 8,
|
||||
PD_CTRL_DR_SWAP = 9,
|
||||
PD_CTRL_PR_SWAP = 10,
|
||||
PD_CTRL_VCONN_SWAP = 11,
|
||||
PD_CTRL_WAIT = 12,
|
||||
PD_CTRL_SOFT_RESET = 13,
|
||||
/* 14-15 Reserved */
|
||||
PD_CTRL_NOT_SUPP = 16,
|
||||
PD_CTRL_GET_SOURCE_CAP_EXT = 17,
|
||||
PD_CTRL_GET_STATUS = 18,
|
||||
PD_CTRL_FR_SWAP = 19,
|
||||
PD_CTRL_GET_PPS_STATUS = 20,
|
||||
PD_CTRL_GET_COUNTRY_CODES = 21,
|
||||
/* 22-31 Reserved */
|
||||
};
|
||||
|
||||
enum pd_data_msg_type {
|
||||
/* 0 Reserved */
|
||||
PD_DATA_SOURCE_CAP = 1,
|
||||
PD_DATA_REQUEST = 2,
|
||||
PD_DATA_BIST = 3,
|
||||
PD_DATA_SINK_CAP = 4,
|
||||
PD_DATA_BATT_STATUS = 5,
|
||||
PD_DATA_ALERT = 6,
|
||||
PD_DATA_GET_COUNTRY_INFO = 7,
|
||||
PD_DATA_ENTER_USB = 8,
|
||||
/* 9-14 Reserved */
|
||||
PD_DATA_VENDOR_DEF = 15,
|
||||
/* 16-31 Reserved */
|
||||
};
|
||||
|
||||
enum pd_ext_msg_type {
|
||||
/* 0 Reserved */
|
||||
PD_EXT_SOURCE_CAP_EXT = 1,
|
||||
PD_EXT_STATUS = 2,
|
||||
PD_EXT_GET_BATT_CAP = 3,
|
||||
PD_EXT_GET_BATT_STATUS = 4,
|
||||
PD_EXT_BATT_CAP = 5,
|
||||
PD_EXT_GET_MANUFACTURER_INFO = 6,
|
||||
PD_EXT_MANUFACTURER_INFO = 7,
|
||||
PD_EXT_SECURITY_REQUEST = 8,
|
||||
PD_EXT_SECURITY_RESPONSE = 9,
|
||||
PD_EXT_FW_UPDATE_REQUEST = 10,
|
||||
PD_EXT_FW_UPDATE_RESPONSE = 11,
|
||||
PD_EXT_PPS_STATUS = 12,
|
||||
PD_EXT_COUNTRY_INFO = 13,
|
||||
PD_EXT_COUNTRY_CODES = 14,
|
||||
/* 15-31 Reserved */
|
||||
};
|
||||
|
||||
#define PD_REV10 0x0
|
||||
#define PD_REV20 0x1
|
||||
#define PD_REV30 0x2
|
||||
#define PD_MAX_REV PD_REV30
|
||||
|
||||
#define PD_HEADER_EXT_HDR BIT(15)
|
||||
#define PD_HEADER_CNT_SHIFT 12
|
||||
#define PD_HEADER_CNT_MASK 0x7
|
||||
#define PD_HEADER_ID_SHIFT 9
|
||||
#define PD_HEADER_ID_MASK 0x7
|
||||
#define PD_HEADER_PWR_ROLE BIT(8)
|
||||
#define PD_HEADER_REV_SHIFT 6
|
||||
#define PD_HEADER_REV_MASK 0x3
|
||||
#define PD_HEADER_DATA_ROLE BIT(5)
|
||||
#define PD_HEADER_TYPE_SHIFT 0
|
||||
#define PD_HEADER_TYPE_MASK 0x1f
|
||||
|
||||
#define PD_HEADER(type, pwr, data, rev, id, cnt, ext_hdr) \
|
||||
((((type) & PD_HEADER_TYPE_MASK) << PD_HEADER_TYPE_SHIFT) | \
|
||||
((pwr) == TYPEC_SOURCE ? PD_HEADER_PWR_ROLE : 0) | \
|
||||
((data) == TYPEC_HOST ? PD_HEADER_DATA_ROLE : 0) | \
|
||||
(rev << PD_HEADER_REV_SHIFT) | \
|
||||
(((id) & PD_HEADER_ID_MASK) << PD_HEADER_ID_SHIFT) | \
|
||||
(((cnt) & PD_HEADER_CNT_MASK) << PD_HEADER_CNT_SHIFT) | \
|
||||
((ext_hdr) ? PD_HEADER_EXT_HDR : 0))
|
||||
|
||||
#define PD_HEADER_LE(type, pwr, data, rev, id, cnt) \
|
||||
cpu_to_le16(PD_HEADER((type), (pwr), (data), (rev), (id), (cnt), (0)))
|
||||
|
||||
static inline unsigned int pd_header_cnt(u16 header)
|
||||
{
|
||||
return (header >> PD_HEADER_CNT_SHIFT) & PD_HEADER_CNT_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pd_header_cnt_le(__le16 header)
|
||||
{
|
||||
return pd_header_cnt(le16_to_cpu(header));
|
||||
}
|
||||
|
||||
static inline unsigned int pd_header_type(u16 header)
|
||||
{
|
||||
return (header >> PD_HEADER_TYPE_SHIFT) & PD_HEADER_TYPE_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pd_header_type_le(__le16 header)
|
||||
{
|
||||
return pd_header_type(le16_to_cpu(header));
|
||||
}
|
||||
|
||||
static inline unsigned int pd_header_msgid(u16 header)
|
||||
{
|
||||
return (header >> PD_HEADER_ID_SHIFT) & PD_HEADER_ID_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pd_header_msgid_le(__le16 header)
|
||||
{
|
||||
return pd_header_msgid(le16_to_cpu(header));
|
||||
}
|
||||
|
||||
static inline unsigned int pd_header_rev(u16 header)
|
||||
{
|
||||
return (header >> PD_HEADER_REV_SHIFT) & PD_HEADER_REV_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pd_header_rev_le(__le16 header)
|
||||
{
|
||||
return pd_header_rev(le16_to_cpu(header));
|
||||
}
|
||||
|
||||
#define PD_EXT_HDR_CHUNKED BIT(15)
|
||||
#define PD_EXT_HDR_CHUNK_NUM_SHIFT 11
|
||||
#define PD_EXT_HDR_CHUNK_NUM_MASK 0xf
|
||||
#define PD_EXT_HDR_REQ_CHUNK BIT(10)
|
||||
#define PD_EXT_HDR_DATA_SIZE_SHIFT 0
|
||||
#define PD_EXT_HDR_DATA_SIZE_MASK 0x1ff
|
||||
|
||||
#define PD_EXT_HDR(data_size, req_chunk, chunk_num, chunked) \
|
||||
((((data_size) & PD_EXT_HDR_DATA_SIZE_MASK) << PD_EXT_HDR_DATA_SIZE_SHIFT) | \
|
||||
((req_chunk) ? PD_EXT_HDR_REQ_CHUNK : 0) | \
|
||||
(((chunk_num) & PD_EXT_HDR_CHUNK_NUM_MASK) << PD_EXT_HDR_CHUNK_NUM_SHIFT) | \
|
||||
((chunked) ? PD_EXT_HDR_CHUNKED : 0))
|
||||
|
||||
#define PD_EXT_HDR_LE(data_size, req_chunk, chunk_num, chunked) \
|
||||
cpu_to_le16(PD_EXT_HDR((data_size), (req_chunk), (chunk_num), (chunked)))
|
||||
|
||||
static inline unsigned int pd_ext_header_chunk_num(u16 ext_header)
|
||||
{
|
||||
return (ext_header >> PD_EXT_HDR_CHUNK_NUM_SHIFT) &
|
||||
PD_EXT_HDR_CHUNK_NUM_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pd_ext_header_data_size(u16 ext_header)
|
||||
{
|
||||
return (ext_header >> PD_EXT_HDR_DATA_SIZE_SHIFT) &
|
||||
PD_EXT_HDR_DATA_SIZE_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pd_ext_header_data_size_le(__le16 ext_header)
|
||||
{
|
||||
return pd_ext_header_data_size(le16_to_cpu(ext_header));
|
||||
}
|
||||
|
||||
#define PD_MAX_PAYLOAD 7
|
||||
#define PD_EXT_MAX_CHUNK_DATA 26
|
||||
|
||||
/**
|
||||
* struct pd_chunked_ext_message_data - PD chunked extended message data as
|
||||
* seen on wire
|
||||
* @header: PD extended message header
|
||||
* @data: PD extended message data
|
||||
*/
|
||||
struct pd_chunked_ext_message_data {
|
||||
__le16 header;
|
||||
u8 data[PD_EXT_MAX_CHUNK_DATA];
|
||||
} __packed;
|
||||
|
||||
/**
|
||||
* struct pd_message - PD message as seen on wire
|
||||
* @header: PD message header
|
||||
* @payload: PD message payload
|
||||
* @ext_msg: PD message chunked extended message data
|
||||
*/
|
||||
struct pd_message {
|
||||
__le16 header;
|
||||
union {
|
||||
__le32 payload[PD_MAX_PAYLOAD];
|
||||
struct pd_chunked_ext_message_data ext_msg;
|
||||
};
|
||||
} __packed;
|
||||
|
||||
/* PDO: Power Data Object */
|
||||
#define PDO_MAX_OBJECTS 7
|
||||
|
||||
enum pd_pdo_type {
|
||||
PDO_TYPE_FIXED = 0,
|
||||
PDO_TYPE_BATT = 1,
|
||||
PDO_TYPE_VAR = 2,
|
||||
PDO_TYPE_APDO = 3,
|
||||
};
|
||||
|
||||
#define PDO_TYPE_SHIFT 30
|
||||
#define PDO_TYPE_MASK 0x3
|
||||
|
||||
#define PDO_TYPE(t) ((t) << PDO_TYPE_SHIFT)
|
||||
|
||||
#define PDO_VOLT_MASK 0x3ff
|
||||
#define PDO_CURR_MASK 0x3ff
|
||||
#define PDO_PWR_MASK 0x3ff
|
||||
|
||||
#define PDO_FIXED_DUAL_ROLE BIT(29) /* Power role swap supported */
|
||||
#define PDO_FIXED_SUSPEND BIT(28) /* USB Suspend supported (Source) */
|
||||
#define PDO_FIXED_HIGHER_CAP BIT(28) /* Requires more than vSafe5V (Sink) */
|
||||
#define PDO_FIXED_EXTPOWER BIT(27) /* Externally powered */
|
||||
#define PDO_FIXED_USB_COMM BIT(26) /* USB communications capable */
|
||||
#define PDO_FIXED_DATA_SWAP BIT(25) /* Data role swap supported */
|
||||
#define PDO_FIXED_UNCHUNK_EXT BIT(24) /* Unchunked Extended Message supported (Source) */
|
||||
#define PDO_FIXED_FRS_CURR_MASK (BIT(24) | BIT(23)) /* FR_Swap Current (Sink) */
|
||||
#define PDO_FIXED_FRS_CURR_SHIFT 23
|
||||
#define PDO_FIXED_VOLT_SHIFT 10 /* 50mV units */
|
||||
#define PDO_FIXED_CURR_SHIFT 0 /* 10mA units */
|
||||
|
||||
#define PDO_FIXED_VOLT(mv) ((((mv) / 50) & PDO_VOLT_MASK) << PDO_FIXED_VOLT_SHIFT)
|
||||
#define PDO_FIXED_CURR(ma) ((((ma) / 10) & PDO_CURR_MASK) << PDO_FIXED_CURR_SHIFT)
|
||||
|
||||
#define PDO_FIXED(mv, ma, flags) \
|
||||
(PDO_TYPE(PDO_TYPE_FIXED) | (flags) | \
|
||||
PDO_FIXED_VOLT(mv) | PDO_FIXED_CURR(ma))
|
||||
|
||||
#define VSAFE5V 5000 /* mv units */
|
||||
|
||||
#define PDO_BATT_MAX_VOLT_SHIFT 20 /* 50mV units */
|
||||
#define PDO_BATT_MIN_VOLT_SHIFT 10 /* 50mV units */
|
||||
#define PDO_BATT_MAX_PWR_SHIFT 0 /* 250mW units */
|
||||
|
||||
#define PDO_BATT_MIN_VOLT(mv) ((((mv) / 50) & PDO_VOLT_MASK) << PDO_BATT_MIN_VOLT_SHIFT)
|
||||
#define PDO_BATT_MAX_VOLT(mv) ((((mv) / 50) & PDO_VOLT_MASK) << PDO_BATT_MAX_VOLT_SHIFT)
|
||||
#define PDO_BATT_MAX_POWER(mw) ((((mw) / 250) & PDO_PWR_MASK) << PDO_BATT_MAX_PWR_SHIFT)
|
||||
|
||||
#define PDO_BATT(min_mv, max_mv, max_mw) \
|
||||
(PDO_TYPE(PDO_TYPE_BATT) | PDO_BATT_MIN_VOLT(min_mv) | \
|
||||
PDO_BATT_MAX_VOLT(max_mv) | PDO_BATT_MAX_POWER(max_mw))
|
||||
|
||||
#define PDO_VAR_MAX_VOLT_SHIFT 20 /* 50mV units */
|
||||
#define PDO_VAR_MIN_VOLT_SHIFT 10 /* 50mV units */
|
||||
#define PDO_VAR_MAX_CURR_SHIFT 0 /* 10mA units */
|
||||
|
||||
#define PDO_VAR_MIN_VOLT(mv) ((((mv) / 50) & PDO_VOLT_MASK) << PDO_VAR_MIN_VOLT_SHIFT)
|
||||
#define PDO_VAR_MAX_VOLT(mv) ((((mv) / 50) & PDO_VOLT_MASK) << PDO_VAR_MAX_VOLT_SHIFT)
|
||||
#define PDO_VAR_MAX_CURR(ma) ((((ma) / 10) & PDO_CURR_MASK) << PDO_VAR_MAX_CURR_SHIFT)
|
||||
|
||||
#define PDO_VAR(min_mv, max_mv, max_ma) \
|
||||
(PDO_TYPE(PDO_TYPE_VAR) | PDO_VAR_MIN_VOLT(min_mv) | \
|
||||
PDO_VAR_MAX_VOLT(max_mv) | PDO_VAR_MAX_CURR(max_ma))
|
||||
|
||||
enum pd_apdo_type {
|
||||
APDO_TYPE_PPS = 0,
|
||||
};
|
||||
|
||||
#define PDO_APDO_TYPE_SHIFT 28 /* Only valid value currently is 0x0 - PPS */
|
||||
#define PDO_APDO_TYPE_MASK 0x3
|
||||
|
||||
#define PDO_APDO_TYPE(t) ((t) << PDO_APDO_TYPE_SHIFT)
|
||||
|
||||
#define PDO_PPS_APDO_MAX_VOLT_SHIFT 17 /* 100mV units */
|
||||
#define PDO_PPS_APDO_MIN_VOLT_SHIFT 8 /* 100mV units */
|
||||
#define PDO_PPS_APDO_MAX_CURR_SHIFT 0 /* 50mA units */
|
||||
|
||||
#define PDO_PPS_APDO_VOLT_MASK 0xff
|
||||
#define PDO_PPS_APDO_CURR_MASK 0x7f
|
||||
|
||||
#define PDO_PPS_APDO_MIN_VOLT(mv) \
|
||||
((((mv) / 100) & PDO_PPS_APDO_VOLT_MASK) << PDO_PPS_APDO_MIN_VOLT_SHIFT)
|
||||
#define PDO_PPS_APDO_MAX_VOLT(mv) \
|
||||
((((mv) / 100) & PDO_PPS_APDO_VOLT_MASK) << PDO_PPS_APDO_MAX_VOLT_SHIFT)
|
||||
#define PDO_PPS_APDO_MAX_CURR(ma) \
|
||||
((((ma) / 50) & PDO_PPS_APDO_CURR_MASK) << PDO_PPS_APDO_MAX_CURR_SHIFT)
|
||||
|
||||
#define PDO_PPS_APDO(min_mv, max_mv, max_ma) \
|
||||
(PDO_TYPE(PDO_TYPE_APDO) | PDO_APDO_TYPE(APDO_TYPE_PPS) | \
|
||||
PDO_PPS_APDO_MIN_VOLT(min_mv) | PDO_PPS_APDO_MAX_VOLT(max_mv) | \
|
||||
PDO_PPS_APDO_MAX_CURR(max_ma))
|
||||
|
||||
static inline enum pd_pdo_type pdo_type(u32 pdo)
|
||||
{
|
||||
return (pdo >> PDO_TYPE_SHIFT) & PDO_TYPE_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_fixed_voltage(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_FIXED_VOLT_SHIFT) & PDO_VOLT_MASK) * 50;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_min_voltage(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_VAR_MIN_VOLT_SHIFT) & PDO_VOLT_MASK) * 50;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_max_voltage(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_VAR_MAX_VOLT_SHIFT) & PDO_VOLT_MASK) * 50;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_max_current(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_VAR_MAX_CURR_SHIFT) & PDO_CURR_MASK) * 10;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_max_power(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_BATT_MAX_PWR_SHIFT) & PDO_PWR_MASK) * 250;
|
||||
}
|
||||
|
||||
static inline enum pd_apdo_type pdo_apdo_type(u32 pdo)
|
||||
{
|
||||
return (pdo >> PDO_APDO_TYPE_SHIFT) & PDO_APDO_TYPE_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_pps_apdo_min_voltage(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_PPS_APDO_MIN_VOLT_SHIFT) &
|
||||
PDO_PPS_APDO_VOLT_MASK) * 100;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_pps_apdo_max_voltage(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_PPS_APDO_MAX_VOLT_SHIFT) &
|
||||
PDO_PPS_APDO_VOLT_MASK) * 100;
|
||||
}
|
||||
|
||||
static inline unsigned int pdo_pps_apdo_max_current(u32 pdo)
|
||||
{
|
||||
return ((pdo >> PDO_PPS_APDO_MAX_CURR_SHIFT) &
|
||||
PDO_PPS_APDO_CURR_MASK) * 50;
|
||||
}
|
||||
|
||||
/* RDO: Request Data Object */
|
||||
#define RDO_OBJ_POS_SHIFT 28
|
||||
#define RDO_OBJ_POS_MASK 0x7
|
||||
#define RDO_GIVE_BACK BIT(27) /* Supports reduced operating current */
|
||||
#define RDO_CAP_MISMATCH BIT(26) /* Not satisfied by source caps */
|
||||
#define RDO_USB_COMM BIT(25) /* USB communications capable */
|
||||
#define RDO_NO_SUSPEND BIT(24) /* USB Suspend not supported */
|
||||
|
||||
#define RDO_PWR_MASK 0x3ff
|
||||
#define RDO_CURR_MASK 0x3ff
|
||||
|
||||
#define RDO_FIXED_OP_CURR_SHIFT 10
|
||||
#define RDO_FIXED_MAX_CURR_SHIFT 0
|
||||
|
||||
#define RDO_OBJ(idx) (((idx) & RDO_OBJ_POS_MASK) << RDO_OBJ_POS_SHIFT)
|
||||
|
||||
#define PDO_FIXED_OP_CURR(ma) ((((ma) / 10) & RDO_CURR_MASK) << RDO_FIXED_OP_CURR_SHIFT)
|
||||
#define PDO_FIXED_MAX_CURR(ma) ((((ma) / 10) & RDO_CURR_MASK) << RDO_FIXED_MAX_CURR_SHIFT)
|
||||
|
||||
#define RDO_FIXED(idx, op_ma, max_ma, flags) \
|
||||
(RDO_OBJ(idx) | (flags) | \
|
||||
PDO_FIXED_OP_CURR(op_ma) | PDO_FIXED_MAX_CURR(max_ma))
|
||||
|
||||
#define RDO_BATT_OP_PWR_SHIFT 10 /* 250mW units */
|
||||
#define RDO_BATT_MAX_PWR_SHIFT 0 /* 250mW units */
|
||||
|
||||
#define RDO_BATT_OP_PWR(mw) ((((mw) / 250) & RDO_PWR_MASK) << RDO_BATT_OP_PWR_SHIFT)
|
||||
#define RDO_BATT_MAX_PWR(mw) ((((mw) / 250) & RDO_PWR_MASK) << RDO_BATT_MAX_PWR_SHIFT)
|
||||
|
||||
#define RDO_BATT(idx, op_mw, max_mw, flags) \
|
||||
(RDO_OBJ(idx) | (flags) | \
|
||||
RDO_BATT_OP_PWR(op_mw) | RDO_BATT_MAX_PWR(max_mw))
|
||||
|
||||
#define RDO_PROG_VOLT_MASK 0x7ff
|
||||
#define RDO_PROG_CURR_MASK 0x7f
|
||||
|
||||
#define RDO_PROG_VOLT_SHIFT 9
|
||||
#define RDO_PROG_CURR_SHIFT 0
|
||||
|
||||
#define RDO_PROG_VOLT_MV_STEP 20
|
||||
#define RDO_PROG_CURR_MA_STEP 50
|
||||
|
||||
#define PDO_PROG_OUT_VOLT(mv) \
|
||||
((((mv) / RDO_PROG_VOLT_MV_STEP) & RDO_PROG_VOLT_MASK) << RDO_PROG_VOLT_SHIFT)
|
||||
#define PDO_PROG_OP_CURR(ma) \
|
||||
((((ma) / RDO_PROG_CURR_MA_STEP) & RDO_PROG_CURR_MASK) << RDO_PROG_CURR_SHIFT)
|
||||
|
||||
#define RDO_PROG(idx, out_mv, op_ma, flags) \
|
||||
(RDO_OBJ(idx) | (flags) | \
|
||||
PDO_PROG_OUT_VOLT(out_mv) | PDO_PROG_OP_CURR(op_ma))
|
||||
|
||||
static inline unsigned int rdo_index(u32 rdo)
|
||||
{
|
||||
return (rdo >> RDO_OBJ_POS_SHIFT) & RDO_OBJ_POS_MASK;
|
||||
}
|
||||
|
||||
static inline unsigned int rdo_op_current(u32 rdo)
|
||||
{
|
||||
return ((rdo >> RDO_FIXED_OP_CURR_SHIFT) & RDO_CURR_MASK) * 10;
|
||||
}
|
||||
|
||||
static inline unsigned int rdo_max_current(u32 rdo)
|
||||
{
|
||||
return ((rdo >> RDO_FIXED_MAX_CURR_SHIFT) &
|
||||
RDO_CURR_MASK) * 10;
|
||||
}
|
||||
|
||||
static inline unsigned int rdo_op_power(u32 rdo)
|
||||
{
|
||||
return ((rdo >> RDO_BATT_OP_PWR_SHIFT) & RDO_PWR_MASK) * 250;
|
||||
}
|
||||
|
||||
static inline unsigned int rdo_max_power(u32 rdo)
|
||||
{
|
||||
return ((rdo >> RDO_BATT_MAX_PWR_SHIFT) & RDO_PWR_MASK) * 250;
|
||||
}
|
||||
|
||||
/* Enter_USB Data Object */
|
||||
#define EUDO_USB_MODE_MASK GENMASK(30, 28)
|
||||
#define EUDO_USB_MODE_SHIFT 28
|
||||
#define EUDO_USB_MODE_USB2 0
|
||||
#define EUDO_USB_MODE_USB3 1
|
||||
#define EUDO_USB_MODE_USB4 2
|
||||
#define EUDO_USB4_DRD BIT(26)
|
||||
#define EUDO_USB3_DRD BIT(25)
|
||||
#define EUDO_CABLE_SPEED_MASK GENMASK(23, 21)
|
||||
#define EUDO_CABLE_SPEED_SHIFT 21
|
||||
#define EUDO_CABLE_SPEED_USB2 0
|
||||
#define EUDO_CABLE_SPEED_USB3_GEN1 1
|
||||
#define EUDO_CABLE_SPEED_USB4_GEN2 2
|
||||
#define EUDO_CABLE_SPEED_USB4_GEN3 3
|
||||
#define EUDO_CABLE_TYPE_MASK GENMASK(20, 19)
|
||||
#define EUDO_CABLE_TYPE_SHIFT 19
|
||||
#define EUDO_CABLE_TYPE_PASSIVE 0
|
||||
#define EUDO_CABLE_TYPE_RE_TIMER 1
|
||||
#define EUDO_CABLE_TYPE_RE_DRIVER 2
|
||||
#define EUDO_CABLE_TYPE_OPTICAL 3
|
||||
#define EUDO_CABLE_CURRENT_MASK GENMASK(18, 17)
|
||||
#define EUDO_CABLE_CURRENT_SHIFT 17
|
||||
#define EUDO_CABLE_CURRENT_NOTSUPP 0
|
||||
#define EUDO_CABLE_CURRENT_3A 2
|
||||
#define EUDO_CABLE_CURRENT_5A 3
|
||||
#define EUDO_PCIE_SUPPORT BIT(16)
|
||||
#define EUDO_DP_SUPPORT BIT(15)
|
||||
#define EUDO_TBT_SUPPORT BIT(14)
|
||||
#define EUDO_HOST_PRESENT BIT(13)
|
||||
|
||||
/* USB PD timers and counters */
|
||||
#define PD_T_NO_RESPONSE 5000 /* 4.5 - 5.5 seconds */
|
||||
#define PD_T_DB_DETECT 10000 /* 10 - 15 seconds */
|
||||
#define PD_T_SEND_SOURCE_CAP 150 /* 100 - 200 ms */
|
||||
#define PD_T_SENDER_RESPONSE 60 /* 24 - 30 ms, relaxed */
|
||||
#define PD_T_RECEIVER_RESPONSE 15 /* 15ms max */
|
||||
#define PD_T_SOURCE_ACTIVITY 45
|
||||
#define PD_T_SINK_ACTIVITY 135
|
||||
#define PD_T_SINK_WAIT_CAP 310 /* 310 - 620 ms */
|
||||
#define PD_T_PS_TRANSITION 500
|
||||
#define PD_T_SRC_TRANSITION 35
|
||||
#define PD_T_DRP_SNK 40
|
||||
#define PD_T_DRP_SRC 30
|
||||
#define PD_T_PS_SOURCE_OFF 920
|
||||
#define PD_T_PS_SOURCE_ON 480
|
||||
#define PD_T_PS_SOURCE_ON_PRS 450 /* 390 - 480ms */
|
||||
#define PD_T_PS_HARD_RESET 30
|
||||
#define PD_T_SRC_RECOVER 760
|
||||
#define PD_T_SRC_RECOVER_MAX 1000
|
||||
#define PD_T_SRC_TURN_ON 275
|
||||
#define PD_T_SAFE_0V 650
|
||||
#define PD_T_VCONN_SOURCE_ON 100
|
||||
#define PD_T_SINK_REQUEST 100 /* 100 ms minimum */
|
||||
#define PD_T_ERROR_RECOVERY 100 /* minimum 25 is insufficient */
|
||||
#define PD_T_SRCSWAPSTDBY 625 /* Maximum of 650ms */
|
||||
#define PD_T_NEWSRC 250 /* Maximum of 275ms */
|
||||
#define PD_T_SWAP_SRC_START 20 /* Minimum of 20ms */
|
||||
#define PD_T_BIST_CONT_MODE 50 /* 30 - 60 ms */
|
||||
#define PD_T_SINK_TX 16 /* 16 - 20 ms */
|
||||
#define PD_T_CHUNK_NOT_SUPP 42 /* 40 - 50 ms */
|
||||
|
||||
#define PD_T_DRP_TRY 100 /* 75 - 150 ms */
|
||||
#define PD_T_DRP_TRYWAIT 600 /* 400 - 800 ms */
|
||||
|
||||
#define PD_T_CC_DEBOUNCE 200 /* 100 - 200 ms */
|
||||
#define PD_T_PD_DEBOUNCE 20 /* 10 - 20 ms */
|
||||
#define PD_T_TRY_CC_DEBOUNCE 15 /* 10 - 20 ms */
|
||||
|
||||
#define PD_N_CAPS_COUNT (PD_T_NO_RESPONSE / PD_T_SEND_SOURCE_CAP)
|
||||
#define PD_N_HARD_RESET_COUNT 1
|
||||
|
||||
#define PD_P_SNK_STDBY_MW 2500 /* 2500 mW */
|
||||
|
||||
#endif /* __LINUX_USB_PD_H */
|
||||
518
include/power/power_delivery/pd_vdo.h
Executable file
518
include/power/power_delivery/pd_vdo.h
Executable file
@@ -0,0 +1,518 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
/*
|
||||
* Copyright 2015-2017 Google, Inc
|
||||
*/
|
||||
|
||||
#ifndef __LINUX_USB_PD_VDO_H
|
||||
#define __LINUX_USB_PD_VDO_H
|
||||
|
||||
#include "pd.h"
|
||||
|
||||
/*
|
||||
* VDO : Vendor Defined Message Object
|
||||
* VDM object is minimum of VDM header + 6 additional data objects.
|
||||
*/
|
||||
|
||||
#define VDO_MAX_OBJECTS 6
|
||||
#define VDO_MAX_SIZE (VDO_MAX_OBJECTS + 1)
|
||||
|
||||
/*
|
||||
* VDM header
|
||||
* ----------
|
||||
* <31:16> :: SVID
|
||||
* <15> :: VDM type ( 1b == structured, 0b == unstructured )
|
||||
* <14:13> :: Structured VDM version
|
||||
* <12:11> :: reserved
|
||||
* <10:8> :: object position (1-7 valid ... used for enter/exit mode only)
|
||||
* <7:6> :: command type (SVDM only?)
|
||||
* <5> :: reserved (SVDM), command type (UVDM)
|
||||
* <4:0> :: command
|
||||
*/
|
||||
#define VDO(vid, type, ver, custom) \
|
||||
(((vid) << 16) | \
|
||||
((type) << 15) | \
|
||||
((ver) << 13) | \
|
||||
((custom) & 0x7FFF))
|
||||
|
||||
#define VDO_SVDM_TYPE (1 << 15)
|
||||
#define VDO_SVDM_VERS(x) ((x) << 13)
|
||||
#define VDO_OPOS(x) ((x) << 8)
|
||||
#define VDO_CMDT(x) ((x) << 6)
|
||||
#define VDO_SVDM_VERS_MASK VDO_SVDM_VERS(0x3)
|
||||
#define VDO_OPOS_MASK VDO_OPOS(0x7)
|
||||
#define VDO_CMDT_MASK VDO_CMDT(0x3)
|
||||
|
||||
#define CMDT_INIT 0
|
||||
#define CMDT_RSP_ACK 1
|
||||
#define CMDT_RSP_NAK 2
|
||||
#define CMDT_RSP_BUSY 3
|
||||
|
||||
/* reserved for SVDM ... for Google UVDM */
|
||||
#define VDO_SRC_INITIATOR (0 << 5)
|
||||
#define VDO_SRC_RESPONDER (1 << 5)
|
||||
|
||||
#define CMD_DISCOVER_IDENT 1
|
||||
#define CMD_DISCOVER_SVID 2
|
||||
#define CMD_DISCOVER_MODES 3
|
||||
#define CMD_ENTER_MODE 4
|
||||
#define CMD_EXIT_MODE 5
|
||||
#define CMD_ATTENTION 6
|
||||
|
||||
#define VDO_CMD_VENDOR(x) (((0x10 + (x)) & 0x1f))
|
||||
|
||||
/* ChromeOS specific commands */
|
||||
#define VDO_CMD_VERSION VDO_CMD_VENDOR(0)
|
||||
#define VDO_CMD_SEND_INFO VDO_CMD_VENDOR(1)
|
||||
#define VDO_CMD_READ_INFO VDO_CMD_VENDOR(2)
|
||||
#define VDO_CMD_REBOOT VDO_CMD_VENDOR(5)
|
||||
#define VDO_CMD_FLASH_ERASE VDO_CMD_VENDOR(6)
|
||||
#define VDO_CMD_FLASH_WRITE VDO_CMD_VENDOR(7)
|
||||
#define VDO_CMD_ERASE_SIG VDO_CMD_VENDOR(8)
|
||||
#define VDO_CMD_PING_ENABLE VDO_CMD_VENDOR(10)
|
||||
#define VDO_CMD_CURRENT VDO_CMD_VENDOR(11)
|
||||
#define VDO_CMD_FLIP VDO_CMD_VENDOR(12)
|
||||
#define VDO_CMD_GET_LOG VDO_CMD_VENDOR(13)
|
||||
#define VDO_CMD_CCD_EN VDO_CMD_VENDOR(14)
|
||||
|
||||
#define PD_VDO_VID(vdo) ((vdo) >> 16)
|
||||
#define PD_VDO_SVDM(vdo) (((vdo) >> 15) & 1)
|
||||
#define PD_VDO_SVDM_VER(vdo) (((vdo) >> 13) & 0x3)
|
||||
#define PD_VDO_OPOS(vdo) (((vdo) >> 8) & 0x7)
|
||||
#define PD_VDO_CMD(vdo) ((vdo) & 0x1f)
|
||||
#define PD_VDO_CMDT(vdo) (((vdo) >> 6) & 0x3)
|
||||
|
||||
/*
|
||||
* SVDM Identity request -> response
|
||||
*
|
||||
* Request is simply properly formatted SVDM header
|
||||
*
|
||||
* Response is 4 data objects:
|
||||
* [0] :: SVDM header
|
||||
* [1] :: Identitiy header
|
||||
* [2] :: Cert Stat VDO
|
||||
* [3] :: (Product | Cable) VDO
|
||||
* [4] :: AMA VDO
|
||||
*
|
||||
*/
|
||||
#define VDO_INDEX_HDR 0
|
||||
#define VDO_INDEX_IDH 1
|
||||
#define VDO_INDEX_CSTAT 2
|
||||
#define VDO_INDEX_CABLE 3
|
||||
#define VDO_INDEX_PRODUCT 3
|
||||
#define VDO_INDEX_AMA 4
|
||||
|
||||
/*
|
||||
* SVDM Identity Header
|
||||
* --------------------
|
||||
* <31> :: data capable as a USB host
|
||||
* <30> :: data capable as a USB device
|
||||
* <29:27> :: product type (UFP / Cable / VPD)
|
||||
* <26> :: modal operation supported (1b == yes)
|
||||
* <25:23> :: product type (DFP) (SVDM version 2.0+ only; set to zero in version 1.0)
|
||||
* <22:21> :: connector type (SVDM version 2.0+ only; set to zero in version 1.0)
|
||||
* <20:16> :: Reserved, Shall be set to zero
|
||||
* <15:0> :: USB-IF assigned VID for this cable vendor
|
||||
*/
|
||||
|
||||
/* PD Rev2.0 definition */
|
||||
#define IDH_PTYPE_UNDEF 0
|
||||
|
||||
/* SOP Product Type (UFP) */
|
||||
#define IDH_PTYPE_NOT_UFP 0
|
||||
#define IDH_PTYPE_HUB 1
|
||||
#define IDH_PTYPE_PERIPH 2
|
||||
#define IDH_PTYPE_PSD 3
|
||||
#define IDH_PTYPE_AMA 5
|
||||
|
||||
/* SOP' Product Type (Cable Plug / VPD) */
|
||||
#define IDH_PTYPE_NOT_CABLE 0
|
||||
#define IDH_PTYPE_PCABLE 3
|
||||
#define IDH_PTYPE_ACABLE 4
|
||||
#define IDH_PTYPE_VPD 6
|
||||
|
||||
/* SOP Product Type (DFP) */
|
||||
#define IDH_PTYPE_NOT_DFP 0
|
||||
#define IDH_PTYPE_DFP_HUB 1
|
||||
#define IDH_PTYPE_DFP_HOST 2
|
||||
#define IDH_PTYPE_DFP_PB 3
|
||||
|
||||
/* ID Header Mask */
|
||||
#define IDH_DFP_MASK GENMASK(25, 23)
|
||||
#define IDH_CONN_MASK GENMASK(22, 21)
|
||||
|
||||
#define VDO_IDH(usbh, usbd, ufp_cable, is_modal, dfp, conn, vid) \
|
||||
((usbh) << 31 | (usbd) << 30 | ((ufp_cable) & 0x7) << 27 \
|
||||
| (is_modal) << 26 | ((dfp) & 0x7) << 23 | ((conn) & 0x3) << 21 \
|
||||
| ((vid) & 0xffff))
|
||||
|
||||
#define PD_IDH_PTYPE(vdo) (((vdo) >> 27) & 0x7)
|
||||
#define PD_IDH_VID(vdo) ((vdo) & 0xffff)
|
||||
#define PD_IDH_MODAL_SUPP(vdo) ((vdo) & (1 << 26))
|
||||
#define PD_IDH_DFP_PTYPE(vdo) (((vdo) >> 23) & 0x7)
|
||||
#define PD_IDH_CONN_TYPE(vdo) (((vdo) >> 21) & 0x3)
|
||||
|
||||
/*
|
||||
* Cert Stat VDO
|
||||
* -------------
|
||||
* <31:0> : USB-IF assigned XID for this cable
|
||||
*/
|
||||
#define PD_CSTAT_XID(vdo) (vdo)
|
||||
#define VDO_CERT(xid) ((xid) & 0xffffffff)
|
||||
|
||||
/*
|
||||
* Product VDO
|
||||
* -----------
|
||||
* <31:16> : USB Product ID
|
||||
* <15:0> : USB bcdDevice
|
||||
*/
|
||||
#define VDO_PRODUCT(pid, bcd) (((pid) & 0xffff) << 16 | ((bcd) & 0xffff))
|
||||
#define PD_PRODUCT_PID(vdo) (((vdo) >> 16) & 0xffff)
|
||||
|
||||
/*
|
||||
* UFP VDO (PD Revision 3.0+ only)
|
||||
* --------
|
||||
* <31:29> :: UFP VDO version
|
||||
* <28> :: Reserved
|
||||
* <27:24> :: Device capability
|
||||
* <23:22> :: Connector type (10b == receptacle, 11b == captive plug)
|
||||
* <21:11> :: Reserved
|
||||
* <10:8> :: Vconn power (AMA only)
|
||||
* <7> :: Vconn required (AMA only, 0b == no, 1b == yes)
|
||||
* <6> :: Vbus required (AMA only, 0b == yes, 1b == no)
|
||||
* <5:3> :: Alternate modes
|
||||
* <2:0> :: USB highest speed
|
||||
*/
|
||||
#define PD_VDO_UFP_DEVCAP(vdo) (((vdo) & GENMASK(27, 24)) >> 24)
|
||||
|
||||
/* UFP VDO Version */
|
||||
#define UFP_VDO_VER1_2 2
|
||||
|
||||
/* Device Capability */
|
||||
#define DEV_USB2_CAPABLE BIT(0)
|
||||
#define DEV_USB2_BILLBOARD BIT(1)
|
||||
#define DEV_USB3_CAPABLE BIT(2)
|
||||
#define DEV_USB4_CAPABLE BIT(3)
|
||||
|
||||
/* Connector Type */
|
||||
#define UFP_RECEPTACLE 2
|
||||
#define UFP_CAPTIVE 3
|
||||
|
||||
/* Vconn Power (AMA only, set to AMA_VCONN_NOT_REQ if Vconn is not required) */
|
||||
#define AMA_VCONN_PWR_1W 0
|
||||
#define AMA_VCONN_PWR_1W5 1
|
||||
#define AMA_VCONN_PWR_2W 2
|
||||
#define AMA_VCONN_PWR_3W 3
|
||||
#define AMA_VCONN_PWR_4W 4
|
||||
#define AMA_VCONN_PWR_5W 5
|
||||
#define AMA_VCONN_PWR_6W 6
|
||||
|
||||
/* Vconn Required (AMA only) */
|
||||
#define AMA_VCONN_NOT_REQ 0
|
||||
#define AMA_VCONN_REQ 1
|
||||
|
||||
/* Vbus Required (AMA only) */
|
||||
#define AMA_VBUS_REQ 0
|
||||
#define AMA_VBUS_NOT_REQ 1
|
||||
|
||||
/* Alternate Modes */
|
||||
#define UFP_ALTMODE_NOT_SUPP 0
|
||||
#define UFP_ALTMODE_TBT3 BIT(0)
|
||||
#define UFP_ALTMODE_RECFG BIT(1)
|
||||
#define UFP_ALTMODE_NO_RECFG BIT(2)
|
||||
|
||||
/* USB Highest Speed */
|
||||
#define UFP_USB2_ONLY 0
|
||||
#define UFP_USB32_GEN1 1
|
||||
#define UFP_USB32_4_GEN2 2
|
||||
#define UFP_USB4_GEN3 3
|
||||
|
||||
#define VDO_UFP(ver, cap, conn, vcpwr, vcr, vbr, alt, spd) \
|
||||
(((ver) & 0x7) << 29 | ((cap) & 0xf) << 24 | ((conn) & 0x3) << 22 \
|
||||
| ((vcpwr) & 0x7) << 8 | (vcr) << 7 | (vbr) << 6 | ((alt) & 0x7) << 3 \
|
||||
| ((spd) & 0x7))
|
||||
|
||||
/*
|
||||
* DFP VDO (PD Revision 3.0+ only)
|
||||
* --------
|
||||
* <31:29> :: DFP VDO version
|
||||
* <28:27> :: Reserved
|
||||
* <26:24> :: Host capability
|
||||
* <23:22> :: Connector type (10b == receptacle, 11b == captive plug)
|
||||
* <21:5> :: Reserved
|
||||
* <4:0> :: Port number
|
||||
*/
|
||||
#define PD_VDO_DFP_HOSTCAP(vdo) (((vdo) & GENMASK(26, 24)) >> 24)
|
||||
|
||||
#define DFP_VDO_VER1_1 1
|
||||
#define HOST_USB2_CAPABLE BIT(0)
|
||||
#define HOST_USB3_CAPABLE BIT(1)
|
||||
#define HOST_USB4_CAPABLE BIT(2)
|
||||
#define DFP_RECEPTACLE 2
|
||||
#define DFP_CAPTIVE 3
|
||||
|
||||
#define VDO_DFP(ver, cap, conn, pnum) \
|
||||
(((ver) & 0x7) << 29 | ((cap) & 0x7) << 24 | ((conn) & 0x3) << 22 \
|
||||
| ((pnum) & 0x1f))
|
||||
|
||||
/*
|
||||
* Cable VDO (for both Passive and Active Cable VDO in PD Rev2.0)
|
||||
* ---------
|
||||
* <31:28> :: Cable HW version
|
||||
* <27:24> :: Cable FW version
|
||||
* <23:20> :: Reserved, Shall be set to zero
|
||||
* <19:18> :: type-C to Type-A/B/C/Captive (00b == A, 01 == B, 10 == C, 11 == Captive)
|
||||
* <17> :: Reserved, Shall be set to zero
|
||||
* <16:13> :: cable latency (0001 == <10ns(~1m length))
|
||||
* <12:11> :: cable termination type (11b == both ends active VCONN req)
|
||||
* <10> :: SSTX1 Directionality support (0b == fixed, 1b == cfgable)
|
||||
* <9> :: SSTX2 Directionality support
|
||||
* <8> :: SSRX1 Directionality support
|
||||
* <7> :: SSRX2 Directionality support
|
||||
* <6:5> :: Vbus current handling capability (01b == 3A, 10b == 5A)
|
||||
* <4> :: Vbus through cable (0b == no, 1b == yes)
|
||||
* <3> :: SOP" controller present? (0b == no, 1b == yes)
|
||||
* <2:0> :: USB SS Signaling support
|
||||
*
|
||||
* Passive Cable VDO (PD Rev3.0+)
|
||||
* ---------
|
||||
* <31:28> :: Cable HW version
|
||||
* <27:24> :: Cable FW version
|
||||
* <23:21> :: VDO version
|
||||
* <20> :: Reserved, Shall be set to zero
|
||||
* <19:18> :: Type-C to Type-C/Captive (10b == C, 11b == Captive)
|
||||
* <17> :: Reserved, Shall be set to zero
|
||||
* <16:13> :: cable latency (0001 == <10ns(~1m length))
|
||||
* <12:11> :: cable termination type (10b == Vconn not req, 01b == Vconn req)
|
||||
* <10:9> :: Maximum Vbus voltage (00b == 20V, 01b == 30V, 10b == 40V, 11b == 50V)
|
||||
* <8:7> :: Reserved, Shall be set to zero
|
||||
* <6:5> :: Vbus current handling capability (01b == 3A, 10b == 5A)
|
||||
* <4:3> :: Reserved, Shall be set to zero
|
||||
* <2:0> :: USB highest speed
|
||||
*
|
||||
* Active Cable VDO 1 (PD Rev3.0+)
|
||||
* ---------
|
||||
* <31:28> :: Cable HW version
|
||||
* <27:24> :: Cable FW version
|
||||
* <23:21> :: VDO version
|
||||
* <20> :: Reserved, Shall be set to zero
|
||||
* <19:18> :: Connector type (10b == C, 11b == Captive)
|
||||
* <17> :: Reserved, Shall be set to zero
|
||||
* <16:13> :: cable latency (0001 == <10ns(~1m length))
|
||||
* <12:11> :: cable termination type (10b == one end active, 11b == both ends active VCONN req)
|
||||
* <10:9> :: Maximum Vbus voltage (00b == 20V, 01b == 30V, 10b == 40V, 11b == 50V)
|
||||
* <8> :: SBU supported (0b == supported, 1b == not supported)
|
||||
* <7> :: SBU type (0b == passive, 1b == active)
|
||||
* <6:5> :: Vbus current handling capability (01b == 3A, 10b == 5A)
|
||||
* <2:0> :: USB highest speed
|
||||
*/
|
||||
/* Cable VDO Version */
|
||||
#define CABLE_VDO_VER1_0 0
|
||||
#define CABLE_VDO_VER1_3 3
|
||||
|
||||
/* Connector Type (_ATYPE and _BTYPE are for PD Rev2.0 only) */
|
||||
#define CABLE_ATYPE 0
|
||||
#define CABLE_BTYPE 1
|
||||
#define CABLE_CTYPE 2
|
||||
#define CABLE_CAPTIVE 3
|
||||
|
||||
/* Cable Latency */
|
||||
#define CABLE_LATENCY_1M 1
|
||||
#define CABLE_LATENCY_2M 2
|
||||
#define CABLE_LATENCY_3M 3
|
||||
#define CABLE_LATENCY_4M 4
|
||||
#define CABLE_LATENCY_5M 5
|
||||
#define CABLE_LATENCY_6M 6
|
||||
#define CABLE_LATENCY_7M 7
|
||||
#define CABLE_LATENCY_7M_PLUS 8
|
||||
|
||||
/* Cable Termination Type */
|
||||
#define PCABLE_VCONN_NOT_REQ 0
|
||||
#define PCABLE_VCONN_REQ 1
|
||||
#define ACABLE_ONE_END 2
|
||||
#define ACABLE_BOTH_END 3
|
||||
|
||||
/* Maximum Vbus Voltage */
|
||||
#define CABLE_MAX_VBUS_20V 0
|
||||
#define CABLE_MAX_VBUS_30V 1
|
||||
#define CABLE_MAX_VBUS_40V 2
|
||||
#define CABLE_MAX_VBUS_50V 3
|
||||
|
||||
/* Active Cable SBU Supported/Type */
|
||||
#define ACABLE_SBU_SUPP 0
|
||||
#define ACABLE_SBU_NOT_SUPP 1
|
||||
#define ACABLE_SBU_PASSIVE 0
|
||||
#define ACABLE_SBU_ACTIVE 1
|
||||
|
||||
/* Vbus Current Handling Capability */
|
||||
#define CABLE_CURR_DEF 0
|
||||
#define CABLE_CURR_3A 1
|
||||
#define CABLE_CURR_5A 2
|
||||
|
||||
/* USB SuperSpeed Signaling Support (PD Rev2.0) */
|
||||
#define CABLE_USBSS_U2_ONLY 0
|
||||
#define CABLE_USBSS_U31_GEN1 1
|
||||
#define CABLE_USBSS_U31_GEN2 2
|
||||
|
||||
/* USB Highest Speed */
|
||||
#define CABLE_USB2_ONLY 0
|
||||
#define CABLE_USB32_GEN1 1
|
||||
#define CABLE_USB32_4_GEN2 2
|
||||
#define CABLE_USB4_GEN3 3
|
||||
|
||||
#define VDO_CABLE(hw, fw, cbl, lat, term, tx1d, tx2d, rx1d, rx2d, cur, vps, sopp, usbss) \
|
||||
(((hw) & 0x7) << 28 | ((fw) & 0x7) << 24 | ((cbl) & 0x3) << 18 \
|
||||
| ((lat) & 0x7) << 13 | ((term) & 0x3) << 11 | (tx1d) << 10 \
|
||||
| (tx2d) << 9 | (rx1d) << 8 | (rx2d) << 7 | ((cur) & 0x3) << 5 \
|
||||
| (vps) << 4 | (sopp) << 3 | ((usbss) & 0x7))
|
||||
#define VDO_PCABLE(hw, fw, ver, conn, lat, term, vbm, cur, spd) \
|
||||
(((hw) & 0xf) << 28 | ((fw) & 0xf) << 24 | ((ver) & 0x7) << 21 \
|
||||
| ((conn) & 0x3) << 18 | ((lat) & 0xf) << 13 | ((term) & 0x3) << 11 \
|
||||
| ((vbm) & 0x3) << 9 | ((cur) & 0x3) << 5 | ((spd) & 0x7))
|
||||
#define VDO_ACABLE1(hw, fw, ver, conn, lat, term, vbm, sbu, sbut, cur, vbt, sopp, spd) \
|
||||
(((hw) & 0xf) << 28 | ((fw) & 0xf) << 24 | ((ver) & 0x7) << 21 \
|
||||
| ((conn) & 0x3) << 18 | ((lat) & 0xf) << 13 | ((term) & 0x3) << 11 \
|
||||
| ((vbm) & 0x3) << 9 | (sbu) << 8 | (sbut) << 7 | ((cur) & 0x3) << 5 \
|
||||
| (vbt) << 4 | (sopp) << 3 | ((spd) & 0x7))
|
||||
|
||||
#define VDO_TYPEC_CABLE_TYPE(vdo) (((vdo) >> 18) & 0x3)
|
||||
|
||||
/*
|
||||
* Active Cable VDO 2
|
||||
* ---------
|
||||
* <31:24> :: Maximum operating temperature
|
||||
* <23:16> :: Shutdown temperature
|
||||
* <15> :: Reserved, Shall be set to zero
|
||||
* <14:12> :: U3/CLd power
|
||||
* <11> :: U3 to U0 transition mode (0b == direct, 1b == through U3S)
|
||||
* <10> :: Physical connection (0b == copper, 1b == optical)
|
||||
* <9> :: Active element (0b == redriver, 1b == retimer)
|
||||
* <8> :: USB4 supported (0b == yes, 1b == no)
|
||||
* <7:6> :: USB2 hub hops consumed
|
||||
* <5> :: USB2 supported (0b == yes, 1b == no)
|
||||
* <4> :: USB3.2 supported (0b == yes, 1b == no)
|
||||
* <3> :: USB lanes supported (0b == one lane, 1b == two lanes)
|
||||
* <2> :: Optically isolated active cable (0b == no, 1b == yes)
|
||||
* <1> :: Reserved, Shall be set to zero
|
||||
* <0> :: USB gen (0b == gen1, 1b == gen2+)
|
||||
*/
|
||||
|
||||
/* U3/CLd Power*/
|
||||
#define ACAB2_U3_CLD_10MW_PLUS 0
|
||||
#define ACAB2_U3_CLD_10MW 1
|
||||
#define ACAB2_U3_CLD_5MW 2
|
||||
#define ACAB2_U3_CLD_1MW 3
|
||||
#define ACAB2_U3_CLD_500UW 4
|
||||
#define ACAB2_U3_CLD_200UW 5
|
||||
#define ACAB2_U3_CLD_50UW 6
|
||||
|
||||
/* Other Active Cable VDO 2 Fields */
|
||||
#define ACAB2_U3U0_DIRECT 0
|
||||
#define ACAB2_U3U0_U3S 1
|
||||
#define ACAB2_PHY_COPPER 0
|
||||
#define ACAB2_PHY_OPTICAL 1
|
||||
#define ACAB2_REDRIVER 0
|
||||
#define ACAB2_RETIMER 1
|
||||
#define ACAB2_USB4_SUPP 0
|
||||
#define ACAB2_USB4_NOT_SUPP 1
|
||||
#define ACAB2_USB2_SUPP 0
|
||||
#define ACAB2_USB2_NOT_SUPP 1
|
||||
#define ACAB2_USB32_SUPP 0
|
||||
#define ACAB2_USB32_NOT_SUPP 1
|
||||
#define ACAB2_LANES_ONE 0
|
||||
#define ACAB2_LANES_TWO 1
|
||||
#define ACAB2_OPT_ISO_NO 0
|
||||
#define ACAB2_OPT_ISO_YES 1
|
||||
#define ACAB2_GEN_1 0
|
||||
#define ACAB2_GEN_2_PLUS 1
|
||||
|
||||
#define VDO_ACABLE2(mtemp, stemp, u3p, trans, phy, ele, u4, hops, u2, u32, lane, iso, gen) \
|
||||
(((mtemp) & 0xff) << 24 | ((stemp) & 0xff) << 16 | ((u3p) & 0x7) << 12 \
|
||||
| (trans) << 11 | (phy) << 10 | (ele) << 9 | (u4) << 8 \
|
||||
| ((hops) & 0x3) << 6 | (u2) << 5 | (u32) << 4 | (lane) << 3 \
|
||||
| (iso) << 2 | (gen))
|
||||
|
||||
/*
|
||||
* AMA VDO (PD Rev2.0)
|
||||
* ---------
|
||||
* <31:28> :: Cable HW version
|
||||
* <27:24> :: Cable FW version
|
||||
* <23:12> :: Reserved, Shall be set to zero
|
||||
* <11> :: SSTX1 Directionality support (0b == fixed, 1b == cfgable)
|
||||
* <10> :: SSTX2 Directionality support
|
||||
* <9> :: SSRX1 Directionality support
|
||||
* <8> :: SSRX2 Directionality support
|
||||
* <7:5> :: Vconn power
|
||||
* <4> :: Vconn power required
|
||||
* <3> :: Vbus power required
|
||||
* <2:0> :: USB SS Signaling support
|
||||
*/
|
||||
#define VDO_AMA(hw, fw, tx1d, tx2d, rx1d, rx2d, vcpwr, vcr, vbr, usbss) \
|
||||
(((hw) & 0x7) << 28 | ((fw) & 0x7) << 24 \
|
||||
| (tx1d) << 11 | (tx2d) << 10 | (rx1d) << 9 | (rx2d) << 8 \
|
||||
| ((vcpwr) & 0x7) << 5 | (vcr) << 4 | (vbr) << 3 \
|
||||
| ((usbss) & 0x7))
|
||||
|
||||
#define PD_VDO_AMA_VCONN_REQ(vdo) (((vdo) >> 4) & 1)
|
||||
#define PD_VDO_AMA_VBUS_REQ(vdo) (((vdo) >> 3) & 1)
|
||||
|
||||
#define AMA_USBSS_U2_ONLY 0
|
||||
#define AMA_USBSS_U31_GEN1 1
|
||||
#define AMA_USBSS_U31_GEN2 2
|
||||
#define AMA_USBSS_BBONLY 3
|
||||
|
||||
/*
|
||||
* VPD VDO
|
||||
* ---------
|
||||
* <31:28> :: HW version
|
||||
* <27:24> :: FW version
|
||||
* <23:21> :: VDO version
|
||||
* <20:17> :: Reserved, Shall be set to zero
|
||||
* <16:15> :: Maximum Vbus voltage (00b == 20V, 01b == 30V, 10b == 40V, 11b == 50V)
|
||||
* <14> :: Charge through current support (0b == 3A, 1b == 5A)
|
||||
* <13> :: Reserved, Shall be set to zero
|
||||
* <12:7> :: Vbus impedance
|
||||
* <6:1> :: Ground impedance
|
||||
* <0> :: Charge through support (0b == no, 1b == yes)
|
||||
*/
|
||||
#define VPD_VDO_VER1_0 0
|
||||
#define VPD_MAX_VBUS_20V 0
|
||||
#define VPD_MAX_VBUS_30V 1
|
||||
#define VPD_MAX_VBUS_40V 2
|
||||
#define VPD_MAX_VBUS_50V 3
|
||||
#define VPDCT_CURR_3A 0
|
||||
#define VPDCT_CURR_5A 1
|
||||
#define VPDCT_NOT_SUPP 0
|
||||
#define VPDCT_SUPP 1
|
||||
|
||||
#define VDO_VPD(hw, fw, ver, vbm, curr, vbi, gi, ct) \
|
||||
(((hw) & 0xf) << 28 | ((fw) & 0xf) << 24 | ((ver) & 0x7) << 21 \
|
||||
| ((vbm) & 0x3) << 15 | (curr) << 14 | ((vbi) & 0x3f) << 7 \
|
||||
| ((gi) & 0x3f) << 1 | (ct))
|
||||
|
||||
/*
|
||||
* SVDM Discover SVIDs request -> response
|
||||
*
|
||||
* Request is properly formatted VDM Header with discover SVIDs command.
|
||||
* Response is a set of SVIDs of all supported SVIDs with all zero's to
|
||||
* mark the end of SVIDs. If more than 12 SVIDs are supported command SHOULD be
|
||||
* repeated.
|
||||
*/
|
||||
#define VDO_SVID(svid0, svid1) (((svid0) & 0xffff) << 16 | ((svid1) & 0xffff))
|
||||
#define PD_VDO_SVID_SVID0(vdo) ((vdo) >> 16)
|
||||
#define PD_VDO_SVID_SVID1(vdo) ((vdo) & 0xffff)
|
||||
|
||||
/* USB-IF SIDs */
|
||||
#define USB_SID_PD 0xff00 /* power delivery */
|
||||
#define USB_SID_DISPLAYPORT 0xff01
|
||||
#define USB_SID_MHL 0xff02 /* Mobile High-Definition Link */
|
||||
|
||||
/* VDM command timeouts (in ms) */
|
||||
|
||||
#define PD_T_VDM_UNSTRUCTURED 500
|
||||
#define PD_T_VDM_BUSY 100
|
||||
#define PD_T_VDM_WAIT_MODE_E 100
|
||||
#define PD_T_VDM_SNDR_RSP 30
|
||||
#define PD_T_VDM_E_MODE 25
|
||||
#define PD_T_VDM_RCVR_RSP 15
|
||||
|
||||
#endif /* __LINUX_USB_PD_VDO_H */
|
||||
36
include/power/power_delivery/power_delivery.h
Executable file
36
include/power/power_delivery/power_delivery.h
Executable file
@@ -0,0 +1,36 @@
|
||||
/*
|
||||
* (C) Copyright 2020 Rockchip Electronics Co., Ltd
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-2.0+
|
||||
*/
|
||||
|
||||
#ifndef _power_delivery_H_
|
||||
#define _power_delivery_H_
|
||||
|
||||
/* Capability */
|
||||
#define FG_CAP_FUEL_GAUGE BIT(0)
|
||||
#define FG_CAP_CHARGER BIT(1)
|
||||
|
||||
struct dm_power_delivery_ops {
|
||||
int (*get_voltage)(struct udevice *dev);
|
||||
int (*get_current)(struct udevice *dev);
|
||||
int (*get_online)(struct udevice *dev);
|
||||
int (*get_vbus_power)(struct udevice *dev);
|
||||
};
|
||||
|
||||
struct power_delivery_data {
|
||||
int voltage;
|
||||
int current;
|
||||
int online;
|
||||
};
|
||||
int power_delivery_get_vbus(struct udevice *dev, int *status);
|
||||
#ifdef CONFIG_DM_POWER_DELIVERY
|
||||
int power_delivery_get_data(struct udevice *dev, struct power_delivery_data *pd_data);
|
||||
#else
|
||||
inline int power_delivery_get_data(struct udevice *dev,
|
||||
struct power_delivery_data *pd_data)
|
||||
{
|
||||
return -ENOSYS;
|
||||
}
|
||||
#endif
|
||||
#endif
|
||||
195
include/power/power_delivery/tcpm.h
Executable file
195
include/power/power_delivery/tcpm.h
Executable file
@@ -0,0 +1,195 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
/*
|
||||
* Copyright 2015-2017 Google, Inc
|
||||
*/
|
||||
|
||||
#ifndef __LINUX_USB_TCPM_H
|
||||
#define __LINUX_USB_TCPM_H
|
||||
|
||||
#include <linux/bitops.h>
|
||||
#include "typec.h"
|
||||
#include "pd.h"
|
||||
|
||||
enum typec_cc_status {
|
||||
TYPEC_CC_OPEN,
|
||||
TYPEC_CC_RA,
|
||||
TYPEC_CC_RD,
|
||||
TYPEC_CC_RP_DEF,
|
||||
TYPEC_CC_RP_1_5,
|
||||
TYPEC_CC_RP_3_0,
|
||||
};
|
||||
|
||||
/* Collision Avoidance */
|
||||
#define SINK_TX_NG TYPEC_CC_RP_1_5
|
||||
#define SINK_TX_OK TYPEC_CC_RP_3_0
|
||||
|
||||
enum typec_cc_polarity {
|
||||
TYPEC_POLARITY_CC1,
|
||||
TYPEC_POLARITY_CC2,
|
||||
};
|
||||
|
||||
/* Time to wait for TCPC to complete transmit */
|
||||
#define PD_T_TCPC_TX_TIMEOUT 100 /* in ms */
|
||||
#define PD_ROLE_SWAP_TIMEOUT (MSEC_PER_SEC * 10)
|
||||
#define PD_PPS_CTRL_TIMEOUT (MSEC_PER_SEC * 10)
|
||||
|
||||
enum tcpm_transmit_status {
|
||||
TCPC_TX_SUCCESS = 0,
|
||||
TCPC_TX_DISCARDED = 1,
|
||||
TCPC_TX_FAILED = 2,
|
||||
};
|
||||
|
||||
enum tcpm_transmit_type {
|
||||
TCPC_TX_SOP = 0,
|
||||
TCPC_TX_SOP_PRIME = 1,
|
||||
TCPC_TX_SOP_PRIME_PRIME = 2,
|
||||
TCPC_TX_SOP_DEBUG_PRIME = 3,
|
||||
TCPC_TX_SOP_DEBUG_PRIME_PRIME = 4,
|
||||
TCPC_TX_HARD_RESET = 5,
|
||||
TCPC_TX_CABLE_RESET = 6,
|
||||
TCPC_TX_BIST_MODE_2 = 7
|
||||
};
|
||||
|
||||
/* Mux state attributes */
|
||||
#define TCPC_MUX_USB_ENABLED BIT(0) /* USB enabled */
|
||||
#define TCPC_MUX_DP_ENABLED BIT(1) /* DP enabled */
|
||||
#define TCPC_MUX_POLARITY_INVERTED BIT(2) /* Polarity inverted */
|
||||
|
||||
/**
|
||||
* struct tcpc_dev - Port configuration and callback functions
|
||||
* @fwnode: Pointer to port fwnode
|
||||
* @get_vbus: Called to read current VBUS state
|
||||
* @get_current_limit:
|
||||
* Optional; called by the tcpm core when configured as a snk
|
||||
* and cc=Rp-def. This allows the tcpm to provide a fallback
|
||||
* current-limit detection method for the cc=Rp-def case.
|
||||
* For example, some tcpcs may include BC1.2 charger detection
|
||||
* and use that in this case.
|
||||
* @set_cc: Called to set value of CC pins
|
||||
* @apply_rc: Optional; Needed to move TCPCI based chipset to APPLY_RC state
|
||||
* as stated by the TCPCI specification.
|
||||
* @get_cc: Called to read current CC pin values
|
||||
* @set_polarity:
|
||||
* Called to set polarity
|
||||
* @set_vconn: Called to enable or disable VCONN
|
||||
* @set_vbus: Called to enable or disable VBUS
|
||||
* @set_current_limit:
|
||||
* Optional; called to set current limit as negotiated
|
||||
* with partner.
|
||||
* @set_pd_rx: Called to enable or disable reception of PD messages
|
||||
* @set_roles: Called to set power and data roles
|
||||
* @start_toggling:
|
||||
* Optional; if supported by hardware, called to start dual-role
|
||||
* toggling or single-role connection detection. Toggling stops
|
||||
* automatically if a connection is established.
|
||||
* @try_role: Optional; called to set a preferred role
|
||||
* @pd_transmit:Called to transmit PD message
|
||||
* @set_bist_data: Turn on/off bist data mode for compliance testing
|
||||
* @enable_frs:
|
||||
* Optional; Called to enable/disable PD 3.0 fast role swap.
|
||||
* Enabling frs is accessory dependent as not all PD3.0
|
||||
* accessories support fast role swap.
|
||||
* @frs_sourcing_vbus:
|
||||
* Optional; Called to notify that vbus is now being sourced.
|
||||
* Low level drivers can perform chip specific operations, if any.
|
||||
* @enable_auto_vbus_discharge:
|
||||
* Optional; TCPCI spec based TCPC implementations can optionally
|
||||
* support hardware to autonomously dischrge vbus upon disconnecting
|
||||
* as sink or source. TCPM signals TCPC to enable the mechanism upon
|
||||
* entering connected state and signals disabling upon disconnect.
|
||||
* @set_auto_vbus_discharge_threshold:
|
||||
* Mandatory when enable_auto_vbus_discharge is implemented. TCPM
|
||||
* calls this function to allow lower levels drivers to program the
|
||||
* vbus threshold voltage below which the vbus discharge circuit
|
||||
* will be turned on. requested_vbus_voltage is set to 0 when vbus
|
||||
* is going to disappear knowingly i.e. during PR_SWAP and
|
||||
* HARD_RESET etc.
|
||||
* @is_vbus_vsafe0v:
|
||||
* Optional; TCPCI spec based TCPC implementations are expected to
|
||||
* detect VSAFE0V voltage level at vbus. When detection of VSAFE0V
|
||||
* is supported by TCPC, set this callback for TCPM to query
|
||||
* whether vbus is at VSAFE0V when needed.
|
||||
* Returns true when vbus is at VSAFE0V, false otherwise.
|
||||
* @set_partner_usb_comm_capable:
|
||||
* Optional; The USB Communications Capable bit indicates if port
|
||||
* partner is capable of communication over the USB data lines
|
||||
* (e.g. D+/- or SS Tx/Rx). Called to notify the status of the bit.
|
||||
* @check_contaminant:
|
||||
* Optional; The callback is called when CC pins report open status
|
||||
* at the end of the toggling period. Chip level drivers are
|
||||
* expected to check for contaminant and re-enable toggling if
|
||||
* needed. When 0 is not returned, check_contaminant is expected to
|
||||
* restart toggling after checking the connector for contaminant.
|
||||
* This forces the TCPM state machine to tranistion to TOGGLING state
|
||||
* without calling start_toggling callback.
|
||||
* @poll_event:
|
||||
* After the PD chip driver is loaded, the callback function will be
|
||||
* called to poll what events have been triggered.
|
||||
* @enter_low_power_mode:
|
||||
* Optional; the pd chip enters low power mode.
|
||||
*/
|
||||
struct tcpc_dev {
|
||||
ofnode connector_node;
|
||||
int (*init)(struct tcpc_dev *dev);
|
||||
int (*get_vbus)(struct tcpc_dev *dev);
|
||||
int (*get_current_limit)(struct tcpc_dev *dev);
|
||||
int (*set_cc)(struct tcpc_dev *dev, enum typec_cc_status cc);
|
||||
int (*apply_rc)(struct tcpc_dev *dev, enum typec_cc_status cc,
|
||||
enum typec_cc_polarity polarity);
|
||||
int (*get_cc)(struct tcpc_dev *dev, enum typec_cc_status *cc1,
|
||||
enum typec_cc_status *cc2);
|
||||
int (*set_polarity)(struct tcpc_dev *dev,
|
||||
enum typec_cc_polarity polarity);
|
||||
int (*set_vconn)(struct tcpc_dev *dev, bool on);
|
||||
int (*set_vbus)(struct tcpc_dev *dev, bool on, bool charge);
|
||||
int (*set_current_limit)(struct tcpc_dev *dev, u32 max_ma, u32 mv);
|
||||
int (*set_pd_rx)(struct tcpc_dev *dev, bool on);
|
||||
int (*set_roles)(struct tcpc_dev *dev, bool attached,
|
||||
enum typec_role role, enum typec_data_role data);
|
||||
int (*start_toggling)(struct tcpc_dev *dev,
|
||||
enum typec_port_type port_type,
|
||||
enum typec_cc_status cc);
|
||||
int (*try_role)(struct tcpc_dev *dev, int role);
|
||||
int (*pd_transmit)(struct tcpc_dev *dev, enum tcpm_transmit_type type,
|
||||
const struct pd_message *msg, unsigned int negotiated_rev);
|
||||
int (*set_bist_data)(struct tcpc_dev *dev, bool on);
|
||||
int (*enable_frs)(struct tcpc_dev *dev, bool enable);
|
||||
void (*frs_sourcing_vbus)(struct tcpc_dev *dev);
|
||||
int (*enable_auto_vbus_discharge)(struct tcpc_dev *dev, bool enable);
|
||||
int (*set_auto_vbus_discharge_threshold)(struct tcpc_dev *dev, enum typec_pwr_opmode mode,
|
||||
bool pps_active, u32 requested_vbus_voltage);
|
||||
int (*check_contaminant)(struct tcpc_dev *dev);
|
||||
bool (*is_vbus_vsafe0v)(struct tcpc_dev *dev);
|
||||
void (*set_partner_usb_comm_capable)(struct tcpc_dev *dev, bool enable);
|
||||
void (*poll_event)(struct tcpc_dev *dev);
|
||||
int (*enter_low_power_mode)(struct tcpc_dev *dev, bool attached, bool pd_capable);
|
||||
};
|
||||
|
||||
struct tcpm_port;
|
||||
|
||||
struct tcpm_port *tcpm_port_init(struct udevice *dev, struct tcpc_dev *tcpc);
|
||||
void tcpm_poll_event(struct tcpm_port *port);
|
||||
int tcpm_get_voltage(struct tcpm_port *port);
|
||||
int tcpm_get_current(struct tcpm_port *port);
|
||||
int tcpm_get_online(struct tcpm_port *port);
|
||||
int tcpm_get_vbus(struct tcpm_port *port);
|
||||
void tcpm_uninit_port(struct tcpm_port *port);
|
||||
|
||||
int tcpm_update_sink_capabilities(struct tcpm_port *port, const u32 *pdo,
|
||||
unsigned int nr_pdo,
|
||||
unsigned int operating_snk_mw);
|
||||
|
||||
void tcpm_vbus_change(struct tcpm_port *port);
|
||||
void tcpm_cc_change(struct tcpm_port *port);
|
||||
void tcpm_sink_frs(struct tcpm_port *port);
|
||||
void tcpm_sourcing_vbus(struct tcpm_port *port);
|
||||
void tcpm_pd_receive(struct tcpm_port *port,
|
||||
const struct pd_message *msg);
|
||||
void tcpm_pd_transmit_complete(struct tcpm_port *port,
|
||||
enum tcpm_transmit_status status);
|
||||
void tcpm_pd_hard_reset(struct tcpm_port *port);
|
||||
void tcpm_tcpc_reset(struct tcpm_port *port);
|
||||
bool tcpm_is_debouncing(struct tcpm_port *tcpm);
|
||||
bool tcpm_is_toggling(struct tcpm_port *port);
|
||||
|
||||
#endif /* __LINUX_USB_TCPM_H */
|
||||
315
include/power/power_delivery/typec.h
Executable file
315
include/power/power_delivery/typec.h
Executable file
@@ -0,0 +1,315 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
|
||||
#ifndef __LINUX_USB_TYPEC_H
|
||||
#define __LINUX_USB_TYPEC_H
|
||||
|
||||
#include <linux/types.h>
|
||||
|
||||
/* USB Type-C Specification releases */
|
||||
#define USB_TYPEC_REV_1_0 0x100 /* 1.0 */
|
||||
#define USB_TYPEC_REV_1_1 0x110 /* 1.1 */
|
||||
#define USB_TYPEC_REV_1_2 0x120 /* 1.2 */
|
||||
#define USB_TYPEC_REV_1_3 0x130 /* 1.3 */
|
||||
#define USB_TYPEC_REV_1_4 0x140 /* 1.4 */
|
||||
#define USB_TYPEC_REV_2_0 0x200 /* 2.0 */
|
||||
|
||||
struct typec_partner;
|
||||
struct typec_cable;
|
||||
struct typec_plug;
|
||||
struct typec_port;
|
||||
struct typec_altmode_ops;
|
||||
|
||||
struct fwnode_handle;
|
||||
struct device;
|
||||
|
||||
enum typec_port_type {
|
||||
TYPEC_PORT_SRC,
|
||||
TYPEC_PORT_SNK,
|
||||
TYPEC_PORT_DRP,
|
||||
};
|
||||
|
||||
enum typec_port_data {
|
||||
TYPEC_PORT_DFP,
|
||||
TYPEC_PORT_UFP,
|
||||
TYPEC_PORT_DRD,
|
||||
};
|
||||
|
||||
enum typec_plug_type {
|
||||
USB_PLUG_NONE,
|
||||
USB_PLUG_TYPE_A,
|
||||
USB_PLUG_TYPE_B,
|
||||
USB_PLUG_TYPE_C,
|
||||
USB_PLUG_CAPTIVE,
|
||||
};
|
||||
|
||||
enum typec_data_role {
|
||||
TYPEC_DEVICE,
|
||||
TYPEC_HOST,
|
||||
};
|
||||
|
||||
enum typec_role {
|
||||
TYPEC_SINK,
|
||||
TYPEC_SOURCE,
|
||||
};
|
||||
|
||||
enum typec_pwr_opmode {
|
||||
TYPEC_PWR_MODE_USB,
|
||||
TYPEC_PWR_MODE_1_5A,
|
||||
TYPEC_PWR_MODE_3_0A,
|
||||
TYPEC_PWR_MODE_PD,
|
||||
};
|
||||
|
||||
enum typec_accessory {
|
||||
TYPEC_ACCESSORY_NONE,
|
||||
TYPEC_ACCESSORY_AUDIO,
|
||||
TYPEC_ACCESSORY_DEBUG,
|
||||
};
|
||||
|
||||
#define TYPEC_MAX_ACCESSORY 3
|
||||
|
||||
enum typec_orientation {
|
||||
TYPEC_ORIENTATION_NONE,
|
||||
TYPEC_ORIENTATION_NORMAL,
|
||||
TYPEC_ORIENTATION_REVERSE,
|
||||
};
|
||||
|
||||
/*
|
||||
* struct enter_usb_data - Enter_USB Message details
|
||||
* @eudo: Enter_USB Data Object
|
||||
* @active_link_training: Active Cable Plug Link Training
|
||||
*
|
||||
* @active_link_training is a flag that should be set with uni-directional SBRX
|
||||
* communication, and left 0 with passive cables and with bi-directional SBRX
|
||||
* communication.
|
||||
*/
|
||||
struct enter_usb_data {
|
||||
u32 eudo;
|
||||
unsigned char active_link_training:1;
|
||||
};
|
||||
|
||||
/*
|
||||
* struct usb_pd_identity - USB Power Delivery identity data
|
||||
* @id_header: ID Header VDO
|
||||
* @cert_stat: Cert Stat VDO
|
||||
* @product: Product VDO
|
||||
* @vdo: Product Type Specific VDOs
|
||||
*
|
||||
* USB power delivery Discover Identity command response data.
|
||||
*
|
||||
* REVISIT: This is USB Power Delivery specific information, so this structure
|
||||
* probable belongs to USB Power Delivery header file once we have them.
|
||||
*/
|
||||
struct usb_pd_identity {
|
||||
u32 id_header;
|
||||
u32 cert_stat;
|
||||
u32 product;
|
||||
u32 vdo[3];
|
||||
};
|
||||
|
||||
int typec_partner_set_identity(struct typec_partner *partner);
|
||||
int typec_cable_set_identity(struct typec_cable *cable);
|
||||
|
||||
/*
|
||||
* struct typec_altmode_desc - USB Type-C Alternate Mode Descriptor
|
||||
* @svid: Standard or Vendor ID
|
||||
* @mode: Index of the Mode
|
||||
* @vdo: VDO returned by Discover Modes USB PD command
|
||||
* @roles: Only for ports. DRP if the mode is available in both roles
|
||||
*
|
||||
* Description of an Alternate Mode which a connector, cable plug or partner
|
||||
* supports.
|
||||
*/
|
||||
struct typec_altmode_desc {
|
||||
u16 svid;
|
||||
u8 mode;
|
||||
u32 vdo;
|
||||
/* Only used with ports */
|
||||
enum typec_port_data roles;
|
||||
};
|
||||
|
||||
void typec_partner_set_pd_revision(struct typec_partner *partner, u16 pd_revision);
|
||||
int typec_partner_set_num_altmodes(struct typec_partner *partner, int num_altmodes);
|
||||
struct typec_altmode
|
||||
*typec_partner_register_altmode(struct typec_partner *partner,
|
||||
const struct typec_altmode_desc *desc);
|
||||
int typec_plug_set_num_altmodes(struct typec_plug *plug, int num_altmodes);
|
||||
struct typec_altmode
|
||||
*typec_plug_register_altmode(struct typec_plug *plug,
|
||||
const struct typec_altmode_desc *desc);
|
||||
struct typec_altmode
|
||||
*typec_port_register_altmode(struct typec_port *port,
|
||||
const struct typec_altmode_desc *desc);
|
||||
|
||||
#ifdef CONFIG_NO_GKI
|
||||
void typec_port_register_altmodes(struct typec_port *port,
|
||||
const struct typec_altmode_ops *ops, void *drvdata,
|
||||
struct typec_altmode **altmodes, size_t n);
|
||||
#else
|
||||
static inline void typec_port_register_altmodes(struct typec_port *port,
|
||||
const struct typec_altmode_ops *ops, void *drvdata,
|
||||
struct typec_altmode **altmodes, size_t n)
|
||||
{
|
||||
}
|
||||
#endif
|
||||
|
||||
void typec_unregister_altmode(struct typec_altmode *altmode);
|
||||
|
||||
struct typec_port *typec_altmode2port(struct typec_altmode *alt);
|
||||
|
||||
void typec_altmode_update_active(struct typec_altmode *alt, bool active);
|
||||
|
||||
enum typec_plug_index {
|
||||
TYPEC_PLUG_SOP_P,
|
||||
TYPEC_PLUG_SOP_PP,
|
||||
};
|
||||
|
||||
/*
|
||||
* struct typec_plug_desc - USB Type-C Cable Plug Descriptor
|
||||
* @index: SOP Prime for the plug connected to DFP and SOP Double Prime for the
|
||||
* plug connected to UFP
|
||||
*
|
||||
* Represents USB Type-C Cable Plug.
|
||||
*/
|
||||
struct typec_plug_desc {
|
||||
enum typec_plug_index index;
|
||||
};
|
||||
|
||||
/*
|
||||
* struct typec_cable_desc - USB Type-C Cable Descriptor
|
||||
* @type: The plug type from USB PD Cable VDO
|
||||
* @active: Is the cable active or passive
|
||||
* @identity: Result of Discover Identity command
|
||||
* @pd_revision: USB Power Delivery Specification revision if supported
|
||||
*
|
||||
* Represents USB Type-C Cable attached to USB Type-C port.
|
||||
*/
|
||||
struct typec_cable_desc {
|
||||
enum typec_plug_type type;
|
||||
unsigned int active:1;
|
||||
struct usb_pd_identity *identity;
|
||||
u16 pd_revision; /* 0300H = "3.0" */
|
||||
|
||||
};
|
||||
|
||||
/*
|
||||
* struct typec_partner_desc - USB Type-C Partner Descriptor
|
||||
* @usb_pd: USB Power Delivery support
|
||||
* @accessory: Audio, Debug or none.
|
||||
* @identity: Discover Identity command data
|
||||
* @pd_revision: USB Power Delivery Specification Revision if supported
|
||||
*
|
||||
* Details about a partner that is attached to USB Type-C port. If @identity
|
||||
* member exists when partner is registered, a directory named "identity" is
|
||||
* created to sysfs for the partner device.
|
||||
*
|
||||
* @pd_revision is based on the setting of the "Specification Revision" field
|
||||
* in the message header on the initial "Source Capabilities" message received
|
||||
* from the partner, or a "Request" message received from the partner, depending
|
||||
* on whether our port is a Sink or a Source.
|
||||
*/
|
||||
struct typec_partner_desc {
|
||||
unsigned int usb_pd:1;
|
||||
enum typec_accessory accessory;
|
||||
struct usb_pd_identity *identity;
|
||||
u16 pd_revision; /* 0300H = "3.0" */
|
||||
};
|
||||
|
||||
/**
|
||||
* struct typec_operations - USB Type-C Port Operations
|
||||
* @try_role: Set data role preference for DRP port
|
||||
* @dr_set: Set Data Role
|
||||
* @pr_set: Set Power Role
|
||||
* @vconn_set: Source VCONN
|
||||
* @port_type_set: Set port type
|
||||
*/
|
||||
struct typec_operations {
|
||||
int (*try_role)(struct typec_port *port, int role);
|
||||
int (*dr_set)(struct typec_port *port, enum typec_data_role role);
|
||||
int (*pr_set)(struct typec_port *port, enum typec_role role);
|
||||
int (*vconn_set)(struct typec_port *port, enum typec_role role);
|
||||
int (*port_type_set)(struct typec_port *port,
|
||||
enum typec_port_type type);
|
||||
};
|
||||
|
||||
enum usb_pd_svdm_ver {
|
||||
SVDM_VER_1_0 = 0,
|
||||
SVDM_VER_2_0 = 1,
|
||||
SVDM_VER_MAX = SVDM_VER_2_0,
|
||||
};
|
||||
|
||||
/*
|
||||
* struct typec_capability - USB Type-C Port Capabilities
|
||||
* @type: Supported power role of the port
|
||||
* @data: Supported data role of the port
|
||||
* @revision: USB Type-C Specification release. Binary coded decimal
|
||||
* @pd_revision: USB Power Delivery Specification revision if supported
|
||||
* @svdm_version: USB PD Structured VDM version if supported
|
||||
* @prefer_role: Initial role preference (DRP ports).
|
||||
* @accessory: Supported Accessory Modes
|
||||
* @fwnode: Optional fwnode of the port
|
||||
* @driver_data: Private pointer for driver specific info
|
||||
* @ops: Port operations vector
|
||||
*
|
||||
* Static capabilities of a single USB Type-C port.
|
||||
*/
|
||||
struct typec_capability {
|
||||
enum typec_port_type type;
|
||||
enum typec_port_data data;
|
||||
u16 revision; /* 0120H = "1.2" */
|
||||
u16 pd_revision; /* 0300H = "3.0" */
|
||||
enum usb_pd_svdm_ver svdm_version;
|
||||
int prefer_role;
|
||||
enum typec_accessory accessory[TYPEC_MAX_ACCESSORY];
|
||||
unsigned int orientation_aware:1;
|
||||
|
||||
struct fwnode_handle *fwnode;
|
||||
void *driver_data;
|
||||
|
||||
const struct typec_operations *ops;
|
||||
};
|
||||
|
||||
/* Specific to try_role(). Indicates the user want's to clear the preference. */
|
||||
#define TYPEC_NO_PREFERRED_ROLE (-1)
|
||||
|
||||
struct typec_port *typec_register_port(struct device *parent,
|
||||
const struct typec_capability *cap);
|
||||
void typec_unregister_port(struct typec_port *port);
|
||||
|
||||
struct typec_partner *typec_register_partner(struct typec_port *port,
|
||||
struct typec_partner_desc *desc);
|
||||
void typec_unregister_partner(struct typec_partner *partner);
|
||||
|
||||
struct typec_cable *typec_register_cable(struct typec_port *port,
|
||||
struct typec_cable_desc *desc);
|
||||
void typec_unregister_cable(struct typec_cable *cable);
|
||||
|
||||
struct typec_cable *typec_cable_get(struct typec_port *port);
|
||||
void typec_cable_put(struct typec_cable *cable);
|
||||
int typec_cable_is_active(struct typec_cable *cable);
|
||||
|
||||
struct typec_plug *typec_register_plug(struct typec_cable *cable,
|
||||
struct typec_plug_desc *desc);
|
||||
void typec_unregister_plug(struct typec_plug *plug);
|
||||
|
||||
void typec_set_data_role(struct typec_port *port, enum typec_data_role role);
|
||||
void typec_set_pwr_role(struct typec_port *port, enum typec_role role);
|
||||
void typec_set_vconn_role(struct typec_port *port, enum typec_role role);
|
||||
void typec_set_pwr_opmode(struct typec_port *port, enum typec_pwr_opmode mode);
|
||||
|
||||
int typec_set_orientation(struct typec_port *port,
|
||||
enum typec_orientation orientation);
|
||||
enum typec_orientation typec_get_orientation(struct typec_port *port);
|
||||
int typec_set_mode(struct typec_port *port, int mode);
|
||||
|
||||
void *typec_get_drvdata(struct typec_port *port);
|
||||
|
||||
int typec_find_pwr_opmode(const char *name);
|
||||
int typec_find_orientation(const char *name);
|
||||
int typec_find_port_power_role(const char *name);
|
||||
int typec_find_power_role(const char *name);
|
||||
int typec_find_port_data_role(const char *name);
|
||||
|
||||
void typec_partner_set_svdm_version(struct typec_partner *partner,
|
||||
enum usb_pd_svdm_ver svdm_version);
|
||||
int typec_get_negotiated_svdm_version(struct typec_port *port);
|
||||
#endif /* __LINUX_USB_TYPEC_H */
|
||||
Reference in New Issue
Block a user