usb: typec: mux: Find the muxes by also matching against the device node
When the connections are defined in firmware, struct device_connection will have the fwnode member pointing to the device node (struct fwnode_handle) of the requested device, and the endpoint will not be used at all in that case. Acked-by: Hans de Goede <hdegoede@redhat.com> Reviewed-by: Andy Shevchenko <andy.shevchenko@gmail.com> Reviewed-by: Jun Li <jun.li@nxp.com> Signed-off-by: Heikki Krogerus <heikki.krogerus@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
637e9e52b1
commit
96a6d031ca
|
@ -11,6 +11,8 @@
|
||||||
#include <linux/list.h>
|
#include <linux/list.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
|
#include <linux/property.h>
|
||||||
|
#include <linux/slab.h>
|
||||||
#include <linux/usb/typec_mux.h>
|
#include <linux/usb/typec_mux.h>
|
||||||
|
|
||||||
static DEFINE_MUTEX(switch_lock);
|
static DEFINE_MUTEX(switch_lock);
|
||||||
|
@ -23,15 +25,25 @@ static void *typec_switch_match(struct device_connection *con, int ep,
|
||||||
{
|
{
|
||||||
struct typec_switch *sw;
|
struct typec_switch *sw;
|
||||||
|
|
||||||
list_for_each_entry(sw, &switch_list, entry)
|
if (!con->fwnode) {
|
||||||
if (!strcmp(con->endpoint[ep], dev_name(sw->dev)))
|
list_for_each_entry(sw, &switch_list, entry)
|
||||||
return sw;
|
if (!strcmp(con->endpoint[ep], dev_name(sw->dev)))
|
||||||
|
return sw;
|
||||||
|
return ERR_PTR(-EPROBE_DEFER);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We only get called if a connection was found, tell the caller to
|
* With OF graph the mux node must have a boolean device property named
|
||||||
* wait for the switch to show up.
|
* "orientation-switch".
|
||||||
*/
|
*/
|
||||||
return ERR_PTR(-EPROBE_DEFER);
|
if (con->id && !fwnode_property_present(con->fwnode, con->id))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
list_for_each_entry(sw, &switch_list, entry)
|
||||||
|
if (dev_fwnode(sw->dev) == con->fwnode)
|
||||||
|
return sw;
|
||||||
|
|
||||||
|
return con->id ? ERR_PTR(-EPROBE_DEFER) : NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -112,17 +124,67 @@ EXPORT_SYMBOL_GPL(typec_switch_unregister);
|
||||||
|
|
||||||
static void *typec_mux_match(struct device_connection *con, int ep, void *data)
|
static void *typec_mux_match(struct device_connection *con, int ep, void *data)
|
||||||
{
|
{
|
||||||
|
const struct typec_altmode_desc *desc = data;
|
||||||
struct typec_mux *mux;
|
struct typec_mux *mux;
|
||||||
|
size_t nval;
|
||||||
|
bool match;
|
||||||
|
u16 *val;
|
||||||
|
int i;
|
||||||
|
|
||||||
list_for_each_entry(mux, &mux_list, entry)
|
if (!con->fwnode) {
|
||||||
if (!strcmp(con->endpoint[ep], dev_name(mux->dev)))
|
list_for_each_entry(mux, &mux_list, entry)
|
||||||
return mux;
|
if (!strcmp(con->endpoint[ep], dev_name(mux->dev)))
|
||||||
|
return mux;
|
||||||
|
return ERR_PTR(-EPROBE_DEFER);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We only get called if a connection was found, tell the caller to
|
* Check has the identifier already been "consumed". If it
|
||||||
* wait for the switch to show up.
|
* has, no need to do any extra connection identification.
|
||||||
*/
|
*/
|
||||||
return ERR_PTR(-EPROBE_DEFER);
|
match = !con->id;
|
||||||
|
if (match)
|
||||||
|
goto find_mux;
|
||||||
|
|
||||||
|
/* Accessory Mode muxes */
|
||||||
|
if (!desc) {
|
||||||
|
match = fwnode_property_present(con->fwnode, "accessory");
|
||||||
|
if (match)
|
||||||
|
goto find_mux;
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Alternate Mode muxes */
|
||||||
|
nval = fwnode_property_read_u16_array(con->fwnode, "svid", NULL, 0);
|
||||||
|
if (nval <= 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
val = kcalloc(nval, sizeof(*val), GFP_KERNEL);
|
||||||
|
if (!val)
|
||||||
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
|
nval = fwnode_property_read_u16_array(con->fwnode, "svid", val, nval);
|
||||||
|
if (nval < 0) {
|
||||||
|
kfree(val);
|
||||||
|
return ERR_PTR(nval);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < nval; i++) {
|
||||||
|
match = val[i] == desc->svid;
|
||||||
|
if (match) {
|
||||||
|
kfree(val);
|
||||||
|
goto find_mux;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
kfree(val);
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
find_mux:
|
||||||
|
list_for_each_entry(mux, &mux_list, entry)
|
||||||
|
if (dev_fwnode(mux->dev) == con->fwnode)
|
||||||
|
return mux;
|
||||||
|
|
||||||
|
return match ? ERR_PTR(-EPROBE_DEFER) : NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
Loading…
Reference in New Issue