[Intel-gfx] [PATCH] drm/i915: cache the EDID for eDP panels
Jani Nikula
jani.nikula at linux.intel.com
Fri Jun 15 12:52:04 CEST 2012
On Fri, 15 Jun 2012, Chris Wilson <chris at chris-wilson.co.uk> wrote:
> They aren't going anywhere, and probing on DDC can cause the panel to
> blank briefly, so read them up front and cache them for later queries.
>
> Jesse's patch revamped. Gotta love those display_info.raw_edid = NULL!
> ---
> drivers/gpu/drm/i915/intel_dp.c | 48 +++++++++++++++++++++++++++---------
> drivers/gpu/drm/i915/intel_drv.h | 1 +
> drivers/gpu/drm/i915/intel_modes.c | 24 ++++++++++++------
> 3 files changed, 54 insertions(+), 19 deletions(-)
>
> diff --git a/drivers/gpu/drm/i915/intel_dp.c b/drivers/gpu/drm/i915/intel_dp.c
> index eb57ec7..207e25f 100644
> --- a/drivers/gpu/drm/i915/intel_dp.c
> +++ b/drivers/gpu/drm/i915/intel_dp.c
> @@ -67,6 +67,7 @@ struct intel_dp {
> struct drm_display_mode *panel_fixed_mode; /* for eDP */
> struct delayed_work panel_vdd_work;
> bool want_panel_vdd;
> + struct edid *edid; /* cached for eDP */
> };
>
> /**
> @@ -2095,26 +2096,50 @@ g4x_dp_detect(struct intel_dp *intel_dp)
> }
>
> static struct edid *
> -intel_dp_get_edid(struct drm_connector *connector, struct i2c_adapter *adapter)
> +_drm_edid_duplicate(struct edid *edid)
> +{
> + struct edid *copy;
> + int size;
> +
> + if (edid == NULL)
> + return NULL;
> +
> + size = EDID_LENGTH * (1 + edid->extensions);
> + *copy = kmalloc(size, GFP_KERNEL);
You may want to remove that '*' there...
Jani.
> + if (copy)
> + memcpy(copy, edid, size);
> +
> + return copy;
> +}
> +
> +static struct edid *
> +intel_dp_get_edid(struct drm_connector *connector)
> {
> struct intel_dp *intel_dp = intel_attached_dp(connector);
> - struct edid *edid;
> + struct edid *edid;
> +
> + if (is_edp(intel_dp))
> + return _drm_edid_duplicate(intel_dp->edid);
>
> ironlake_edp_panel_vdd_on(intel_dp);
> - edid = drm_get_edid(connector, adapter);
> + edid = drm_get_edid(connector, &intel_dp->adapter);
> ironlake_edp_panel_vdd_off(intel_dp, false);
> +
> return edid;
> }
>
> static int
> -intel_dp_get_edid_modes(struct drm_connector *connector, struct i2c_adapter *adapter)
> +intel_dp_get_edid_modes(struct drm_connector *connector)
> {
> struct intel_dp *intel_dp = intel_attached_dp(connector);
> - int ret;
> + struct edid *edid;
> + int ret;
> +
> + edid = intel_dp_get_edid(intel_dp);
> + ret = intel_connector_attach_edid(connector, edid);
> + connector->display_info.raw_edid = NULL;
> + kfree(edid);
>
> - ironlake_edp_panel_vdd_on(intel_dp);
> - ret = intel_ddc_get_modes(connector, adapter);
> - ironlake_edp_panel_vdd_off(intel_dp, false);
> return ret;
> }
>
> @@ -2172,7 +2197,7 @@ static int intel_dp_get_modes(struct drm_connector *connector)
> /* We should parse the EDID data and find out if it has an audio sink
> */
>
> - ret = intel_dp_get_edid_modes(connector, &intel_dp->adapter);
> + ret = intel_dp_get_edid_modes(connector);
> if (ret) {
> if (is_edp(intel_dp) && !intel_dp->panel_fixed_mode) {
> struct drm_display_mode *newmode;
> @@ -2485,6 +2510,8 @@ intel_dp_init(struct drm_device *dev, int output_reg)
> break;
> }
>
> + intel_dp_i2c_init(intel_dp, intel_connector, name);
> +
> /* Cache some DPCD data in the eDP case */
> if (is_edp(intel_dp)) {
> bool ret;
> @@ -2543,6 +2570,7 @@ intel_dp_init(struct drm_device *dev, int output_reg)
>
> ironlake_edp_panel_vdd_on(intel_dp);
> ret = intel_dp_get_dpcd(intel_dp);
> + intel_dp->edid = drm_get_edid(connector, &intel_dp->adapter);
> ironlake_edp_panel_vdd_off(intel_dp, false);
>
> if (ret) {
> @@ -2559,8 +2587,6 @@ intel_dp_init(struct drm_device *dev, int output_reg)
> }
> }
>
> - intel_dp_i2c_init(intel_dp, intel_connector, name);
> -
> intel_encoder->hot_plug = intel_dp_hot_plug;
>
> if (is_edp(intel_dp)) {
> diff --git a/drivers/gpu/drm/i915/intel_drv.h b/drivers/gpu/drm/i915/intel_drv.h
> index 3e09188..316413d 100644
> --- a/drivers/gpu/drm/i915/intel_drv.h
> +++ b/drivers/gpu/drm/i915/intel_drv.h
> @@ -336,6 +336,7 @@ struct intel_fbc_work {
>
> int intel_ddc_get_modes(struct drm_connector *c, struct i2c_adapter *adapter);
> extern bool intel_ddc_probe(struct intel_encoder *intel_encoder, int ddc_bus);
> +int intel_connector_attach_edid(struct drm_connector *connector, struct edid *edid);
>
> extern void intel_attach_force_audio_property(struct drm_connector *connector);
> extern void intel_attach_broadcast_rgb_property(struct drm_connector *connector);
> diff --git a/drivers/gpu/drm/i915/intel_modes.c b/drivers/gpu/drm/i915/intel_modes.c
> index d67ec3a..089ed8b 100644
> --- a/drivers/gpu/drm/i915/intel_modes.c
> +++ b/drivers/gpu/drm/i915/intel_modes.c
> @@ -60,6 +60,18 @@ bool intel_ddc_probe(struct intel_encoder *intel_encoder, int ddc_bus)
> msgs, 2) == 2;
> }
>
> +int intel_connector_attach_edid(struct drm_connector *connector,
> + struct edid *edid)
> +{
> + int ret;
> +
> + drm_mode_connector_update_edid_property(connector, edid);
> + ret = drm_add_edid_modes(connector, edid);
> + drm_edid_to_eld(connector, edid);
> +
> + return ret;
> +}
> +
> /**
> * intel_ddc_get_modes - get modelist from monitor
> * @connector: DRM connector device to use
> @@ -71,16 +83,12 @@ int intel_ddc_get_modes(struct drm_connector *connector,
> struct i2c_adapter *adapter)
> {
> struct edid *edid;
> - int ret = 0;
> + int ret;
>
> edid = drm_get_edid(connector, adapter);
> - if (edid) {
> - drm_mode_connector_update_edid_property(connector, edid);
> - ret = drm_add_edid_modes(connector, edid);
> - drm_edid_to_eld(connector, edid);
> - connector->display_info.raw_edid = NULL;
> - kfree(edid);
> - }
> + ret = intel_connector_attach_edid(connector, edid);
> + connector->display_info.raw_edid = NULL;
> + kfree(edid);
>
> return ret;
> }
> --
> 1.7.10
>
> _______________________________________________
> Intel-gfx mailing list
> Intel-gfx at lists.freedesktop.org
> http://lists.freedesktop.org/mailman/listinfo/intel-gfx
More information about the Intel-gfx
mailing list