mirror of https://gitee.com/openkylin/linux.git
drm/i915: Update LVDS connector status when receiving ACPI LID event
Dirk reports that nothing is displayed on LVDS when using ubuntu 9.1 after close/reopen the LID. And I also reproduce this issue on another laptop. After some tests and debug, it seems that it is related with that the LVDS status is not updated in time in course of suspend/resume. Now the LID state is used to check whether the LVDS is connected or disconnected. And when the LID is closed, it means that the LVDS is disconnected. When it is reopened, it means that the LVDS is connected. At the same time on some distributions the LID event is also used to put the system into suspend state. When the LID is closed, the system will enter the suspend state. When the LID is reopened, the system will be resumed. In such case when the LID is closed, user-space script will receive the LID notification event and detect the LVDS as disconnected. Then the system will enter the suspended state. When the LID is reopened, the system will be resumed. As the LVDS status is not updated in course of resume, it will cause that the LVDS connector is marked as unused and disabled. After the resume is finished,user-space script will try to configure the display mode for LVDS. But unfortunately as the LVDS status is not updated in time and it is still marked as disconnected, the LVDS and its corresponding CRTC will be disabled again in the function of drm_helper_disable_unused_functions after changing mode for LVDS. So we had better check and update the status of LVDS connector after receiving the LID notication event. Then after the system is resumed from suspended state, we can set the display mode for LVDS correctly. Signed-off-by: Zhao Yakui <yakui.zhao@intel.com> Reported-by: Dirk Hohndel <hohndel@infradead.org> Reviewed-by: Jesse Barnes <jbarnes@virtuousgeek.org> CC: stable@kernel.org Signed-off-by: Eric Anholt <eric@anholt.net>
This commit is contained in:
parent
a3cb5195f6
commit
a2565377a5
drivers/gpu/drm/i915
|
@ -561,6 +561,7 @@ typedef struct drm_i915_private {
|
||||||
u16 orig_clock;
|
u16 orig_clock;
|
||||||
int child_dev_num;
|
int child_dev_num;
|
||||||
struct child_device_config *child_dev;
|
struct child_device_config *child_dev;
|
||||||
|
struct drm_connector *int_lvds_connector;
|
||||||
} drm_i915_private_t;
|
} drm_i915_private_t;
|
||||||
|
|
||||||
/** driver private structure attached to each drm_gem_object */
|
/** driver private structure attached to each drm_gem_object */
|
||||||
|
|
|
@ -686,7 +686,14 @@ static int intel_lid_notify(struct notifier_block *nb, unsigned long val,
|
||||||
struct drm_i915_private *dev_priv =
|
struct drm_i915_private *dev_priv =
|
||||||
container_of(nb, struct drm_i915_private, lid_notifier);
|
container_of(nb, struct drm_i915_private, lid_notifier);
|
||||||
struct drm_device *dev = dev_priv->dev;
|
struct drm_device *dev = dev_priv->dev;
|
||||||
|
struct drm_connector *connector = dev_priv->int_lvds_connector;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* check and update the status of LVDS connector after receiving
|
||||||
|
* the LID nofication event.
|
||||||
|
*/
|
||||||
|
if (connector)
|
||||||
|
connector->status = connector->funcs->detect(connector);
|
||||||
if (!acpi_lid_open()) {
|
if (!acpi_lid_open()) {
|
||||||
dev_priv->modeset_on_lid = 1;
|
dev_priv->modeset_on_lid = 1;
|
||||||
return NOTIFY_OK;
|
return NOTIFY_OK;
|
||||||
|
@ -1124,6 +1131,8 @@ void intel_lvds_init(struct drm_device *dev)
|
||||||
DRM_DEBUG_KMS("lid notifier registration failed\n");
|
DRM_DEBUG_KMS("lid notifier registration failed\n");
|
||||||
dev_priv->lid_notifier.notifier_call = NULL;
|
dev_priv->lid_notifier.notifier_call = NULL;
|
||||||
}
|
}
|
||||||
|
/* keep the LVDS connector */
|
||||||
|
dev_priv->int_lvds_connector = connector;
|
||||||
drm_sysfs_connector_add(connector);
|
drm_sysfs_connector_add(connector);
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue