drm/fb-helper: atomic restore_fbdev_mode()..
Add support for using atomic code-paths for restore_fbdev_mode(). Signed-off-by: Rob Clark <robdclark@gmail.com> [danvet: Bikeshed comments slightly.] Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
This commit is contained in:
parent
9685cd9df7
commit
bbb1e52402
|
@ -1553,21 +1553,9 @@ retry:
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = drm_atomic_set_crtc_for_plane(plane_state, NULL);
|
ret = __drm_atomic_helper_disable_plane(plane, plane_state);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
drm_atomic_set_fb_for_plane(plane_state, NULL);
|
|
||||||
plane_state->crtc_x = 0;
|
|
||||||
plane_state->crtc_y = 0;
|
|
||||||
plane_state->crtc_h = 0;
|
|
||||||
plane_state->crtc_w = 0;
|
|
||||||
plane_state->src_x = 0;
|
|
||||||
plane_state->src_y = 0;
|
|
||||||
plane_state->src_h = 0;
|
|
||||||
plane_state->src_w = 0;
|
|
||||||
|
|
||||||
if (plane == plane->crtc->cursor)
|
|
||||||
state->legacy_cursor_update = true;
|
|
||||||
|
|
||||||
ret = drm_atomic_commit(state);
|
ret = drm_atomic_commit(state);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
|
@ -1597,6 +1585,32 @@ backoff:
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(drm_atomic_helper_disable_plane);
|
EXPORT_SYMBOL(drm_atomic_helper_disable_plane);
|
||||||
|
|
||||||
|
/* just used from fb-helper and atomic-helper: */
|
||||||
|
int __drm_atomic_helper_disable_plane(struct drm_plane *plane,
|
||||||
|
struct drm_plane_state *plane_state)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = drm_atomic_set_crtc_for_plane(plane_state, NULL);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
drm_atomic_set_fb_for_plane(plane_state, NULL);
|
||||||
|
plane_state->crtc_x = 0;
|
||||||
|
plane_state->crtc_y = 0;
|
||||||
|
plane_state->crtc_h = 0;
|
||||||
|
plane_state->crtc_w = 0;
|
||||||
|
plane_state->src_x = 0;
|
||||||
|
plane_state->src_y = 0;
|
||||||
|
plane_state->src_h = 0;
|
||||||
|
plane_state->src_w = 0;
|
||||||
|
|
||||||
|
if (plane->crtc && (plane == plane->crtc->cursor))
|
||||||
|
plane_state->state->legacy_cursor_update = true;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int update_output_state(struct drm_atomic_state *state,
|
static int update_output_state(struct drm_atomic_state *state,
|
||||||
struct drm_mode_set *set)
|
struct drm_mode_set *set)
|
||||||
{
|
{
|
||||||
|
@ -1680,8 +1694,6 @@ int drm_atomic_helper_set_config(struct drm_mode_set *set)
|
||||||
{
|
{
|
||||||
struct drm_atomic_state *state;
|
struct drm_atomic_state *state;
|
||||||
struct drm_crtc *crtc = set->crtc;
|
struct drm_crtc *crtc = set->crtc;
|
||||||
struct drm_crtc_state *crtc_state;
|
|
||||||
struct drm_plane_state *primary_state;
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
state = drm_atomic_state_alloc(crtc->dev);
|
state = drm_atomic_state_alloc(crtc->dev);
|
||||||
|
@ -1690,64 +1702,10 @@ int drm_atomic_helper_set_config(struct drm_mode_set *set)
|
||||||
|
|
||||||
state->acquire_ctx = drm_modeset_legacy_acquire_ctx(crtc);
|
state->acquire_ctx = drm_modeset_legacy_acquire_ctx(crtc);
|
||||||
retry:
|
retry:
|
||||||
crtc_state = drm_atomic_get_crtc_state(state, crtc);
|
ret = __drm_atomic_helper_set_config(set, state);
|
||||||
if (IS_ERR(crtc_state)) {
|
|
||||||
ret = PTR_ERR(crtc_state);
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
primary_state = drm_atomic_get_plane_state(state, crtc->primary);
|
|
||||||
if (IS_ERR(primary_state)) {
|
|
||||||
ret = PTR_ERR(primary_state);
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!set->mode) {
|
|
||||||
WARN_ON(set->fb);
|
|
||||||
WARN_ON(set->num_connectors);
|
|
||||||
|
|
||||||
ret = drm_atomic_set_mode_for_crtc(crtc_state, NULL);
|
|
||||||
if (ret != 0)
|
|
||||||
goto fail;
|
|
||||||
|
|
||||||
crtc_state->active = false;
|
|
||||||
|
|
||||||
ret = drm_atomic_set_crtc_for_plane(primary_state, NULL);
|
|
||||||
if (ret != 0)
|
|
||||||
goto fail;
|
|
||||||
|
|
||||||
drm_atomic_set_fb_for_plane(primary_state, NULL);
|
|
||||||
|
|
||||||
goto commit;
|
|
||||||
}
|
|
||||||
|
|
||||||
WARN_ON(!set->fb);
|
|
||||||
WARN_ON(!set->num_connectors);
|
|
||||||
|
|
||||||
ret = drm_atomic_set_mode_for_crtc(crtc_state, set->mode);
|
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
crtc_state->active = true;
|
|
||||||
|
|
||||||
ret = drm_atomic_set_crtc_for_plane(primary_state, crtc);
|
|
||||||
if (ret != 0)
|
|
||||||
goto fail;
|
|
||||||
drm_atomic_set_fb_for_plane(primary_state, set->fb);
|
|
||||||
primary_state->crtc_x = 0;
|
|
||||||
primary_state->crtc_y = 0;
|
|
||||||
primary_state->crtc_h = set->mode->vdisplay;
|
|
||||||
primary_state->crtc_w = set->mode->hdisplay;
|
|
||||||
primary_state->src_x = set->x << 16;
|
|
||||||
primary_state->src_y = set->y << 16;
|
|
||||||
primary_state->src_h = set->mode->vdisplay << 16;
|
|
||||||
primary_state->src_w = set->mode->hdisplay << 16;
|
|
||||||
|
|
||||||
commit:
|
|
||||||
ret = update_output_state(state, set);
|
|
||||||
if (ret)
|
|
||||||
goto fail;
|
|
||||||
|
|
||||||
ret = drm_atomic_commit(state);
|
ret = drm_atomic_commit(state);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
@ -1776,6 +1734,73 @@ backoff:
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(drm_atomic_helper_set_config);
|
EXPORT_SYMBOL(drm_atomic_helper_set_config);
|
||||||
|
|
||||||
|
/* just used from fb-helper and atomic-helper: */
|
||||||
|
int __drm_atomic_helper_set_config(struct drm_mode_set *set,
|
||||||
|
struct drm_atomic_state *state)
|
||||||
|
{
|
||||||
|
struct drm_crtc_state *crtc_state;
|
||||||
|
struct drm_plane_state *primary_state;
|
||||||
|
struct drm_crtc *crtc = set->crtc;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
crtc_state = drm_atomic_get_crtc_state(state, crtc);
|
||||||
|
if (IS_ERR(crtc_state))
|
||||||
|
return PTR_ERR(crtc_state);
|
||||||
|
|
||||||
|
primary_state = drm_atomic_get_plane_state(state, crtc->primary);
|
||||||
|
if (IS_ERR(primary_state))
|
||||||
|
return PTR_ERR(primary_state);
|
||||||
|
|
||||||
|
if (!set->mode) {
|
||||||
|
WARN_ON(set->fb);
|
||||||
|
WARN_ON(set->num_connectors);
|
||||||
|
|
||||||
|
ret = drm_atomic_set_mode_for_crtc(crtc_state, NULL);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
crtc_state->active = false;
|
||||||
|
|
||||||
|
ret = drm_atomic_set_crtc_for_plane(primary_state, NULL);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
drm_atomic_set_fb_for_plane(primary_state, NULL);
|
||||||
|
|
||||||
|
goto commit;
|
||||||
|
}
|
||||||
|
|
||||||
|
WARN_ON(!set->fb);
|
||||||
|
WARN_ON(!set->num_connectors);
|
||||||
|
|
||||||
|
ret = drm_atomic_set_mode_for_crtc(crtc_state, set->mode);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
crtc_state->active = true;
|
||||||
|
|
||||||
|
ret = drm_atomic_set_crtc_for_plane(primary_state, crtc);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
drm_atomic_set_fb_for_plane(primary_state, set->fb);
|
||||||
|
primary_state->crtc_x = 0;
|
||||||
|
primary_state->crtc_y = 0;
|
||||||
|
primary_state->crtc_h = set->mode->vdisplay;
|
||||||
|
primary_state->crtc_w = set->mode->hdisplay;
|
||||||
|
primary_state->src_x = set->x << 16;
|
||||||
|
primary_state->src_y = set->y << 16;
|
||||||
|
primary_state->src_h = set->mode->vdisplay << 16;
|
||||||
|
primary_state->src_w = set->mode->hdisplay << 16;
|
||||||
|
|
||||||
|
commit:
|
||||||
|
ret = update_output_state(state, set);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* drm_atomic_helper_crtc_set_property - helper for crtc properties
|
* drm_atomic_helper_crtc_set_property - helper for crtc properties
|
||||||
* @crtc: DRM crtc
|
* @crtc: DRM crtc
|
||||||
|
|
|
@ -38,6 +38,8 @@
|
||||||
#include <drm/drm_crtc.h>
|
#include <drm/drm_crtc.h>
|
||||||
#include <drm/drm_fb_helper.h>
|
#include <drm/drm_fb_helper.h>
|
||||||
#include <drm/drm_crtc_helper.h>
|
#include <drm/drm_crtc_helper.h>
|
||||||
|
#include <drm/drm_atomic.h>
|
||||||
|
#include <drm/drm_atomic_helper.h>
|
||||||
|
|
||||||
static bool drm_fbdev_emulation = true;
|
static bool drm_fbdev_emulation = true;
|
||||||
module_param_named(fbdev_emulation, drm_fbdev_emulation, bool, 0600);
|
module_param_named(fbdev_emulation, drm_fbdev_emulation, bool, 0600);
|
||||||
|
@ -334,6 +336,72 @@ int drm_fb_helper_debug_leave(struct fb_info *info)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(drm_fb_helper_debug_leave);
|
EXPORT_SYMBOL(drm_fb_helper_debug_leave);
|
||||||
|
|
||||||
|
static int restore_fbdev_mode_atomic(struct drm_fb_helper *fb_helper)
|
||||||
|
{
|
||||||
|
struct drm_device *dev = fb_helper->dev;
|
||||||
|
struct drm_plane *plane;
|
||||||
|
struct drm_atomic_state *state;
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
state = drm_atomic_state_alloc(dev);
|
||||||
|
if (!state)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
state->acquire_ctx = dev->mode_config.acquire_ctx;
|
||||||
|
retry:
|
||||||
|
drm_for_each_plane(plane, dev) {
|
||||||
|
struct drm_plane_state *plane_state;
|
||||||
|
|
||||||
|
plane_state = drm_atomic_get_plane_state(state, plane);
|
||||||
|
if (IS_ERR(plane_state)) {
|
||||||
|
ret = PTR_ERR(plane_state);
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = drm_atomic_plane_set_property(plane, plane_state,
|
||||||
|
dev->mode_config.rotation_property,
|
||||||
|
BIT(DRM_ROTATE_0));
|
||||||
|
if (ret != 0)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
/* disable non-primary: */
|
||||||
|
if (plane->type == DRM_PLANE_TYPE_PRIMARY)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
ret = __drm_atomic_helper_disable_plane(plane, plane_state);
|
||||||
|
if (ret != 0)
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
for(i = 0; i < fb_helper->crtc_count; i++) {
|
||||||
|
struct drm_mode_set *mode_set = &fb_helper->crtc_info[i].mode_set;
|
||||||
|
|
||||||
|
ret = __drm_atomic_helper_set_config(mode_set, state);
|
||||||
|
if (ret != 0)
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = drm_atomic_commit(state);
|
||||||
|
if (ret != 0)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (ret == -EDEADLK)
|
||||||
|
goto backoff;
|
||||||
|
|
||||||
|
drm_atomic_state_free(state);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
backoff:
|
||||||
|
drm_atomic_state_clear(state);
|
||||||
|
drm_atomic_legacy_backoff(state);
|
||||||
|
|
||||||
|
goto retry;
|
||||||
|
}
|
||||||
|
|
||||||
static int restore_fbdev_mode(struct drm_fb_helper *fb_helper)
|
static int restore_fbdev_mode(struct drm_fb_helper *fb_helper)
|
||||||
{
|
{
|
||||||
struct drm_device *dev = fb_helper->dev;
|
struct drm_device *dev = fb_helper->dev;
|
||||||
|
@ -342,6 +410,9 @@ static int restore_fbdev_mode(struct drm_fb_helper *fb_helper)
|
||||||
|
|
||||||
drm_warn_on_modeset_not_all_locked(dev);
|
drm_warn_on_modeset_not_all_locked(dev);
|
||||||
|
|
||||||
|
if (fb_helper->atomic)
|
||||||
|
return restore_fbdev_mode_atomic(fb_helper);
|
||||||
|
|
||||||
drm_for_each_plane(plane, dev) {
|
drm_for_each_plane(plane, dev) {
|
||||||
if (plane->type != DRM_PLANE_TYPE_PRIMARY)
|
if (plane->type != DRM_PLANE_TYPE_PRIMARY)
|
||||||
drm_plane_force_disable(plane);
|
drm_plane_force_disable(plane);
|
||||||
|
@ -644,6 +715,8 @@ int drm_fb_helper_init(struct drm_device *dev,
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fb_helper->atomic = !!drm_core_check_feature(dev, DRIVER_ATOMIC);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
out_free:
|
out_free:
|
||||||
drm_fb_helper_crtc_free(fb_helper);
|
drm_fb_helper_crtc_free(fb_helper);
|
||||||
|
|
|
@ -30,6 +30,8 @@
|
||||||
|
|
||||||
#include <drm/drm_crtc.h>
|
#include <drm/drm_crtc.h>
|
||||||
|
|
||||||
|
struct drm_atomic_state;
|
||||||
|
|
||||||
int drm_atomic_helper_check_modeset(struct drm_device *dev,
|
int drm_atomic_helper_check_modeset(struct drm_device *dev,
|
||||||
struct drm_atomic_state *state);
|
struct drm_atomic_state *state);
|
||||||
int drm_atomic_helper_check_planes(struct drm_device *dev,
|
int drm_atomic_helper_check_planes(struct drm_device *dev,
|
||||||
|
@ -73,7 +75,11 @@ int drm_atomic_helper_update_plane(struct drm_plane *plane,
|
||||||
uint32_t src_x, uint32_t src_y,
|
uint32_t src_x, uint32_t src_y,
|
||||||
uint32_t src_w, uint32_t src_h);
|
uint32_t src_w, uint32_t src_h);
|
||||||
int drm_atomic_helper_disable_plane(struct drm_plane *plane);
|
int drm_atomic_helper_disable_plane(struct drm_plane *plane);
|
||||||
|
int __drm_atomic_helper_disable_plane(struct drm_plane *plane,
|
||||||
|
struct drm_plane_state *plane_state);
|
||||||
int drm_atomic_helper_set_config(struct drm_mode_set *set);
|
int drm_atomic_helper_set_config(struct drm_mode_set *set);
|
||||||
|
int __drm_atomic_helper_set_config(struct drm_mode_set *set,
|
||||||
|
struct drm_atomic_state *state);
|
||||||
|
|
||||||
int drm_atomic_helper_crtc_set_property(struct drm_crtc *crtc,
|
int drm_atomic_helper_crtc_set_property(struct drm_crtc *crtc,
|
||||||
struct drm_property *property,
|
struct drm_property *property,
|
||||||
|
|
|
@ -134,6 +134,17 @@ struct drm_fb_helper {
|
||||||
/* we got a hotplug but fbdev wasn't running the console
|
/* we got a hotplug but fbdev wasn't running the console
|
||||||
delay until next set_par */
|
delay until next set_par */
|
||||||
bool delayed_hotplug;
|
bool delayed_hotplug;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @atomic:
|
||||||
|
*
|
||||||
|
* Use atomic updates for restore_fbdev_mode(), etc. This defaults to
|
||||||
|
* true if driver has DRIVER_ATOMIC feature flag, but drivers can
|
||||||
|
* override it to true after drm_fb_helper_init() if they support atomic
|
||||||
|
* modeset but do not yet advertise DRIVER_ATOMIC (note that fb-helper
|
||||||
|
* does not require ASYNC commits).
|
||||||
|
*/
|
||||||
|
bool atomic;
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_DRM_FBDEV_EMULATION
|
#ifdef CONFIG_DRM_FBDEV_EMULATION
|
||||||
|
|
Loading…
Reference in New Issue