|
@@ -342,6 +342,7 @@ static int restore_fbdev_mode_atomic(struct drm_fb_helper *fb_helper)
|
|
|
struct drm_plane *plane;
|
|
|
struct drm_atomic_state *state;
|
|
|
int i, ret;
|
|
|
+ unsigned plane_mask;
|
|
|
|
|
|
state = drm_atomic_state_alloc(dev);
|
|
|
if (!state)
|
|
@@ -349,11 +350,10 @@ static int restore_fbdev_mode_atomic(struct drm_fb_helper *fb_helper)
|
|
|
|
|
|
state->acquire_ctx = dev->mode_config.acquire_ctx;
|
|
|
retry:
|
|
|
+ plane_mask = 0;
|
|
|
drm_for_each_plane(plane, dev) {
|
|
|
struct drm_plane_state *plane_state;
|
|
|
|
|
|
- plane->old_fb = plane->fb;
|
|
|
-
|
|
|
plane_state = drm_atomic_get_plane_state(state, plane);
|
|
|
if (IS_ERR(plane_state)) {
|
|
|
ret = PTR_ERR(plane_state);
|
|
@@ -362,6 +362,9 @@ retry:
|
|
|
|
|
|
plane_state->rotation = BIT(DRM_ROTATE_0);
|
|
|
|
|
|
+ plane->old_fb = plane->fb;
|
|
|
+ plane_mask |= 1 << drm_plane_index(plane);
|
|
|
+
|
|
|
/* disable non-primary: */
|
|
|
if (plane->type == DRM_PLANE_TYPE_PRIMARY)
|
|
|
continue;
|
|
@@ -382,19 +385,7 @@ retry:
|
|
|
ret = drm_atomic_commit(state);
|
|
|
|
|
|
fail:
|
|
|
- drm_for_each_plane(plane, dev) {
|
|
|
- if (ret == 0) {
|
|
|
- struct drm_framebuffer *new_fb = plane->state->fb;
|
|
|
- if (new_fb)
|
|
|
- drm_framebuffer_reference(new_fb);
|
|
|
- plane->fb = new_fb;
|
|
|
- plane->crtc = plane->state->crtc;
|
|
|
-
|
|
|
- if (plane->old_fb)
|
|
|
- drm_framebuffer_unreference(plane->old_fb);
|
|
|
- }
|
|
|
- plane->old_fb = NULL;
|
|
|
- }
|
|
|
+ drm_atomic_clean_old_fb(dev, plane_mask, ret);
|
|
|
|
|
|
if (ret == -EDEADLK)
|
|
|
goto backoff;
|
|
@@ -1236,7 +1227,9 @@ static int pan_display_atomic(struct fb_var_screeninfo *var,
|
|
|
struct drm_fb_helper *fb_helper = info->par;
|
|
|
struct drm_device *dev = fb_helper->dev;
|
|
|
struct drm_atomic_state *state;
|
|
|
+ struct drm_plane *plane;
|
|
|
int i, ret;
|
|
|
+ unsigned plane_mask;
|
|
|
|
|
|
state = drm_atomic_state_alloc(dev);
|
|
|
if (!state)
|
|
@@ -1244,19 +1237,22 @@ static int pan_display_atomic(struct fb_var_screeninfo *var,
|
|
|
|
|
|
state->acquire_ctx = dev->mode_config.acquire_ctx;
|
|
|
retry:
|
|
|
+ plane_mask = 0;
|
|
|
for(i = 0; i < fb_helper->crtc_count; i++) {
|
|
|
struct drm_mode_set *mode_set;
|
|
|
|
|
|
mode_set = &fb_helper->crtc_info[i].mode_set;
|
|
|
|
|
|
- mode_set->crtc->primary->old_fb = mode_set->crtc->primary->fb;
|
|
|
-
|
|
|
mode_set->x = var->xoffset;
|
|
|
mode_set->y = var->yoffset;
|
|
|
|
|
|
ret = __drm_atomic_helper_set_config(mode_set, state);
|
|
|
if (ret != 0)
|
|
|
goto fail;
|
|
|
+
|
|
|
+ plane = mode_set->crtc->primary;
|
|
|
+ plane_mask |= drm_plane_index(plane);
|
|
|
+ plane->old_fb = plane->fb;
|
|
|
}
|
|
|
|
|
|
ret = drm_atomic_commit(state);
|
|
@@ -1268,26 +1264,7 @@ retry:
|
|
|
|
|
|
|
|
|
fail:
|
|
|
- for(i = 0; i < fb_helper->crtc_count; i++) {
|
|
|
- struct drm_mode_set *mode_set;
|
|
|
- struct drm_plane *plane;
|
|
|
-
|
|
|
- mode_set = &fb_helper->crtc_info[i].mode_set;
|
|
|
- plane = mode_set->crtc->primary;
|
|
|
-
|
|
|
- if (ret == 0) {
|
|
|
- struct drm_framebuffer *new_fb = plane->state->fb;
|
|
|
-
|
|
|
- if (new_fb)
|
|
|
- drm_framebuffer_reference(new_fb);
|
|
|
- plane->fb = new_fb;
|
|
|
- plane->crtc = plane->state->crtc;
|
|
|
-
|
|
|
- if (plane->old_fb)
|
|
|
- drm_framebuffer_unreference(plane->old_fb);
|
|
|
- }
|
|
|
- plane->old_fb = NULL;
|
|
|
- }
|
|
|
+ drm_atomic_clean_old_fb(dev, plane_mask, ret);
|
|
|
|
|
|
if (ret == -EDEADLK)
|
|
|
goto backoff;
|