[U-Boot] [PATCH v4 4/7] drivers/i2c/muxes/pca954x: Add pca9547 I2C mux support
Chris Packham
judge.packham at gmail.com
Thu Sep 28 21:27:51 UTC 2017
Hi All,
On Sat, Jun 10, 2017 at 5:28 AM, Marek BehĂșn <marek.behun at nic.cz> wrote:
> This I2C mux is found, for example, on the Turris Omnia board.
>
> Signed-off-by: Marek Behun <marek.behun at nic.cz>
> Reviewed-by: Heiko Schocher <hs at denx.de>
>
> diff --git a/drivers/i2c/muxes/pca954x.c b/drivers/i2c/muxes/pca954x.c
> index 1a6761858c..383f72f552 100644
> --- a/drivers/i2c/muxes/pca954x.c
> +++ b/drivers/i2c/muxes/pca954x.c
> @@ -13,11 +13,40 @@
>
> DECLARE_GLOBAL_DATA_PTR;
>
> +enum pca_type {
> + PCA9544,
> + PCA9547,
> + PCA9548
> +};
> +
> +struct chip_desc {
> + u8 enable;
> + enum muxtype {
> + pca954x_ismux = 0,
> + pca954x_isswi,
> + } muxtype;
> +};
> +
> struct pca954x_priv {
> u32 addr; /* I2C mux address */
> u32 width; /* I2C mux width - number of busses */
> };
>
> +static const struct chip_desc chips[] = {
> + [PCA9544] = {
> + .enable = 0x4,
> + .muxtype = pca954x_ismux,
> + },
> + [PCA9547] = {
> + .enable = 0x8,
> + .muxtype = pca954x_ismux,
> + },
> + [PCA9548] = {
> + .enable = 0x8,
> + .muxtype = pca954x_isswi,
> + },
> +};
> +
> static int pca954x_deselect(struct udevice *mux, struct udevice *bus,
> uint channel)
> {
> @@ -31,7 +60,13 @@ static int pca954x_select(struct udevice *mux, struct udevice *bus,
> uint channel)
> {
> struct pca954x_priv *priv = dev_get_priv(mux);
> - uchar byte = 1 << channel;
> + const struct chip_desc *chip = &chips[dev_get_driver_data(mux)];
> + uchar byte;
> +
> + if (chip->muxtype == pca954x_ismux)
> + byte = channel | chip->enable;
> + else
> + byte = 1 << channel;
>
> return dm_i2c_write(mux, priv->addr, &byte, 1);
> }
> @@ -42,8 +77,9 @@ static const struct i2c_mux_ops pca954x_ops = {
> };
>
> static const struct udevice_id pca954x_ids[] = {
> - { .compatible = "nxp,pca9548", .data = (ulong)8 },
> - { .compatible = "nxp,pca9544", .data = (ulong)4 },
> + { .compatible = "nxp,pca9544", .data = PCA9544 },
> + { .compatible = "nxp,pca9547", .data = PCA9547 },
> + { .compatible = "nxp,pca9548", .data = PCA9548 },
> { }
> };
>
I think this actually breaks this driver. The problem is in
pca954x_ofdata_to_platdata() where it does
priv->width = dev_get_driver_data(dev);
Which is now not the width but one of the enum values. It would work
for 9457 or 9548 but I happen to be using a9544 which is 0 in the
enum. I'll post a patch shortly to fix this.
More information about the U-Boot
mailing list