aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/char/drm/i915_drv.h4
-rw-r--r--drivers/char/drm/i915_irq.c36
2 files changed, 21 insertions, 19 deletions
diff --git a/drivers/char/drm/i915_drv.h b/drivers/char/drm/i915_drv.h
index 334b0ce8181d..93cdcfe6aa84 100644
--- a/drivers/char/drm/i915_drv.h
+++ b/drivers/char/drm/i915_drv.h
@@ -46,9 +46,11 @@
46 * 1.3: Add vblank support 46 * 1.3: Add vblank support
47 * 1.4: Fix cmdbuffer path, add heap destroy 47 * 1.4: Fix cmdbuffer path, add heap destroy
48 * 1.5: Add vblank pipe configuration 48 * 1.5: Add vblank pipe configuration
49 * 1.6: - New ioctl for scheduling buffer swaps on vertical blank
50 * - Support vertical blank on secondary display pipe
49 */ 51 */
50#define DRIVER_MAJOR 1 52#define DRIVER_MAJOR 1
51#define DRIVER_MINOR 5 53#define DRIVER_MINOR 6
52#define DRIVER_PATCHLEVEL 0 54#define DRIVER_PATCHLEVEL 0
53 55
54typedef struct _drm_i915_ring_buffer { 56typedef struct _drm_i915_ring_buffer {
diff --git a/drivers/char/drm/i915_irq.c b/drivers/char/drm/i915_irq.c
index 1a5edec7e190..e5463b111fc0 100644
--- a/drivers/char/drm/i915_irq.c
+++ b/drivers/char/drm/i915_irq.c
@@ -153,20 +153,26 @@ irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS)
153 DRM_WAKEUP(&dev_priv->irq_queue); 153 DRM_WAKEUP(&dev_priv->irq_queue);
154 154
155 if (temp & (VSYNC_PIPEA_FLAG | VSYNC_PIPEB_FLAG)) { 155 if (temp & (VSYNC_PIPEA_FLAG | VSYNC_PIPEB_FLAG)) {
156 if ((dev_priv->vblank_pipe & 156 int vblank_pipe = dev_priv->vblank_pipe;
157
158 if ((vblank_pipe &
157 (DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B)) 159 (DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B))
158 == (DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B)) { 160 == (DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B)) {
159 if (temp & VSYNC_PIPEA_FLAG) 161 if (temp & VSYNC_PIPEA_FLAG)
160 atomic_inc(&dev->vbl_received); 162 atomic_inc(&dev->vbl_received);
161 if (temp & VSYNC_PIPEB_FLAG) 163 if (temp & VSYNC_PIPEB_FLAG)
162 atomic_inc(&dev->vbl_received2); 164 atomic_inc(&dev->vbl_received2);
163 } else 165 } else if (((temp & VSYNC_PIPEA_FLAG) &&
166 (vblank_pipe & DRM_I915_VBLANK_PIPE_A)) ||
167 ((temp & VSYNC_PIPEB_FLAG) &&
168 (vblank_pipe & DRM_I915_VBLANK_PIPE_B)))
164 atomic_inc(&dev->vbl_received); 169 atomic_inc(&dev->vbl_received);
165 170
166 DRM_WAKEUP(&dev->vbl_queue); 171 DRM_WAKEUP(&dev->vbl_queue);
167 drm_vbl_send_signals(dev); 172 drm_vbl_send_signals(dev);
168 173
169 drm_locked_tasklet(dev, i915_vblank_tasklet); 174 if (dev_priv->swaps_pending > 0)
175 drm_locked_tasklet(dev, i915_vblank_tasklet);
170 } 176 }
171 177
172 return IRQ_HANDLED; 178 return IRQ_HANDLED;
@@ -397,7 +403,7 @@ int i915_vblank_swap(DRM_IOCTL_ARGS)
397 sizeof(swap)); 403 sizeof(swap));
398 404
399 if (swap.seqtype & ~(_DRM_VBLANK_RELATIVE | _DRM_VBLANK_ABSOLUTE | 405 if (swap.seqtype & ~(_DRM_VBLANK_RELATIVE | _DRM_VBLANK_ABSOLUTE |
400 _DRM_VBLANK_SECONDARY)) { 406 _DRM_VBLANK_SECONDARY | _DRM_VBLANK_NEXTONMISS)) {
401 DRM_ERROR("Invalid sequence type 0x%x\n", swap.seqtype); 407 DRM_ERROR("Invalid sequence type 0x%x\n", swap.seqtype);
402 return DRM_ERR(EINVAL); 408 return DRM_ERR(EINVAL);
403 } 409 }
@@ -406,11 +412,6 @@ int i915_vblank_swap(DRM_IOCTL_ARGS)
406 412
407 seqtype = swap.seqtype & (_DRM_VBLANK_RELATIVE | _DRM_VBLANK_ABSOLUTE); 413 seqtype = swap.seqtype & (_DRM_VBLANK_RELATIVE | _DRM_VBLANK_ABSOLUTE);
408 414
409 if (seqtype == _DRM_VBLANK_RELATIVE && swap.sequence == 0) {
410 DRM_DEBUG("Not scheduling swap for current sequence\n");
411 return DRM_ERR(EINVAL);
412 }
413
414 if (!(dev_priv->vblank_pipe & (1 << pipe))) { 415 if (!(dev_priv->vblank_pipe & (1 << pipe))) {
415 DRM_ERROR("Invalid pipe %d\n", pipe); 416 DRM_ERROR("Invalid pipe %d\n", pipe);
416 return DRM_ERR(EINVAL); 417 return DRM_ERR(EINVAL);
@@ -428,21 +429,20 @@ int i915_vblank_swap(DRM_IOCTL_ARGS)
428 429
429 curseq = atomic_read(pipe ? &dev->vbl_received2 : &dev->vbl_received); 430 curseq = atomic_read(pipe ? &dev->vbl_received2 : &dev->vbl_received);
430 431
431 spin_lock_irqsave(&dev_priv->swaps_lock, irqflags); 432 if (seqtype == _DRM_VBLANK_RELATIVE)
432
433 switch (seqtype) {
434 case _DRM_VBLANK_RELATIVE:
435 swap.sequence += curseq; 433 swap.sequence += curseq;
436 break; 434
437 case _DRM_VBLANK_ABSOLUTE: 435 if ((curseq - swap.sequence) <= (1<<23)) {
438 if ((curseq - swap.sequence) <= (1<<23)) { 436 if (swap.seqtype & _DRM_VBLANK_NEXTONMISS) {
439 spin_unlock_irqrestore(&dev_priv->swaps_lock, irqflags); 437 swap.sequence = curseq + 1;
438 } else {
440 DRM_DEBUG("Missed target sequence\n"); 439 DRM_DEBUG("Missed target sequence\n");
441 return DRM_ERR(EINVAL); 440 return DRM_ERR(EINVAL);
442 } 441 }
443 break;
444 } 442 }
445 443
444 spin_lock_irqsave(&dev_priv->swaps_lock, irqflags);
445
446 list_for_each(list, &dev_priv->vbl_swaps.head) { 446 list_for_each(list, &dev_priv->vbl_swaps.head) {
447 vbl_swap = list_entry(list, drm_i915_vbl_swap_t, head); 447 vbl_swap = list_entry(list, drm_i915_vbl_swap_t, head);
448 448