Skip to content

Commit

Permalink
usb: typec: hd3ss3220: support configuring port type
Browse files Browse the repository at this point in the history
The TI HD3SS3220 Type-C controller supports configuring the port type
it will operate as through the MODE_SELECT field of the General
Control Register.

Configure the port type based on the fwnode property "power-role"
during probe, if present. If the property is absent, leave the
operation mode at the default, which is defined by the PORT pin
of the chip.
Support configuring the port type through the port_type_set
typec_operation as well.

The MODE_SELECT field can only be changed when the controller is in
unattached state, so follow the sequence recommended by the datasheet to:
1. disable termination on CC pins to disable the controller
2. change the mode
3. re-enable termination

This will effectively cause a connected device to disconnect
for the duration of the mode change.

Signed-off-by: Oliver Facklam <oliver.facklam@zuehlke.com>
Reviewed-by: Heikki Krogerus <heikki.krogerus@linux.intel.com>
Link: https://lore.kernel.org/r/20241211-usb-typec-controller-enhancements-v3-2-e4bc1b6e1441@zuehlke.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
  • Loading branch information
Oliver Facklam authored and Greg Kroah-Hartman committed Dec 24, 2024
1 parent 14ba185 commit 5d2c32d
Showing 1 changed file with 87 additions and 1 deletion.
88 changes: 87 additions & 1 deletion drivers/usb/typec/hd3ss3220.c
Original file line number Diff line number Diff line change
Expand Up @@ -35,10 +35,16 @@
#define HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS BIT(4)

/* Register HD3SS3220_REG_GEN_CTRL*/
#define HD3SS3220_REG_GEN_CTRL_DISABLE_TERM BIT(0)
#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_MASK (BIT(2) | BIT(1))
#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT 0x00
#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SNK BIT(1)
#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SRC (BIT(2) | BIT(1))
#define HD3SS3220_REG_GEN_CTRL_MODE_SELECT_MASK (BIT(5) | BIT(4))
#define HD3SS3220_REG_GEN_CTRL_MODE_SELECT_DEFAULT 0x00
#define HD3SS3220_REG_GEN_CTRL_MODE_SELECT_DFP BIT(5)
#define HD3SS3220_REG_GEN_CTRL_MODE_SELECT_UFP BIT(4)
#define HD3SS3220_REG_GEN_CTRL_MODE_SELECT_DRP (BIT(5) | BIT(4))

struct hd3ss3220 {
struct device *dev;
Expand Down Expand Up @@ -75,6 +81,52 @@ static int hd3ss3220_set_power_opmode(struct hd3ss3220 *hd3ss3220, int power_opm
current_mode);
}

static int hd3ss3220_set_port_type(struct hd3ss3220 *hd3ss3220, int type)
{
int mode_select, err;

switch (type) {
case TYPEC_PORT_SRC:
mode_select = HD3SS3220_REG_GEN_CTRL_MODE_SELECT_DFP;
break;
case TYPEC_PORT_SNK:
mode_select = HD3SS3220_REG_GEN_CTRL_MODE_SELECT_UFP;
break;
case TYPEC_PORT_DRP:
mode_select = HD3SS3220_REG_GEN_CTRL_MODE_SELECT_DRP;
break;
default:
dev_err(hd3ss3220->dev, "bad port type: %d\n", type);
return -EINVAL;
}

/* Disable termination before changing MODE_SELECT as required by datasheet */
err = regmap_update_bits(hd3ss3220->regmap, HD3SS3220_REG_GEN_CTRL,
HD3SS3220_REG_GEN_CTRL_DISABLE_TERM,
HD3SS3220_REG_GEN_CTRL_DISABLE_TERM);
if (err < 0) {
dev_err(hd3ss3220->dev, "Failed to disable port for mode change: %d\n", err);
return err;
}

err = regmap_update_bits(hd3ss3220->regmap, HD3SS3220_REG_GEN_CTRL,
HD3SS3220_REG_GEN_CTRL_MODE_SELECT_MASK,
mode_select);
if (err < 0) {
dev_err(hd3ss3220->dev, "Failed to change mode: %d\n", err);
regmap_update_bits(hd3ss3220->regmap, HD3SS3220_REG_GEN_CTRL,
HD3SS3220_REG_GEN_CTRL_DISABLE_TERM, 0);
return err;
}

err = regmap_update_bits(hd3ss3220->regmap, HD3SS3220_REG_GEN_CTRL,
HD3SS3220_REG_GEN_CTRL_DISABLE_TERM, 0);
if (err < 0)
dev_err(hd3ss3220->dev, "Failed to re-enable port after mode change: %d\n", err);

return err;
}

static int hd3ss3220_set_source_pref(struct hd3ss3220 *hd3ss3220, int src_pref)
{
return regmap_update_bits(hd3ss3220->regmap, HD3SS3220_REG_GEN_CTRL,
Expand Down Expand Up @@ -131,8 +183,16 @@ static int hd3ss3220_dr_set(struct typec_port *port, enum typec_data_role role)
return ret;
}

static int hd3ss3220_port_type_set(struct typec_port *port, enum typec_port_type type)
{
struct hd3ss3220 *hd3ss3220 = typec_get_drvdata(port);

return hd3ss3220_set_port_type(hd3ss3220, type);
}

static const struct typec_operations hd3ss3220_ops = {
.dr_set = hd3ss3220_dr_set
.dr_set = hd3ss3220_dr_set,
.port_type_set = hd3ss3220_port_type_set,
};

static void hd3ss3220_set_role(struct hd3ss3220 *hd3ss3220)
Expand Down Expand Up @@ -211,6 +271,28 @@ static int hd3ss3220_configure_power_opmode(struct hd3ss3220 *hd3ss3220,
return hd3ss3220_set_power_opmode(hd3ss3220, power_opmode);
}

static int hd3ss3220_configure_port_type(struct hd3ss3220 *hd3ss3220,
struct fwnode_handle *connector,
struct typec_capability *cap)
{
/*
* Port type can be configured through device tree
*/
const char *cap_str;
int ret;

ret = fwnode_property_read_string(connector, "power-role", &cap_str);
if (ret)
return 0;

ret = typec_find_port_power_role(cap_str);
if (ret < 0)
return ret;

cap->type = ret;
return hd3ss3220_set_port_type(hd3ss3220, cap->type);
}

static const struct regmap_config config = {
.reg_bits = 8,
.val_bits = 8,
Expand Down Expand Up @@ -266,6 +348,10 @@ static int hd3ss3220_probe(struct i2c_client *client)
typec_cap.ops = &hd3ss3220_ops;
typec_cap.fwnode = connector;

ret = hd3ss3220_configure_port_type(hd3ss3220, connector, &typec_cap);
if (ret < 0)
goto err_put_role;

hd3ss3220->port = typec_register_port(&client->dev, &typec_cap);
if (IS_ERR(hd3ss3220->port)) {
ret = PTR_ERR(hd3ss3220->port);
Expand Down

0 comments on commit 5d2c32d

Please sign in to comment.