2
0
mirror of git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git synced 2025-09-04 20:19:47 +08:00

drm/gud: Use the shadow plane helper

Use the shadow plane helper to take care of mapping the framebuffer for
CPU access. The synchronous flushing is now done inline without the use of
a worker. The async path now uses a shadow buffer to hold framebuffer
changes and it doesn't read the framebuffer behind userspace's back
anymore.

v2:
- Use src as variable name for iosys_map (Thomas)
- Prepare imported buffer for CPU access in the driver (Thomas)

Reviewed-by: Thomas Zimmermann <tzimmermann@suse.de>
Signed-off-by: Noralf Trønnes <noralf@tronnes.org>
Link: https://patchwork.freedesktop.org/patch/msgid/20221122-gud-shadow-plane-v2-5-435037990a83@tronnes.org
This commit is contained in:
Noralf Trønnes 2022-11-30 20:26:53 +01:00
parent 562fd7cc67
commit c17d048609
3 changed files with 57 additions and 26 deletions

View File

@ -365,6 +365,7 @@ static void gud_debugfs_init(struct drm_minor *minor)
static const struct drm_simple_display_pipe_funcs gud_pipe_funcs = { static const struct drm_simple_display_pipe_funcs gud_pipe_funcs = {
.check = gud_pipe_check, .check = gud_pipe_check,
.update = gud_pipe_update, .update = gud_pipe_update,
DRM_GEM_SIMPLE_DISPLAY_PIPE_SHADOW_PLANE_FUNCS
}; };
static const struct drm_mode_config_funcs gud_mode_config_funcs = { static const struct drm_mode_config_funcs gud_mode_config_funcs = {

View File

@ -43,6 +43,7 @@ struct gud_device {
struct drm_framebuffer *fb; struct drm_framebuffer *fb;
struct drm_rect damage; struct drm_rect damage;
bool prev_flush_failed; bool prev_flush_failed;
void *shadow_buf;
}; };
static inline struct gud_device *to_gud_device(struct drm_device *drm) static inline struct gud_device *to_gud_device(struct drm_device *drm)

View File

@ -358,10 +358,10 @@ static void gud_flush_damage(struct gud_device *gdrm, struct drm_framebuffer *fb
void gud_flush_work(struct work_struct *work) void gud_flush_work(struct work_struct *work)
{ {
struct gud_device *gdrm = container_of(work, struct gud_device, work); struct gud_device *gdrm = container_of(work, struct gud_device, work);
struct iosys_map gem_map = { }, fb_map = { }; struct iosys_map shadow_map;
struct drm_framebuffer *fb; struct drm_framebuffer *fb;
struct drm_rect damage; struct drm_rect damage;
int idx, ret; int idx;
if (!drm_dev_enter(&gdrm->drm, &idx)) if (!drm_dev_enter(&gdrm->drm, &idx))
return; return;
@ -369,6 +369,7 @@ void gud_flush_work(struct work_struct *work)
mutex_lock(&gdrm->damage_lock); mutex_lock(&gdrm->damage_lock);
fb = gdrm->fb; fb = gdrm->fb;
gdrm->fb = NULL; gdrm->fb = NULL;
iosys_map_set_vaddr(&shadow_map, gdrm->shadow_buf);
damage = gdrm->damage; damage = gdrm->damage;
gud_clear_damage(gdrm); gud_clear_damage(gdrm);
mutex_unlock(&gdrm->damage_lock); mutex_unlock(&gdrm->damage_lock);
@ -376,33 +377,33 @@ void gud_flush_work(struct work_struct *work)
if (!fb) if (!fb)
goto out; goto out;
ret = drm_gem_fb_vmap(fb, &gem_map, &fb_map); gud_flush_damage(gdrm, fb, &shadow_map, true, &damage);
if (ret)
goto fb_put;
ret = drm_gem_fb_begin_cpu_access(fb, DMA_FROM_DEVICE);
if (ret)
goto vunmap;
/* Imported buffers are assumed to be WriteCombined with uncached reads */
gud_flush_damage(gdrm, fb, &fb_map, !fb->obj[0]->import_attach, &damage);
drm_gem_fb_end_cpu_access(fb, DMA_FROM_DEVICE);
vunmap:
drm_gem_fb_vunmap(fb, &gem_map);
fb_put:
drm_framebuffer_put(fb); drm_framebuffer_put(fb);
out: out:
drm_dev_exit(idx); drm_dev_exit(idx);
} }
static void gud_fb_queue_damage(struct gud_device *gdrm, struct drm_framebuffer *fb, static int gud_fb_queue_damage(struct gud_device *gdrm, struct drm_framebuffer *fb,
struct drm_rect *damage) const struct iosys_map *src, struct drm_rect *damage)
{ {
struct drm_framebuffer *old_fb = NULL; struct drm_framebuffer *old_fb = NULL;
struct iosys_map shadow_map;
mutex_lock(&gdrm->damage_lock); mutex_lock(&gdrm->damage_lock);
if (!gdrm->shadow_buf) {
gdrm->shadow_buf = vzalloc(fb->pitches[0] * fb->height);
if (!gdrm->shadow_buf) {
mutex_unlock(&gdrm->damage_lock);
return -ENOMEM;
}
}
iosys_map_set_vaddr(&shadow_map, gdrm->shadow_buf);
iosys_map_incr(&shadow_map, drm_fb_clip_offset(fb->pitches[0], fb->format, damage));
drm_fb_memcpy(&shadow_map, fb->pitches, src, fb, damage);
if (fb != gdrm->fb) { if (fb != gdrm->fb) {
old_fb = gdrm->fb; old_fb = gdrm->fb;
drm_framebuffer_get(fb); drm_framebuffer_get(fb);
@ -420,6 +421,26 @@ static void gud_fb_queue_damage(struct gud_device *gdrm, struct drm_framebuffer
if (old_fb) if (old_fb)
drm_framebuffer_put(old_fb); drm_framebuffer_put(old_fb);
return 0;
}
static void gud_fb_handle_damage(struct gud_device *gdrm, struct drm_framebuffer *fb,
const struct iosys_map *src, struct drm_rect *damage)
{
int ret;
if (gdrm->flags & GUD_DISPLAY_FLAG_FULL_UPDATE)
drm_rect_init(damage, 0, 0, fb->width, fb->height);
if (gud_async_flush) {
ret = gud_fb_queue_damage(gdrm, fb, src, damage);
if (ret != -ENOMEM)
return;
}
/* Imported buffers are assumed to be WriteCombined with uncached reads */
gud_flush_damage(gdrm, fb, src, !fb->obj[0]->import_attach, damage);
} }
int gud_pipe_check(struct drm_simple_display_pipe *pipe, int gud_pipe_check(struct drm_simple_display_pipe *pipe,
@ -544,10 +565,11 @@ void gud_pipe_update(struct drm_simple_display_pipe *pipe,
struct drm_device *drm = pipe->crtc.dev; struct drm_device *drm = pipe->crtc.dev;
struct gud_device *gdrm = to_gud_device(drm); struct gud_device *gdrm = to_gud_device(drm);
struct drm_plane_state *state = pipe->plane.state; struct drm_plane_state *state = pipe->plane.state;
struct drm_shadow_plane_state *shadow_plane_state = to_drm_shadow_plane_state(state);
struct drm_framebuffer *fb = state->fb; struct drm_framebuffer *fb = state->fb;
struct drm_crtc *crtc = &pipe->crtc; struct drm_crtc *crtc = &pipe->crtc;
struct drm_rect damage; struct drm_rect damage;
int idx; int ret, idx;
if (crtc->state->mode_changed || !crtc->state->enable) { if (crtc->state->mode_changed || !crtc->state->enable) {
cancel_work_sync(&gdrm->work); cancel_work_sync(&gdrm->work);
@ -557,6 +579,8 @@ void gud_pipe_update(struct drm_simple_display_pipe *pipe,
gdrm->fb = NULL; gdrm->fb = NULL;
} }
gud_clear_damage(gdrm); gud_clear_damage(gdrm);
vfree(gdrm->shadow_buf);
gdrm->shadow_buf = NULL;
mutex_unlock(&gdrm->damage_lock); mutex_unlock(&gdrm->damage_lock);
} }
@ -572,14 +596,19 @@ void gud_pipe_update(struct drm_simple_display_pipe *pipe,
if (crtc->state->active_changed) if (crtc->state->active_changed)
gud_usb_set_u8(gdrm, GUD_REQ_SET_DISPLAY_ENABLE, crtc->state->active); gud_usb_set_u8(gdrm, GUD_REQ_SET_DISPLAY_ENABLE, crtc->state->active);
if (drm_atomic_helper_damage_merged(old_state, state, &damage)) { if (!fb)
if (gdrm->flags & GUD_DISPLAY_FLAG_FULL_UPDATE) goto ctrl_disable;
drm_rect_init(&damage, 0, 0, fb->width, fb->height);
gud_fb_queue_damage(gdrm, fb, &damage);
if (!gud_async_flush)
flush_work(&gdrm->work);
}
ret = drm_gem_fb_begin_cpu_access(fb, DMA_FROM_DEVICE);
if (ret)
goto ctrl_disable;
if (drm_atomic_helper_damage_merged(old_state, state, &damage))
gud_fb_handle_damage(gdrm, fb, &shadow_plane_state->data[0], &damage);
drm_gem_fb_end_cpu_access(fb, DMA_FROM_DEVICE);
ctrl_disable:
if (!crtc->state->enable) if (!crtc->state->enable)
gud_usb_set_u8(gdrm, GUD_REQ_SET_CONTROLLER_ENABLE, 0); gud_usb_set_u8(gdrm, GUD_REQ_SET_CONTROLLER_ENABLE, 0);