Commit 78c6e170 authored by Chris Wilson's avatar Chris Wilson

drm/i915: Suppress spurious vblank interrupts

Hugh Dickins found that characters in xterm were going missing and oft
delayed. Being the curious type, he managed to associate this with the
new high-precision vblank patches; disabling these he found, restored
the orderliness of his characters.

The oddness begins when one realised that Hugh was not using vblanks at
all on his system (fvwm and some xterms). Instead, all he had to go on
were warning of a pipe underrun, curiously enough at around 60Hz. He
poked and found that in addition to the underrun warning, the hardware
was flagging the start of a new frame, a vblank, which in turn was
kicking off the pending vblank processing code.

There is little we can do for the underruns on Hugh's machine, a
Crestline [965GM], which must have its FIFO watermarks set to 8.
However, we do not need to process the vblank if we know that they are
disabled...
Reported-by: default avatarHugh Dickins <hughd@google.com>
Signed-off-by: default avatarChris Wilson <chris@chris-wilson.co.uk>
parent 021a8455
...@@ -1250,7 +1250,7 @@ void drm_handle_vblank_events(struct drm_device *dev, int crtc) ...@@ -1250,7 +1250,7 @@ void drm_handle_vblank_events(struct drm_device *dev, int crtc)
* Drivers should call this routine in their vblank interrupt handlers to * Drivers should call this routine in their vblank interrupt handlers to
* update the vblank counter and send any signals that may be pending. * update the vblank counter and send any signals that may be pending.
*/ */
void drm_handle_vblank(struct drm_device *dev, int crtc) bool drm_handle_vblank(struct drm_device *dev, int crtc)
{ {
u32 vblcount; u32 vblcount;
s64 diff_ns; s64 diff_ns;
...@@ -1258,7 +1258,7 @@ void drm_handle_vblank(struct drm_device *dev, int crtc) ...@@ -1258,7 +1258,7 @@ void drm_handle_vblank(struct drm_device *dev, int crtc)
unsigned long irqflags; unsigned long irqflags;
if (!dev->num_crtcs) if (!dev->num_crtcs)
return; return false;
/* Need timestamp lock to prevent concurrent execution with /* Need timestamp lock to prevent concurrent execution with
* vblank enable/disable, as this would cause inconsistent * vblank enable/disable, as this would cause inconsistent
...@@ -1269,7 +1269,7 @@ void drm_handle_vblank(struct drm_device *dev, int crtc) ...@@ -1269,7 +1269,7 @@ void drm_handle_vblank(struct drm_device *dev, int crtc)
/* Vblank irq handling disabled. Nothing to do. */ /* Vblank irq handling disabled. Nothing to do. */
if (!dev->vblank_enabled[crtc]) { if (!dev->vblank_enabled[crtc]) {
spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags); spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags);
return; return false;
} }
/* Fetch corresponding timestamp for this vblank interval from /* Fetch corresponding timestamp for this vblank interval from
...@@ -1311,5 +1311,6 @@ void drm_handle_vblank(struct drm_device *dev, int crtc) ...@@ -1311,5 +1311,6 @@ void drm_handle_vblank(struct drm_device *dev, int crtc)
drm_handle_vblank_events(dev, crtc); drm_handle_vblank_events(dev, crtc);
spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags); spin_unlock_irqrestore(&dev->vblank_time_lock, irqflags);
return true;
} }
EXPORT_SYMBOL(drm_handle_vblank); EXPORT_SYMBOL(drm_handle_vblank);
...@@ -1196,18 +1196,18 @@ irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS) ...@@ -1196,18 +1196,18 @@ irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS)
intel_finish_page_flip_plane(dev, 1); intel_finish_page_flip_plane(dev, 1);
} }
if (pipea_stats & vblank_status) { if (pipea_stats & vblank_status &&
drm_handle_vblank(dev, 0)) {
vblank++; vblank++;
drm_handle_vblank(dev, 0);
if (!dev_priv->flip_pending_is_done) { if (!dev_priv->flip_pending_is_done) {
i915_pageflip_stall_check(dev, 0); i915_pageflip_stall_check(dev, 0);
intel_finish_page_flip(dev, 0); intel_finish_page_flip(dev, 0);
} }
} }
if (pipeb_stats & vblank_status) { if (pipeb_stats & vblank_status &&
drm_handle_vblank(dev, 1)) {
vblank++; vblank++;
drm_handle_vblank(dev, 1);
if (!dev_priv->flip_pending_is_done) { if (!dev_priv->flip_pending_is_done) {
i915_pageflip_stall_check(dev, 1); i915_pageflip_stall_check(dev, 1);
intel_finish_page_flip(dev, 1); intel_finish_page_flip(dev, 1);
......
...@@ -1367,7 +1367,7 @@ extern int drm_vblank_wait(struct drm_device *dev, unsigned int *vbl_seq); ...@@ -1367,7 +1367,7 @@ extern int drm_vblank_wait(struct drm_device *dev, unsigned int *vbl_seq);
extern u32 drm_vblank_count(struct drm_device *dev, int crtc); extern u32 drm_vblank_count(struct drm_device *dev, int crtc);
extern u32 drm_vblank_count_and_time(struct drm_device *dev, int crtc, extern u32 drm_vblank_count_and_time(struct drm_device *dev, int crtc,
struct timeval *vblanktime); struct timeval *vblanktime);
extern void drm_handle_vblank(struct drm_device *dev, int crtc); extern bool drm_handle_vblank(struct drm_device *dev, int crtc);
extern int drm_vblank_get(struct drm_device *dev, int crtc); extern int drm_vblank_get(struct drm_device *dev, int crtc);
extern void drm_vblank_put(struct drm_device *dev, int crtc); extern void drm_vblank_put(struct drm_device *dev, int crtc);
extern void drm_vblank_off(struct drm_device *dev, int crtc); extern void drm_vblank_off(struct drm_device *dev, int crtc);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment