i2c: mux/i801: Switch to use descriptor passing

This switches the i801 GPIO mux to use GPIO descriptors for
handling the GPIO lines. The previous hack which was reaching
inside the GPIO chips etc cannot live on. We pass descriptors
along with the GPIO mux device at creation instead.

The GPIO mux was only used by way of platform data with a
platform device from one place in the kernel: the i801 i2c bus
driver. Let's just associate the GPIO descriptor table with
the actual device like everyone else and dynamically create
a descriptor table passed along with the GPIO i2c mux.

This enables simplification of the GPIO i2c mux driver to
use only the descriptor API and the OF probe path gets
simplified in the process.

The i801 driver was registering the GPIO i2c mux with
PLATFORM_DEVID_AUTO which would make it hard to predict the
device name and assign the descriptor table properly, but
this seems to be a mistake to begin with: all of the
GPIO mux devices are hardcoded to look up GPIO lines from
the "gpio_ich" GPIO chip. If there are more than one mux,
there is certainly more than one gpio chip as well, and
then we have more serious problems. Switch to
PLATFORM_DEVID_NONE instead. There can be only one.

Cc: Mika Westerberg <mika.westerberg@linux.intel.com>
Cc: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Cc: Peter Rosin <peda@axentia.se>
Cc: Jean Delvare <jdelvare@suse.com>
Signed-off-by: Serge Semin <fancer.lancer@gmail.com>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
Reviewed-by: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Reviewed-by: Mika Westerberg <mika.westerberg@linux.intel.com>
[Removed a newline, suggested by Andy. /Peter]
Signed-off-by: Peter Rosin <peda@axentia.se>
This commit is contained in:
Linus Walleij 2019-06-18 12:58:33 +02:00 committed by Peter Rosin
parent 90af27317b
commit d308dfbf62
3 changed files with 59 additions and 99 deletions

View File

