// SPDX-License-Identifier: GPL-2.0
|
/*
|
* USB Type-C Multiplexer/DeMultiplexer Switch support
|
*
|
* Copyright (C) 2018 Intel Corporation
|
* Author: Heikki Krogerus <heikki.krogerus@linux.intel.com>
|
* Hans de Goede <hdegoede@redhat.com>
|
*/
|
|
#include <linux/device.h>
|
#include <linux/list.h>
|
#include <linux/module.h>
|
#include <linux/mutex.h>
|
#include <linux/property.h>
|
#include <linux/slab.h>
|
#include <linux/usb/typec_mux.h>
|
|
#include "bus.h"
|
|
static bool dev_name_ends_with(struct device *dev, const char *suffix)
|
{
|
const char *name = dev_name(dev);
|
const int name_len = strlen(name);
|
const int suffix_len = strlen(suffix);
|
|
if (suffix_len > name_len)
|
return false;
|
|
return strcmp(name + (name_len - suffix_len), suffix) == 0;
|
}
|
|
static int switch_fwnode_match(struct device *dev, const void *fwnode)
|
{
|
return dev_fwnode(dev) == fwnode && dev_name_ends_with(dev, "-switch");
|
}
|
|
static void *typec_switch_match(struct fwnode_handle *fwnode, const char *id,
|
void *data)
|
{
|
struct device *dev;
|
|
if (id && !fwnode_property_present(fwnode, id))
|
return NULL;
|
|
dev = class_find_device(&typec_mux_class, NULL, fwnode,
|
switch_fwnode_match);
|
|
return dev ? to_typec_switch(dev) : ERR_PTR(-EPROBE_DEFER);
|
}
|
|
/**
|
* fwnode_typec_switch_get - Find USB Type-C orientation switch
|
* @fwnode: The caller device node
|
*
|
* Finds a switch linked with @dev. Returns a reference to the switch on
|
* success, NULL if no matching connection was found, or
|
* ERR_PTR(-EPROBE_DEFER) when a connection was found but the switch
|
* has not been enumerated yet.
|
*/
|
struct typec_switch *fwnode_typec_switch_get(struct fwnode_handle *fwnode)
|
{
|
struct typec_switch *sw;
|
|
sw = fwnode_connection_find_match(fwnode, "orientation-switch", NULL,
|
typec_switch_match);
|
if (!IS_ERR_OR_NULL(sw))
|
WARN_ON(!try_module_get(sw->dev.parent->driver->owner));
|
|
return sw;
|
}
|
EXPORT_SYMBOL_GPL(fwnode_typec_switch_get);
|
|
/**
|
* typec_switch_put - Release USB Type-C orientation switch
|
* @sw: USB Type-C orientation switch
|
*
|
* Decrement reference count for @sw.
|
*/
|
void typec_switch_put(struct typec_switch *sw)
|
{
|
if (!IS_ERR_OR_NULL(sw)) {
|
module_put(sw->dev.parent->driver->owner);
|
put_device(&sw->dev);
|
}
|
}
|
EXPORT_SYMBOL_GPL(typec_switch_put);
|
|
static void typec_switch_release(struct device *dev)
|
{
|
kfree(to_typec_switch(dev));
|
}
|
|
static const struct device_type typec_switch_dev_type = {
|
.name = "orientation_switch",
|
.release = typec_switch_release,
|
};
|
|
/**
|
* typec_switch_register - Register USB Type-C orientation switch
|
* @parent: Parent device
|
* @desc: Orientation switch description
|
*
|
* This function registers a switch that can be used for routing the correct
|
* data pairs depending on the cable plug orientation from the USB Type-C
|
* connector to the USB controllers. USB Type-C plugs can be inserted
|
* right-side-up or upside-down.
|
*/
|
struct typec_switch *
|
typec_switch_register(struct device *parent,
|
const struct typec_switch_desc *desc)
|
{
|
struct typec_switch *sw;
|
int ret;
|
|
if (!desc || !desc->set)
|
return ERR_PTR(-EINVAL);
|
|
sw = kzalloc(sizeof(*sw), GFP_KERNEL);
|
if (!sw)
|
return ERR_PTR(-ENOMEM);
|
|
sw->set = desc->set;
|
|
device_initialize(&sw->dev);
|
sw->dev.parent = parent;
|
sw->dev.fwnode = desc->fwnode;
|
sw->dev.class = &typec_mux_class;
|
sw->dev.type = &typec_switch_dev_type;
|
sw->dev.driver_data = desc->drvdata;
|
ret = dev_set_name(&sw->dev, "%s-switch", desc->name ? desc->name : dev_name(parent));
|
if (ret) {
|
put_device(&sw->dev);
|
return ERR_PTR(ret);
|
}
|
|
ret = device_add(&sw->dev);
|
if (ret) {
|
dev_err(parent, "failed to register switch (%d)\n", ret);
|
put_device(&sw->dev);
|
return ERR_PTR(ret);
|
}
|
|
return sw;
|
}
|
EXPORT_SYMBOL_GPL(typec_switch_register);
|
|
int typec_switch_set(struct typec_switch *sw,
|
enum typec_orientation orientation)
|
{
|
if (IS_ERR_OR_NULL(sw))
|
return 0;
|
|
return sw->set(sw, orientation);
|
}
|
EXPORT_SYMBOL_GPL(typec_switch_set);
|
|
/**
|
* typec_switch_unregister - Unregister USB Type-C orientation switch
|
* @sw: USB Type-C orientation switch
|
*
|
* Unregister switch that was registered with typec_switch_register().
|
*/
|
void typec_switch_unregister(struct typec_switch *sw)
|
{
|
if (!IS_ERR_OR_NULL(sw))
|
device_unregister(&sw->dev);
|
}
|
EXPORT_SYMBOL_GPL(typec_switch_unregister);
|
|
void typec_switch_set_drvdata(struct typec_switch *sw, void *data)
|
{
|
dev_set_drvdata(&sw->dev, data);
|
}
|
EXPORT_SYMBOL_GPL(typec_switch_set_drvdata);
|
|
void *typec_switch_get_drvdata(struct typec_switch *sw)
|
{
|
return dev_get_drvdata(&sw->dev);
|
}
|
EXPORT_SYMBOL_GPL(typec_switch_get_drvdata);
|
|
/* ------------------------------------------------------------------------- */
|
|
static int mux_fwnode_match(struct device *dev, const void *fwnode)
|
{
|
return dev_fwnode(dev) == fwnode && dev_name_ends_with(dev, "-mux");
|
}
|
|
static void *typec_mux_match(struct fwnode_handle *fwnode, const char *id,
|
void *data)
|
{
|
const struct typec_altmode_desc *desc = data;
|
struct device *dev;
|
bool match;
|
int nval;
|
u16 *val;
|
int ret;
|
int i;
|
|
/*
|
* Check has the identifier already been "consumed". If it
|
* has, no need to do any extra connection identification.
|
*/
|
match = !id;
|
if (match)
|
goto find_mux;
|
|
/* Accessory Mode muxes */
|
if (!desc) {
|
match = fwnode_property_present(fwnode, "accessory");
|
if (match)
|
goto find_mux;
|
return NULL;
|
}
|
|
/* Alternate Mode muxes */
|
nval = fwnode_property_count_u16(fwnode, "svid");
|
if (nval <= 0)
|
return NULL;
|
|
val = kcalloc(nval, sizeof(*val), GFP_KERNEL);
|
if (!val)
|
return ERR_PTR(-ENOMEM);
|
|
ret = fwnode_property_read_u16_array(fwnode, "svid", val, nval);
|
if (ret < 0) {
|
kfree(val);
|
return ERR_PTR(ret);
|
}
|
|
for (i = 0; i < nval; i++) {
|
match = val[i] == desc->svid;
|
if (match) {
|
kfree(val);
|
goto find_mux;
|
}
|
}
|
kfree(val);
|
return NULL;
|
|
find_mux:
|
dev = class_find_device(&typec_mux_class, NULL, fwnode,
|
mux_fwnode_match);
|
|
return dev ? to_typec_mux(dev) : ERR_PTR(-EPROBE_DEFER);
|
}
|
|
/**
|
* fwnode_typec_mux_get - Find USB Type-C Multiplexer
|
* @fwnode: The caller device node
|
* @desc: Alt Mode description
|
*
|
* Finds a mux linked to the caller. This function is primarily meant for the
|
* Type-C drivers. Returns a reference to the mux on success, NULL if no
|
* matching connection was found, or ERR_PTR(-EPROBE_DEFER) when a connection
|
* was found but the mux has not been enumerated yet.
|
*/
|
struct typec_mux *fwnode_typec_mux_get(struct fwnode_handle *fwnode,
|
const struct typec_altmode_desc *desc)
|
{
|
struct typec_mux *mux;
|
|
mux = fwnode_connection_find_match(fwnode, "mode-switch", (void *)desc,
|
typec_mux_match);
|
if (!IS_ERR_OR_NULL(mux))
|
WARN_ON(!try_module_get(mux->dev.parent->driver->owner));
|
|
return mux;
|
}
|
EXPORT_SYMBOL_GPL(fwnode_typec_mux_get);
|
|
/**
|
* typec_mux_put - Release handle to a Multiplexer
|
* @mux: USB Type-C Connector Multiplexer/DeMultiplexer
|
*
|
* Decrements reference count for @mux.
|
*/
|
void typec_mux_put(struct typec_mux *mux)
|
{
|
if (!IS_ERR_OR_NULL(mux)) {
|
module_put(mux->dev.parent->driver->owner);
|
put_device(&mux->dev);
|
}
|
}
|
EXPORT_SYMBOL_GPL(typec_mux_put);
|
|
int typec_mux_set(struct typec_mux *mux, struct typec_mux_state *state)
|
{
|
if (IS_ERR_OR_NULL(mux))
|
return 0;
|
|
return mux->set(mux, state);
|
}
|
EXPORT_SYMBOL_GPL(typec_mux_set);
|
|
static void typec_mux_release(struct device *dev)
|
{
|
kfree(to_typec_mux(dev));
|
}
|
|
static const struct device_type typec_mux_dev_type = {
|
.name = "mode_switch",
|
.release = typec_mux_release,
|
};
|
|
/**
|
* typec_mux_register - Register Multiplexer routing USB Type-C pins
|
* @parent: Parent device
|
* @desc: Multiplexer description
|
*
|
* USB Type-C connectors can be used for alternate modes of operation besides
|
* USB when Accessory/Alternate Modes are supported. With some of those modes,
|
* the pins on the connector need to be reconfigured. This function registers
|
* multiplexer switches routing the pins on the connector.
|
*/
|
struct typec_mux *
|
typec_mux_register(struct device *parent, const struct typec_mux_desc *desc)
|
{
|
struct typec_mux *mux;
|
int ret;
|
|
if (!desc || !desc->set)
|
return ERR_PTR(-EINVAL);
|
|
mux = kzalloc(sizeof(*mux), GFP_KERNEL);
|
if (!mux)
|
return ERR_PTR(-ENOMEM);
|
|
mux->set = desc->set;
|
|
device_initialize(&mux->dev);
|
mux->dev.parent = parent;
|
mux->dev.fwnode = desc->fwnode;
|
mux->dev.class = &typec_mux_class;
|
mux->dev.type = &typec_mux_dev_type;
|
mux->dev.driver_data = desc->drvdata;
|
ret = dev_set_name(&mux->dev, "%s-mux", desc->name ? desc->name : dev_name(parent));
|
if (ret) {
|
put_device(&mux->dev);
|
return ERR_PTR(ret);
|
}
|
|
ret = device_add(&mux->dev);
|
if (ret) {
|
dev_err(parent, "failed to register mux (%d)\n", ret);
|
put_device(&mux->dev);
|
return ERR_PTR(ret);
|
}
|
|
return mux;
|
}
|
EXPORT_SYMBOL_GPL(typec_mux_register);
|
|
/**
|
* typec_mux_unregister - Unregister Multiplexer Switch
|
* @mux: USB Type-C Connector Multiplexer/DeMultiplexer
|
*
|
* Unregister mux that was registered with typec_mux_register().
|
*/
|
void typec_mux_unregister(struct typec_mux *mux)
|
{
|
if (!IS_ERR_OR_NULL(mux))
|
device_unregister(&mux->dev);
|
}
|
EXPORT_SYMBOL_GPL(typec_mux_unregister);
|
|
void typec_mux_set_drvdata(struct typec_mux *mux, void *data)
|
{
|
dev_set_drvdata(&mux->dev, data);
|
}
|
EXPORT_SYMBOL_GPL(typec_mux_set_drvdata);
|
|
void *typec_mux_get_drvdata(struct typec_mux *mux)
|
{
|
return dev_get_drvdata(&mux->dev);
|
}
|
EXPORT_SYMBOL_GPL(typec_mux_get_drvdata);
|
|
struct class typec_mux_class = {
|
.name = "typec_mux",
|
.owner = THIS_MODULE,
|
};
|