mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-20 11:34:02 +08:00
drm/i915: Unpin the flip target if we fail to queue the flip
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> Cc: Jesse Barnes <jbarnes@virtuousgeek.org> Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
This commit is contained in:
parent
a1e969e033
commit
83d4092b03
@ -7658,14 +7658,14 @@ static int intel_gen2_queue_flip(struct drm_device *dev,
|
|||||||
|
|
||||||
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err;
|
||||||
|
|
||||||
/* Offset into the new buffer for cases of shared fbs between CRTCs */
|
/* Offset into the new buffer for cases of shared fbs between CRTCs */
|
||||||
offset = crtc->y * fb->pitches[0] + crtc->x * fb->bits_per_pixel/8;
|
offset = crtc->y * fb->pitches[0] + crtc->x * fb->bits_per_pixel/8;
|
||||||
|
|
||||||
ret = BEGIN_LP_RING(6);
|
ret = BEGIN_LP_RING(6);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err_unpin;
|
||||||
|
|
||||||
/* Can't queue multiple flips, so wait for the previous
|
/* Can't queue multiple flips, so wait for the previous
|
||||||
* one to finish before executing the next.
|
* one to finish before executing the next.
|
||||||
@ -7682,7 +7682,11 @@ static int intel_gen2_queue_flip(struct drm_device *dev,
|
|||||||
OUT_RING(obj->gtt_offset + offset);
|
OUT_RING(obj->gtt_offset + offset);
|
||||||
OUT_RING(0); /* aux display base address, unused */
|
OUT_RING(0); /* aux display base address, unused */
|
||||||
ADVANCE_LP_RING();
|
ADVANCE_LP_RING();
|
||||||
out:
|
return 0;
|
||||||
|
|
||||||
|
err_unpin:
|
||||||
|
intel_unpin_fb_obj(obj);
|
||||||
|
err:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -7699,14 +7703,14 @@ static int intel_gen3_queue_flip(struct drm_device *dev,
|
|||||||
|
|
||||||
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err;
|
||||||
|
|
||||||
/* Offset into the new buffer for cases of shared fbs between CRTCs */
|
/* Offset into the new buffer for cases of shared fbs between CRTCs */
|
||||||
offset = crtc->y * fb->pitches[0] + crtc->x * fb->bits_per_pixel/8;
|
offset = crtc->y * fb->pitches[0] + crtc->x * fb->bits_per_pixel/8;
|
||||||
|
|
||||||
ret = BEGIN_LP_RING(6);
|
ret = BEGIN_LP_RING(6);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err_unpin;
|
||||||
|
|
||||||
if (intel_crtc->plane)
|
if (intel_crtc->plane)
|
||||||
flip_mask = MI_WAIT_FOR_PLANE_B_FLIP;
|
flip_mask = MI_WAIT_FOR_PLANE_B_FLIP;
|
||||||
@ -7721,7 +7725,11 @@ static int intel_gen3_queue_flip(struct drm_device *dev,
|
|||||||
OUT_RING(MI_NOOP);
|
OUT_RING(MI_NOOP);
|
||||||
|
|
||||||
ADVANCE_LP_RING();
|
ADVANCE_LP_RING();
|
||||||
out:
|
return 0;
|
||||||
|
|
||||||
|
err_unpin:
|
||||||
|
intel_unpin_fb_obj(obj);
|
||||||
|
err:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -7737,11 +7745,11 @@ static int intel_gen4_queue_flip(struct drm_device *dev,
|
|||||||
|
|
||||||
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err;
|
||||||
|
|
||||||
ret = BEGIN_LP_RING(4);
|
ret = BEGIN_LP_RING(4);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err_unpin;
|
||||||
|
|
||||||
/* i965+ uses the linear or tiled offsets from the
|
/* i965+ uses the linear or tiled offsets from the
|
||||||
* Display Registers (which do not change across a page-flip)
|
* Display Registers (which do not change across a page-flip)
|
||||||
@ -7760,7 +7768,11 @@ static int intel_gen4_queue_flip(struct drm_device *dev,
|
|||||||
pipesrc = I915_READ(PIPESRC(intel_crtc->pipe)) & 0x0fff0fff;
|
pipesrc = I915_READ(PIPESRC(intel_crtc->pipe)) & 0x0fff0fff;
|
||||||
OUT_RING(pf | pipesrc);
|
OUT_RING(pf | pipesrc);
|
||||||
ADVANCE_LP_RING();
|
ADVANCE_LP_RING();
|
||||||
out:
|
return 0;
|
||||||
|
|
||||||
|
err_unpin:
|
||||||
|
intel_unpin_fb_obj(obj);
|
||||||
|
err:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -7776,11 +7788,11 @@ static int intel_gen6_queue_flip(struct drm_device *dev,
|
|||||||
|
|
||||||
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
ret = intel_pin_and_fence_fb_obj(dev, obj, LP_RING(dev_priv));
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err;
|
||||||
|
|
||||||
ret = BEGIN_LP_RING(4);
|
ret = BEGIN_LP_RING(4);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err_unpin;
|
||||||
|
|
||||||
OUT_RING(MI_DISPLAY_FLIP |
|
OUT_RING(MI_DISPLAY_FLIP |
|
||||||
MI_DISPLAY_FLIP_PLANE(intel_crtc->plane));
|
MI_DISPLAY_FLIP_PLANE(intel_crtc->plane));
|
||||||
@ -7791,7 +7803,11 @@ static int intel_gen6_queue_flip(struct drm_device *dev,
|
|||||||
pipesrc = I915_READ(PIPESRC(intel_crtc->pipe)) & 0x0fff0fff;
|
pipesrc = I915_READ(PIPESRC(intel_crtc->pipe)) & 0x0fff0fff;
|
||||||
OUT_RING(pf | pipesrc);
|
OUT_RING(pf | pipesrc);
|
||||||
ADVANCE_LP_RING();
|
ADVANCE_LP_RING();
|
||||||
out:
|
return 0;
|
||||||
|
|
||||||
|
err_unpin:
|
||||||
|
intel_unpin_fb_obj(obj);
|
||||||
|
err:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -7813,18 +7829,22 @@ static int intel_gen7_queue_flip(struct drm_device *dev,
|
|||||||
|
|
||||||
ret = intel_pin_and_fence_fb_obj(dev, obj, ring);
|
ret = intel_pin_and_fence_fb_obj(dev, obj, ring);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err;
|
||||||
|
|
||||||
ret = intel_ring_begin(ring, 4);
|
ret = intel_ring_begin(ring, 4);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto err_unpin;
|
||||||
|
|
||||||
intel_ring_emit(ring, MI_DISPLAY_FLIP_I915 | (intel_crtc->plane << 19));
|
intel_ring_emit(ring, MI_DISPLAY_FLIP_I915 | (intel_crtc->plane << 19));
|
||||||
intel_ring_emit(ring, (fb->pitches[0] | obj->tiling_mode));
|
intel_ring_emit(ring, (fb->pitches[0] | obj->tiling_mode));
|
||||||
intel_ring_emit(ring, (obj->gtt_offset));
|
intel_ring_emit(ring, (obj->gtt_offset));
|
||||||
intel_ring_emit(ring, (MI_NOOP));
|
intel_ring_emit(ring, (MI_NOOP));
|
||||||
intel_ring_advance(ring);
|
intel_ring_advance(ring);
|
||||||
out:
|
return 0;
|
||||||
|
|
||||||
|
err_unpin:
|
||||||
|
intel_unpin_fb_obj(obj);
|
||||||
|
err:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user