@ -107,7 +107,7 @@
#include <linux/pm_runtime.h> #include <linux/pm_runtime.h>
#if IS_ENABLED(CONFIG_I2C_MUX_GPIO) && defined CONFIG_DMI #if IS_ENABLED(CONFIG_I2C_MUX_GPIO) && defined CONFIG_DMI
#include <linux/gpio.h> #include <linux/gpio/machine.h>
#include <linux/platform_data/i2c-mux-gpio.h> #include <linux/platform_data/i2c-mux-gpio.h>
#endif #endif
@ -274,6 +274,7 @@ struct i801_priv {
#if IS_ENABLED(CONFIG_I2C_MUX_GPIO) && defined CONFIG_DMI #if IS_ENABLED(CONFIG_I2C_MUX_GPIO) && defined CONFIG_DMI
const struct i801_mux_config *mux_drvdata; const struct i801_mux_config *mux_drvdata;
struct platform_device *mux_pdev; struct platform_device *mux_pdev;
struct gpiod_lookup_table *lookup;
#endif #endif
struct platform_device *tco_pdev; struct platform_device *tco_pdev;
@ -1258,7 +1259,8 @@ static int i801_add_mux(struct i801_priv *priv)
struct device *dev = &priv->adapter.dev; struct device *dev = &priv->adapter.dev;
const struct i801_mux_config *mux_config; const struct i801_mux_config *mux_config;
struct i2c_mux_gpio_platform_data gpio_data; struct i2c_mux_gpio_platform_data gpio_data;
int err; struct gpiod_lookup_table *lookup;
int err, i;
if (!priv->mux_drvdata) if (!priv->mux_drvdata)
return 0; return 0;
@ -1270,17 +1272,36 @@ static int i801_add_mux(struct i801_priv *priv)
gpio_data.values = mux_config->values; gpio_data.values = mux_config->values;
gpio_data.n_values = mux_config->n_values; gpio_data.n_values = mux_config->n_values;
gpio_data.classes = mux_config->classes; gpio_data.classes = mux_config->classes;
gpio_data.gpio_chip = mux_config->gpio_chip;
gpio_data.gpios = mux_config->gpios;
gpio_data.n_gpios = mux_config->n_gpios;
gpio_data.idle = I2C_MUX_GPIO_NO_IDLE; gpio_data.idle = I2C_MUX_GPIO_NO_IDLE;
/* Register the mux device */ /* Register GPIO descriptor lookup table */
lookup = devm_kzalloc(dev,
struct_size(lookup, table, mux_config->n_gpios),
GFP_KERNEL);
if (!lookup)
return -ENOMEM;
lookup->dev_id = "i2c-mux-gpio";
for (i = 0; i < mux_config->n_gpios; i++) {
lookup->table[i].chip_label = mux_config->gpio_chip;
lookup->table[i].chip_hwnum = mux_config->gpios[i];
lookup->table[i].con_id = "mux";
}
gpiod_add_lookup_table(lookup);
priv->lookup = lookup;
/*
* Register the mux device, we use PLATFORM_DEVID_NONE here
* because since we are referring to the GPIO chip by name we are
* anyways in deep trouble if there is more than one of these
* devices, and there should likely only be one platform controller
* hub.
*/
priv->mux_pdev = platform_device_register_data(dev, "i2c-mux-gpio", priv->mux_pdev = platform_device_register_data(dev, "i2c-mux-gpio",
PLATFORM_DEVID_AUTO, &gpio_data, PLATFORM_DEVID_NONE, &gpio_data,
sizeof(struct i2c_mux_gpio_platform_data)); sizeof(struct i2c_mux_gpio_platform_data));
if (IS_ERR(priv->mux_pdev)) { if (IS_ERR(priv->mux_pdev)) {
err = PTR_ERR(priv->mux_pdev); err = PTR_ERR(priv->mux_pdev);
gpiod_remove_lookup_table(lookup);
priv->mux_pdev = NULL; priv->mux_pdev = NULL;
dev_err(dev, "Failed to register i2c-mux-gpio device\n"); dev_err(dev, "Failed to register i2c-mux-gpio device\n");
return err; return err;
@ -1293,6 +1314,8 @@ static void i801_del_mux(struct i801_priv *priv)
{ {
if (priv->mux_pdev) if (priv->mux_pdev)
platform_device_unregister(priv->mux_pdev); platform_device_unregister(priv->mux_pdev);
if (priv->lookup)
gpiod_remove_lookup_table(priv->lookup);
} }
static unsigned int i801_get_adapter_class(struct i801_priv *priv) static unsigned int i801_get_adapter_class(struct i801_priv *priv)

View File

@ -14,13 +14,14 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/gpio.h> #include <linux/bits.h>
#include <linux/gpio/consumer.h>
/* FIXME: stop poking around inside gpiolib */
#include "../../gpio/gpiolib.h" #include "../../gpio/gpiolib.h"
#include <linux/of_gpio.h>
struct gpiomux { struct gpiomux {
struct i2c_mux_gpio_platform_data data; struct i2c_mux_gpio_platform_data data;
unsigned gpio_base; int ngpios;
struct gpio_desc **gpios; struct gpio_desc **gpios;
}; };
@ -30,8 +31,7 @@ static void i2c_mux_gpio_set(const struct gpiomux *mux, unsigned val)
values[0] = val; values[0] = val;
gpiod_set_array_value_cansleep(mux->data.n_gpios, mux->gpios, NULL, gpiod_set_array_value_cansleep(mux->ngpios, mux->gpios, NULL, values);
values);
} }
static int i2c_mux_gpio_select(struct i2c_mux_core *muxc, u32 chan) static int i2c_mux_gpio_select(struct i2c_mux_core *muxc, u32 chan)
@ -52,12 +52,6 @@ static int i2c_mux_gpio_deselect(struct i2c_mux_core *muxc, u32 chan)
return 0; return 0;
} }
static int match_gpio_chip_by_label(struct gpio_chip *chip,
void *data)
{
return !strcmp(chip->label, data);
}
#ifdef CONFIG_OF #ifdef CONFIG_OF
static int i2c_mux_gpio_probe_dt(struct gpiomux *mux, static int i2c_mux_gpio_probe_dt(struct gpiomux *mux,
struct platform_device *pdev) struct platform_device *pdev)
@ -65,8 +59,8 @@ static int i2c_mux_gpio_probe_dt(struct gpiomux *mux,
struct device_node *np = pdev->dev.of_node; struct device_node *np = pdev->dev.of_node;
struct device_node *adapter_np, *child; struct device_node *adapter_np, *child;
struct i2c_adapter *adapter; struct i2c_adapter *adapter;
unsigned *values, *gpios; unsigned *values;
int i = 0, ret; int i = 0;
if (!np) if (!np)
return -ENODEV; return -ENODEV;
@ -103,29 +97,6 @@ static int i2c_mux_gpio_probe_dt(struct gpiomux *mux,
if (of_property_read_u32(np, "idle-state", &mux->data.idle)) if (of_property_read_u32(np, "idle-state", &mux->data.idle))
mux->data.idle = I2C_MUX_GPIO_NO_IDLE; mux->data.idle = I2C_MUX_GPIO_NO_IDLE;
mux->data.n_gpios = of_gpio_named_count(np, "mux-gpios");
if (mux->data.n_gpios < 0) {
dev_err(&pdev->dev, "Missing mux-gpios property in the DT.\n");
return -EINVAL;
}
gpios = devm_kcalloc(&pdev->dev,
mux->data.n_gpios, sizeof(*mux->data.gpios),
GFP_KERNEL);
if (!gpios) {
dev_err(&pdev->dev, "Cannot allocate gpios array");
return -ENOMEM;
}
for (i = 0; i < mux->data.n_gpios; i++) {
ret = of_get_named_gpio(np, "mux-gpios", i);
if (ret < 0)
return ret;
gpios[i] = ret;
}
mux->data.gpios = gpios;
return 0; return 0;
} }
#else #else
@ -142,8 +113,8 @@ static int i2c_mux_gpio_probe(struct platform_device *pdev)
struct gpiomux *mux; struct gpiomux *mux;
struct i2c_adapter *parent; struct i2c_adapter *parent;
struct i2c_adapter *root; struct i2c_adapter *root;
unsigned initial_state, gpio_base; unsigned initial_state;
int i, ret; int i, ngpios, ret;
mux = devm_kzalloc(&pdev->dev, sizeof(*mux), GFP_KERNEL); mux = devm_kzalloc(&pdev->dev, sizeof(*mux), GFP_KERNEL);
if (!mux) if (!mux)
@ -158,29 +129,19 @@ static int i2c_mux_gpio_probe(struct platform_device *pdev)
sizeof(mux->data)); sizeof(mux->data));
} }
/* ngpios = gpiod_count(&pdev->dev, "mux");
* If a GPIO chip name is provided, the GPIO pin numbers provided are if (ngpios <= 0) {
* relative to its base GPIO number. Otherwise they are absolute. dev_err(&pdev->dev, "no valid gpios provided\n");
*/ return ngpios ?: -EINVAL;
if (mux->data.gpio_chip) {
struct gpio_chip *gpio;
gpio = gpiochip_find(mux->data.gpio_chip,
match_gpio_chip_by_label);
if (!gpio)
return -EPROBE_DEFER;
gpio_base = gpio->base;
} else {
gpio_base = 0;
} }
mux->ngpios = ngpios;
parent = i2c_get_adapter(mux->data.parent); parent = i2c_get_adapter(mux->data.parent);
if (!parent) if (!parent)
return -EPROBE_DEFER; return -EPROBE_DEFER;
muxc = i2c_mux_alloc(parent, &pdev->dev, mux->data.n_values, muxc = i2c_mux_alloc(parent, &pdev->dev, mux->data.n_values,
mux->data.n_gpios * sizeof(*mux->gpios), 0, ngpios * sizeof(*mux->gpios), 0,
i2c_mux_gpio_select, NULL); i2c_mux_gpio_select, NULL);
if (!muxc) { if (!muxc) {
ret = -ENOMEM; ret = -ENOMEM;
@ -194,7 +155,6 @@ static int i2c_mux_gpio_probe(struct platform_device *pdev)
root = i2c_root_adapter(&parent->dev); root = i2c_root_adapter(&parent->dev);
muxc->mux_locked = true; muxc->mux_locked = true;
mux->gpio_base = gpio_base;
if (mux->data.idle != I2C_MUX_GPIO_NO_IDLE) { if (mux->data.idle != I2C_MUX_GPIO_NO_IDLE) {
initial_state = mux->data.idle; initial_state = mux->data.idle;
@ -203,34 +163,28 @@ static int i2c_mux_gpio_probe(struct platform_device *pdev)
initial_state = mux->data.values[0]; initial_state = mux->data.values[0];
} }
for (i = 0; i < mux->data.n_gpios; i++) { for (i = 0; i < ngpios; i++) {
struct device *gpio_dev; struct device *gpio_dev;
struct gpio_desc *gpio_desc; struct gpio_desc *gpiod;
enum gpiod_flags flag;
ret = gpio_request(gpio_base + mux->data.gpios[i], "i2c-mux-gpio"); if (initial_state & BIT(i))
if (ret) { flag = GPIOD_OUT_HIGH;
dev_err(&pdev->dev, "Failed to request GPIO %d\n", else
mux->data.gpios[i]); flag = GPIOD_OUT_LOW;
goto err_request_gpio; gpiod = devm_gpiod_get_index(&pdev->dev, "mux", i, flag);
if (IS_ERR(gpiod)) {
ret = PTR_ERR(gpiod);
goto alloc_failed;
} }
ret = gpio_direction_output(gpio_base + mux->data.gpios[i], mux->gpios[i] = gpiod;
initial_state & (1 << i));
if (ret) {
dev_err(&pdev->dev,
"Failed to set direction of GPIO %d to output\n",
mux->data.gpios[i]);
i++; /* gpio_request above succeeded, so must free */
goto err_request_gpio;
}
gpio_desc = gpio_to_desc(gpio_base + mux->data.gpios[i]);
mux->gpios[i] = gpio_desc;
if (!muxc->mux_locked) if (!muxc->mux_locked)
continue; continue;
gpio_dev = &gpio_desc->gdev->dev; /* FIXME: find a proper way to access the GPIO device */
gpio_dev = &gpiod->gdev->dev;
muxc->mux_locked = i2c_root_adapter(gpio_dev) == root; muxc->mux_locked = i2c_root_adapter(gpio_dev) == root;
} }
@ -253,10 +207,6 @@ static int i2c_mux_gpio_probe(struct platform_device *pdev)
add_adapter_failed: add_adapter_failed:
i2c_mux_del_adapters(muxc); i2c_mux_del_adapters(muxc);
i = mux->data.n_gpios;
err_request_gpio:
for (; i > 0; i--)
gpio_free(gpio_base + mux->data.gpios[i - 1]);
alloc_failed: alloc_failed:
i2c_put_adapter(parent); i2c_put_adapter(parent);
@ -266,14 +216,8 @@ alloc_failed:
static int i2c_mux_gpio_remove(struct platform_device *pdev) static int i2c_mux_gpio_remove(struct platform_device *pdev)
{ {
struct i2c_mux_core *muxc = platform_get_drvdata(pdev); struct i2c_mux_core *muxc = platform_get_drvdata(pdev);
struct gpiomux *mux = i2c_mux_priv(muxc);
int i;
i2c_mux_del_adapters(muxc); i2c_mux_del_adapters(muxc);
for (i = 0; i < mux->data.n_gpios; i++)
gpio_free(mux->gpio_base + mux->data.gpios[i]);
i2c_put_adapter(muxc->parent); i2c_put_adapter(muxc->parent);
return 0; return 0;

View File

@ -22,10 +22,6 @@
* position * position
* @n_values: Number of multiplexer positions (busses to instantiate) * @n_values: Number of multiplexer positions (busses to instantiate)
* @classes: Optional I2C auto-detection classes * @classes: Optional I2C auto-detection classes
* @gpio_chip: Optional GPIO chip name; if set, GPIO pin numbers are given
* relative to the base GPIO number of that chip
* @gpios: Array of GPIO numbers used to control MUX
* @n_gpios: Number of GPIOs used to control MUX
* @idle: Bitmask to write to MUX when idle or GPIO_I2CMUX_NO_IDLE if not used * @idle: Bitmask to write to MUX when idle or GPIO_I2CMUX_NO_IDLE if not used
*/ */
struct i2c_mux_gpio_platform_data { struct i2c_mux_gpio_platform_data {
@ -34,9 +30,6 @@ struct i2c_mux_gpio_platform_data {
const unsigned *values; const unsigned *values;
int n_values; int n_values;
const unsigned *classes; const unsigned *classes;
char *gpio_chip;
const unsigned *gpios;
int n_gpios;
unsigned idle; unsigned idle;
}; };