modesetting: Fix dirty updates for sw rotation
Rotation is broken for all drm drivers not providing hardware rotation support. Drivers that give direct access to vram and not needing dirty updates still work but only by accident. The problem is caused by modesetting not sending the correct fb_id to drmModeDirtyFB() and passing the damage rects in the rotated state and not as the crtc expects them. This patch takes care of both problems. Signed-off-by: Patrik Jakobsson <pjakobsson@suse.de>
This commit is contained in:
		
							parent
							
								
									ecdf21035e
								
							
						
					
					
						commit
						db9e9d45e8
					
				|  | @ -515,8 +515,40 @@ GetRec(ScrnInfoPtr pScrn) | ||||||
|     return TRUE; |     return TRUE; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void | ||||||
|  | rotate_clip(PixmapPtr pixmap, BoxPtr rect, drmModeClip *clip, Rotation rotation) | ||||||
|  | { | ||||||
|  |     int w = pixmap->drawable.width; | ||||||
|  |     int h = pixmap->drawable.height; | ||||||
|  | 
 | ||||||
|  |     if (rotation == RR_Rotate_90) { | ||||||
|  | 	/* Rotate 90 degrees counter clockwise */ | ||||||
|  |         clip->x1 = rect->y1; | ||||||
|  | 	clip->x2 = rect->y2; | ||||||
|  | 	clip->y1 = w - rect->x2; | ||||||
|  | 	clip->y2 = w - rect->x1; | ||||||
|  |     } else if (rotation == RR_Rotate_180) { | ||||||
|  | 	/* Rotate 180 degrees */ | ||||||
|  |         clip->x1 = w - rect->x2; | ||||||
|  | 	clip->x2 = w - rect->x1; | ||||||
|  | 	clip->y1 = h - rect->y2; | ||||||
|  | 	clip->y2 = h - rect->y1; | ||||||
|  |     } else if (rotation == RR_Rotate_270) { | ||||||
|  | 	/* Rotate 90 degrees clockwise */ | ||||||
|  |         clip->x1 = h - rect->y2; | ||||||
|  | 	clip->x2 = h - rect->y1; | ||||||
|  | 	clip->y1 = rect->x1; | ||||||
|  | 	clip->y2 = rect->x2; | ||||||
|  |     } else { | ||||||
|  | 	clip->x1 = rect->x1; | ||||||
|  | 	clip->x2 = rect->x2; | ||||||
|  | 	clip->y1 = rect->y1; | ||||||
|  | 	clip->y2 = rect->y2; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int | static int | ||||||
| dispatch_dirty_region(ScrnInfoPtr scrn, | dispatch_dirty_region(ScrnInfoPtr scrn, xf86CrtcPtr crtc, | ||||||
| 		      PixmapPtr pixmap, DamagePtr damage, int fb_id) | 		      PixmapPtr pixmap, DamagePtr damage, int fb_id) | ||||||
| { | { | ||||||
|     modesettingPtr ms = modesettingPTR(scrn); |     modesettingPtr ms = modesettingPTR(scrn); | ||||||
|  | @ -532,13 +564,9 @@ dispatch_dirty_region(ScrnInfoPtr scrn, | ||||||
|         if (!clip) |         if (!clip) | ||||||
|             return -ENOMEM; |             return -ENOMEM; | ||||||
| 
 | 
 | ||||||
|         /* XXX no need for copy? */ |         /* Rotate and copy rects into clips */ | ||||||
|         for (i = 0; i < num_cliprects; i++, rect++) { |         for (i = 0; i < num_cliprects; i++, rect++) | ||||||
|             clip[i].x1 = rect->x1; | 	    rotate_clip(pixmap, rect, &clip[i], crtc->rotation); | ||||||
|             clip[i].y1 = rect->y1; |  | ||||||
|             clip[i].x2 = rect->x2; |  | ||||||
|             clip[i].y2 = rect->y2; |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         /* TODO query connector property to see if this is needed */ |         /* TODO query connector property to see if this is needed */ | ||||||
|         ret = drmModeDirtyFB(ms->fd, fb_id, clip, num_cliprects); |         ret = drmModeDirtyFB(ms->fd, fb_id, clip, num_cliprects); | ||||||
|  | @ -561,12 +589,22 @@ static void | ||||||
| dispatch_dirty(ScreenPtr pScreen) | dispatch_dirty(ScreenPtr pScreen) | ||||||
| { | { | ||||||
|     ScrnInfoPtr scrn = xf86ScreenToScrn(pScreen); |     ScrnInfoPtr scrn = xf86ScreenToScrn(pScreen); | ||||||
|  |     xf86CrtcConfigPtr xf86_config = XF86_CRTC_CONFIG_PTR(scrn); | ||||||
|     modesettingPtr ms = modesettingPTR(scrn); |     modesettingPtr ms = modesettingPTR(scrn); | ||||||
|     PixmapPtr pixmap = pScreen->GetScreenPixmap(pScreen); |     PixmapPtr pixmap = pScreen->GetScreenPixmap(pScreen); | ||||||
|     int fb_id = ms->drmmode.fb_id; |     uint32_t fb_id; | ||||||
|     int ret; |     int ret, c, x, y ; | ||||||
| 
 | 
 | ||||||
|     ret = dispatch_dirty_region(scrn, pixmap, ms->damage, fb_id); |     for (c = 0; c < xf86_config->num_crtc; c++) { | ||||||
|  |         xf86CrtcPtr crtc = xf86_config->crtc[c]; | ||||||
|  |         drmmode_crtc_private_ptr drmmode_crtc = crtc->driver_private; | ||||||
|  | 
 | ||||||
|  |         if (!drmmode_crtc) | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  | 	drmmode_crtc_get_fb_id(crtc, &fb_id, &x, &y); | ||||||
|  | 
 | ||||||
|  |         ret = dispatch_dirty_region(scrn, crtc, pixmap, ms->damage, fb_id); | ||||||
|         if (ret == -EINVAL || ret == -ENOSYS) { |         if (ret == -EINVAL || ret == -ENOSYS) { | ||||||
|             ms->dirty_enabled = FALSE; |             ms->dirty_enabled = FALSE; | ||||||
|             DamageUnregister(ms->damage); |             DamageUnregister(ms->damage); | ||||||
|  | @ -577,6 +615,7 @@ dispatch_dirty(ScreenPtr pScreen) | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| static void | static void | ||||||
| dispatch_dirty_pixmap(ScrnInfoPtr scrn, xf86CrtcPtr crtc, PixmapPtr ppix) | dispatch_dirty_pixmap(ScrnInfoPtr scrn, xf86CrtcPtr crtc, PixmapPtr ppix) | ||||||
|  | @ -586,7 +625,7 @@ dispatch_dirty_pixmap(ScrnInfoPtr scrn, xf86CrtcPtr crtc, PixmapPtr ppix) | ||||||
|     DamagePtr damage = ppriv->secondary_damage; |     DamagePtr damage = ppriv->secondary_damage; | ||||||
|     int fb_id = ppriv->fb_id; |     int fb_id = ppriv->fb_id; | ||||||
| 
 | 
 | ||||||
|     dispatch_dirty_region(scrn, ppix, damage, fb_id); |     dispatch_dirty_region(scrn, crtc, ppix, damage, fb_id); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void | static void | ||||||
|  |  | ||||||
|  | @ -627,7 +627,7 @@ drmmode_crtc_can_test_mode(xf86CrtcPtr crtc) | ||||||
|     return ms->atomic_modeset; |     return ms->atomic_modeset; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static Bool | Bool | ||||||
| drmmode_crtc_get_fb_id(xf86CrtcPtr crtc, uint32_t *fb_id, int *x, int *y) | drmmode_crtc_get_fb_id(xf86CrtcPtr crtc, uint32_t *fb_id, int *x, int *y) | ||||||
| { | { | ||||||
|     drmmode_crtc_private_ptr drmmode_crtc = crtc->driver_private; |     drmmode_crtc_private_ptr drmmode_crtc = crtc->driver_private; | ||||||
|  |  | ||||||
|  | @ -311,6 +311,8 @@ void drmmode_copy_fb(ScrnInfoPtr pScrn, drmmode_ptr drmmode); | ||||||
| 
 | 
 | ||||||
| int drmmode_crtc_flip(xf86CrtcPtr crtc, uint32_t fb_id, uint32_t flags, void *data); | int drmmode_crtc_flip(xf86CrtcPtr crtc, uint32_t fb_id, uint32_t flags, void *data); | ||||||
| 
 | 
 | ||||||
|  | Bool drmmode_crtc_get_fb_id(xf86CrtcPtr crtc, uint32_t *fb_id, int *x, int *y); | ||||||
|  | 
 | ||||||
| void drmmode_set_dpms(ScrnInfoPtr scrn, int PowerManagementMode, int flags); | void drmmode_set_dpms(ScrnInfoPtr scrn, int PowerManagementMode, int flags); | ||||||
| void drmmode_crtc_set_vrr(xf86CrtcPtr crtc, Bool enabled); | void drmmode_crtc_set_vrr(xf86CrtcPtr crtc, Bool enabled); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue