[PATCH] drm/bridge: sii902x: Retry status read after DDI I2C
Liviu Dudau
Liviu.Dudau at arm.com
Mon Mar 5 08:05:51 PST 2018
On Mon, Mar 05, 2018 at 11:17:02AM +0100, Linus Walleij wrote:
> The following happens when connection a DVI output driven
> from the SiI9022 using a DVI-to-VGA adapter plug:
>
> i2c i2c-0: sendbytes: NAK bailout.
> i2c i2c-0: sendbytes: NAK bailout.
>
> Then no picture. Apparently the I2C engine inside the SiI9022
> is not smart enough to try to fall back to DDC I2C. Or the
> vendor have not integrated the electronics properly. I don't
> know which one it is.
>
> After this, the I2C bus seems stalled and the first attempt to
> read the status register fails, and the code returns with
> negative return value, and the display fails to initialized.
>
> Instead, retry status readout five times and continue even
> if this fails.
>
> Tested on the ARM Versatile Express with a DVI-to-VGA
> connector, it now gives picture.
>
> Introduce a helper struct device *dev variable to make
> the code more readable.
>
> Cc: Ville Syrjälä <ville.syrjala at linux.intel.com>
> Cc: Liviu Dudau <Liviu.Dudau at arm.com>
I don't have a suitable configuration to test this, but looks goot to me.
Reviewed-by: Liviu Dudau <liviu.dudau at arm.com>
BTW, Linus, could it be that your DVI-to-VGA connector doesn't wire
correctly the EDID pin?
Best regards,
Liviu
> Signed-off-by: Linus Walleij <linus.walleij at linaro.org>
> ---
> ChangeLog v1->v2:
> - Found the real problem instead of trying to explicitly
> shoehorn EDID modes into the error path.
> ---
> drivers/gpu/drm/bridge/sii902x.c | 20 ++++++++++++++++----
> 1 file changed, 16 insertions(+), 4 deletions(-)
>
> diff --git a/drivers/gpu/drm/bridge/sii902x.c b/drivers/gpu/drm/bridge/sii902x.c
> index b1ab4ab09532..60373d7eb220 100644
> --- a/drivers/gpu/drm/bridge/sii902x.c
> +++ b/drivers/gpu/drm/bridge/sii902x.c
> @@ -137,7 +137,9 @@ static int sii902x_get_modes(struct drm_connector *connector)
> struct sii902x *sii902x = connector_to_sii902x(connector);
> struct regmap *regmap = sii902x->regmap;
> u32 bus_format = MEDIA_BUS_FMT_RGB888_1X24;
> + struct device *dev = &sii902x->i2c->dev;
> unsigned long timeout;
> + unsigned int retries;
> unsigned int status;
> struct edid *edid;
> int num = 0;
> @@ -159,7 +161,7 @@ static int sii902x_get_modes(struct drm_connector *connector)
> time_before(jiffies, timeout));
>
> if (!(status & SII902X_SYS_CTRL_DDC_BUS_GRTD)) {
> - dev_err(&sii902x->i2c->dev, "failed to acquire the i2c bus\n");
> + dev_err(dev, "failed to acquire the i2c bus\n");
> return -ETIMEDOUT;
> }
>
> @@ -179,9 +181,19 @@ static int sii902x_get_modes(struct drm_connector *connector)
> if (ret)
> return ret;
>
> - ret = regmap_read(regmap, SII902X_SYS_CTRL_DATA, &status);
> + /*
> + * Sometimes the I2C bus can stall after failure to use the
> + * EDID channel. Retry a few times to see if things clear
> + * up, else continue anyway.
> + */
> + retries = 5;
> + do {
> + ret = regmap_read(regmap, SII902X_SYS_CTRL_DATA,
> + &status);
> + retries--;
> + } while (ret && retries);
> if (ret)
> - return ret;
> + dev_err(dev, "failed to read status (%d)\n", ret);
>
> ret = regmap_update_bits(regmap, SII902X_SYS_CTRL_DATA,
> SII902X_SYS_CTRL_DDC_BUS_REQ |
> @@ -201,7 +213,7 @@ static int sii902x_get_modes(struct drm_connector *connector)
>
> if (status & (SII902X_SYS_CTRL_DDC_BUS_REQ |
> SII902X_SYS_CTRL_DDC_BUS_GRTD)) {
> - dev_err(&sii902x->i2c->dev, "failed to release the i2c bus\n");
> + dev_err(dev, "failed to release the i2c bus\n");
> return -ETIMEDOUT;
> }
>
> --
> 2.14.3
>
--
====================
| I would like to |
| fix the world, |
| but they're not |
| giving me the |
\ source code! /
---------------
¯\_(ツ)_/¯
More information about the linux-arm-kernel
mailing list