Commit 9b0ae699 authored by Badhri Jagan Sridharan's avatar Badhri Jagan Sridharan Committed by Greg Kroah-Hartman

staging: typec: tcpm: set port type callback

The port type callback call enquires the tcpc_dev if
the requested port type is supported. If supported, then
performs a tcpm reset if required after setting the tcpm
internal port_type variable.

Check against the tcpm port_type instead of checking
against caps.type as port_type reflects the current
configuration.
Signed-off-by: default avatarBadhri Jagan Sridharan <Badhri@google.com>
Reviewed-by: default avatarGuenter Roeck <linux@roeck-us.net>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent a033c3b1
...@@ -197,6 +197,7 @@ struct tcpm_port { ...@@ -197,6 +197,7 @@ struct tcpm_port {
bool attached; bool attached;
bool connected; bool connected;
enum typec_port_type port_type;
bool vbus_present; bool vbus_present;
bool vbus_never_low; bool vbus_never_low;
bool vbus_source; bool vbus_source;
...@@ -334,7 +335,7 @@ struct pd_rx_event { ...@@ -334,7 +335,7 @@ struct pd_rx_event {
static enum tcpm_state tcpm_default_state(struct tcpm_port *port) static enum tcpm_state tcpm_default_state(struct tcpm_port *port)
{ {
if (port->typec_caps.type == TYPEC_PORT_DRP) { if (port->port_type == TYPEC_PORT_DRP) {
if (port->try_role == TYPEC_SINK) if (port->try_role == TYPEC_SINK)
return SNK_UNATTACHED; return SNK_UNATTACHED;
else if (port->try_role == TYPEC_SOURCE) else if (port->try_role == TYPEC_SOURCE)
...@@ -342,7 +343,7 @@ static enum tcpm_state tcpm_default_state(struct tcpm_port *port) ...@@ -342,7 +343,7 @@ static enum tcpm_state tcpm_default_state(struct tcpm_port *port)
else if (port->tcpc->config->default_role == TYPEC_SINK) else if (port->tcpc->config->default_role == TYPEC_SINK)
return SNK_UNATTACHED; return SNK_UNATTACHED;
/* Fall through to return SRC_UNATTACHED */ /* Fall through to return SRC_UNATTACHED */
} else if (port->typec_caps.type == TYPEC_PORT_UFP) { } else if (port->port_type == TYPEC_PORT_UFP) {
return SNK_UNATTACHED; return SNK_UNATTACHED;
} }
return SRC_UNATTACHED; return SRC_UNATTACHED;
...@@ -1463,7 +1464,7 @@ static void tcpm_pd_ctrl_request(struct tcpm_port *port, ...@@ -1463,7 +1464,7 @@ static void tcpm_pd_ctrl_request(struct tcpm_port *port,
tcpm_set_state(port, SOFT_RESET, 0); tcpm_set_state(port, SOFT_RESET, 0);
break; break;
case PD_CTRL_DR_SWAP: case PD_CTRL_DR_SWAP:
if (port->typec_caps.type != TYPEC_PORT_DRP) { if (port->port_type != TYPEC_PORT_DRP) {
tcpm_queue_message(port, PD_MSG_CTRL_REJECT); tcpm_queue_message(port, PD_MSG_CTRL_REJECT);
break; break;
} }
...@@ -1483,7 +1484,7 @@ static void tcpm_pd_ctrl_request(struct tcpm_port *port, ...@@ -1483,7 +1484,7 @@ static void tcpm_pd_ctrl_request(struct tcpm_port *port,
} }
break; break;
case PD_CTRL_PR_SWAP: case PD_CTRL_PR_SWAP:
if (port->typec_caps.type != TYPEC_PORT_DRP) { if (port->port_type != TYPEC_PORT_DRP) {
tcpm_queue_message(port, PD_MSG_CTRL_REJECT); tcpm_queue_message(port, PD_MSG_CTRL_REJECT);
break; break;
} }
...@@ -1858,7 +1859,7 @@ static bool tcpm_start_drp_toggling(struct tcpm_port *port) ...@@ -1858,7 +1859,7 @@ static bool tcpm_start_drp_toggling(struct tcpm_port *port)
int ret; int ret;
if (port->tcpc->start_drp_toggling && if (port->tcpc->start_drp_toggling &&
port->typec_caps.type == TYPEC_PORT_DRP) { port->port_type == TYPEC_PORT_DRP) {
tcpm_log_force(port, "Start DRP toggling"); tcpm_log_force(port, "Start DRP toggling");
ret = port->tcpc->start_drp_toggling(port->tcpc, ret = port->tcpc->start_drp_toggling(port->tcpc,
tcpm_rp_cc(port)); tcpm_rp_cc(port));
...@@ -2168,7 +2169,7 @@ static void run_state_machine(struct tcpm_port *port) ...@@ -2168,7 +2169,7 @@ static void run_state_machine(struct tcpm_port *port)
break; break;
} }
tcpm_set_cc(port, tcpm_rp_cc(port)); tcpm_set_cc(port, tcpm_rp_cc(port));
if (port->typec_caps.type == TYPEC_PORT_DRP) if (port->port_type == TYPEC_PORT_DRP)
tcpm_set_state(port, SNK_UNATTACHED, PD_T_DRP_SNK); tcpm_set_state(port, SNK_UNATTACHED, PD_T_DRP_SNK);
break; break;
case SRC_ATTACH_WAIT: case SRC_ATTACH_WAIT:
...@@ -2325,7 +2326,7 @@ static void run_state_machine(struct tcpm_port *port) ...@@ -2325,7 +2326,7 @@ static void run_state_machine(struct tcpm_port *port)
break; break;
} }
tcpm_set_cc(port, TYPEC_CC_RD); tcpm_set_cc(port, TYPEC_CC_RD);
if (port->typec_caps.type == TYPEC_PORT_DRP) if (port->port_type == TYPEC_PORT_DRP)
tcpm_set_state(port, SRC_UNATTACHED, PD_T_DRP_SRC); tcpm_set_state(port, SRC_UNATTACHED, PD_T_DRP_SRC);
break; break;
case SNK_ATTACH_WAIT: case SNK_ATTACH_WAIT:
...@@ -2416,7 +2417,7 @@ static void run_state_machine(struct tcpm_port *port) ...@@ -2416,7 +2417,7 @@ static void run_state_machine(struct tcpm_port *port)
* see USB power delivery specification, section 8.3.3.6.1.5.1). * see USB power delivery specification, section 8.3.3.6.1.5.1).
*/ */
tcpm_set_state(port, hard_reset_state(port), tcpm_set_state(port, hard_reset_state(port),
port->typec_caps.type == TYPEC_PORT_DRP ? port->port_type == TYPEC_PORT_DRP ?
PD_T_DB_DETECT : PD_T_NO_RESPONSE); PD_T_DB_DETECT : PD_T_NO_RESPONSE);
break; break;
case SNK_DISCOVERY_DEBOUNCE: case SNK_DISCOVERY_DEBOUNCE:
...@@ -3172,7 +3173,7 @@ static int tcpm_dr_set(const struct typec_capability *cap, ...@@ -3172,7 +3173,7 @@ static int tcpm_dr_set(const struct typec_capability *cap,
mutex_lock(&port->swap_lock); mutex_lock(&port->swap_lock);
mutex_lock(&port->lock); mutex_lock(&port->lock);
if (port->typec_caps.type != TYPEC_PORT_DRP) { if (port->port_type != TYPEC_PORT_DRP) {
ret = -EINVAL; ret = -EINVAL;
goto port_unlock; goto port_unlock;
} }
...@@ -3240,7 +3241,7 @@ static int tcpm_pr_set(const struct typec_capability *cap, ...@@ -3240,7 +3241,7 @@ static int tcpm_pr_set(const struct typec_capability *cap,
mutex_lock(&port->swap_lock); mutex_lock(&port->swap_lock);
mutex_lock(&port->lock); mutex_lock(&port->lock);
if (port->typec_caps.type != TYPEC_PORT_DRP) { if (port->port_type != TYPEC_PORT_DRP) {
ret = -EINVAL; ret = -EINVAL;
goto port_unlock; goto port_unlock;
} }
...@@ -3362,6 +3363,34 @@ static void tcpm_init(struct tcpm_port *port) ...@@ -3362,6 +3363,34 @@ static void tcpm_init(struct tcpm_port *port)
tcpm_set_state(port, PORT_RESET, 0); tcpm_set_state(port, PORT_RESET, 0);
} }
static int tcpm_port_type_set(const struct typec_capability *cap,
enum typec_port_type type)
{
struct tcpm_port *port = typec_cap_to_tcpm(cap);
mutex_lock(&port->lock);
if (type == port->port_type)
goto port_unlock;
port->port_type = type;
if (!port->connected) {
tcpm_set_state(port, PORT_RESET, 0);
} else if (type == TYPEC_PORT_UFP) {
if (!(port->pwr_role == TYPEC_SINK &&
port->data_role == TYPEC_DEVICE))
tcpm_set_state(port, PORT_RESET, 0);
} else if (type == TYPEC_PORT_DFP) {
if (!(port->pwr_role == TYPEC_SOURCE &&
port->data_role == TYPEC_HOST))
tcpm_set_state(port, PORT_RESET, 0);
}
port_unlock:
mutex_unlock(&port->lock);
return 0;
}
void tcpm_tcpc_reset(struct tcpm_port *port) void tcpm_tcpc_reset(struct tcpm_port *port)
{ {
mutex_lock(&port->lock); mutex_lock(&port->lock);
...@@ -3509,9 +3538,10 @@ struct tcpm_port *tcpm_register_port(struct device *dev, struct tcpc_dev *tcpc) ...@@ -3509,9 +3538,10 @@ struct tcpm_port *tcpm_register_port(struct device *dev, struct tcpc_dev *tcpc)
port->typec_caps.pr_set = tcpm_pr_set; port->typec_caps.pr_set = tcpm_pr_set;
port->typec_caps.vconn_set = tcpm_vconn_set; port->typec_caps.vconn_set = tcpm_vconn_set;
port->typec_caps.try_role = tcpm_try_role; port->typec_caps.try_role = tcpm_try_role;
port->typec_caps.port_type_set = tcpm_port_type_set;
port->partner_desc.identity = &port->partner_ident; port->partner_desc.identity = &port->partner_ident;
port->port_type = tcpc->config->type;
/* /*
* TODO: * TODO:
* - alt_modes, set_alt_mode * - alt_modes, set_alt_mode
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment