Commit ac235daf authored by Ville Syrjälä's avatar Ville Syrjälä Committed by Dave Airlie

drm: Fix drm_mode_attachmode_crtc()

Change drm_mode_attachmode_crtc() to take an "all or nothing" approach.
If an error is returned, there are no side effects visible.

Also change the function to always duplicate the mode passed in.

Also change the function to not give up when it finds the first
connector without and encoder.

A simpler approach would be to just remove the function completely as
it's unused currently.
Signed-off-by: default avatarVille Syrjälä <ville.syrjala@linux.intel.com>
Reviewed-by: default avatarAlex Deucher <alexander.deucher@amd.com>
Signed-off-by: default avatarDave Airlie <airlied@redhat.com>
parent 5f61bb42
...@@ -2421,24 +2421,40 @@ static void drm_mode_attachmode(struct drm_device *dev, ...@@ -2421,24 +2421,40 @@ static void drm_mode_attachmode(struct drm_device *dev,
} }
int drm_mode_attachmode_crtc(struct drm_device *dev, struct drm_crtc *crtc, int drm_mode_attachmode_crtc(struct drm_device *dev, struct drm_crtc *crtc,
struct drm_display_mode *mode) const struct drm_display_mode *mode)
{ {
struct drm_connector *connector; struct drm_connector *connector;
struct drm_display_mode *dup_mode; int ret = 0;
int need_dup = 0; struct drm_display_mode *dup_mode, *next;
LIST_HEAD(list);
list_for_each_entry(connector, &dev->mode_config.connector_list, head) { list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
if (!connector->encoder) if (!connector->encoder)
break; continue;
if (connector->encoder->crtc == crtc) { if (connector->encoder->crtc == crtc) {
if (need_dup) dup_mode = drm_mode_duplicate(dev, mode);
dup_mode = drm_mode_duplicate(dev, mode); if (!dup_mode) {
else ret = -ENOMEM;
dup_mode = mode; goto out;
drm_mode_attachmode(dev, connector, dup_mode); }
need_dup = 1; list_add_tail(&dup_mode->head, &list);
} }
} }
return 0;
list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
if (!connector->encoder)
continue;
if (connector->encoder->crtc == crtc)
list_move_tail(list.next, &connector->user_modes);
}
WARN_ON(!list_empty(&list));
out:
list_for_each_entry_safe(dup_mode, next, &list, head)
drm_mode_destroy(dev, dup_mode);
return ret;
} }
EXPORT_SYMBOL(drm_mode_attachmode_crtc); EXPORT_SYMBOL(drm_mode_attachmode_crtc);
......
...@@ -869,7 +869,7 @@ extern int drm_mode_height(struct drm_display_mode *mode); ...@@ -869,7 +869,7 @@ extern int drm_mode_height(struct drm_display_mode *mode);
/* for us by fb module */ /* for us by fb module */
extern int drm_mode_attachmode_crtc(struct drm_device *dev, extern int drm_mode_attachmode_crtc(struct drm_device *dev,
struct drm_crtc *crtc, struct drm_crtc *crtc,
struct drm_display_mode *mode); const struct drm_display_mode *mode);
extern int drm_mode_detachmode_crtc(struct drm_device *dev, struct drm_display_mode *mode); extern int drm_mode_detachmode_crtc(struct drm_device *dev, struct drm_display_mode *mode);
extern struct drm_display_mode *drm_mode_create(struct drm_device *dev); extern struct drm_display_mode *drm_mode_create(struct drm_device *dev);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment