c8b5a7baf30e1730d02603a8f258e7a7978acf83
[firefly-linux-kernel-4.4.55.git] / drivers / gpu / drm / drm_irq.c
1 /**
2  * \file drm_irq.c
3  * IRQ support
4  *
5  * \author Rickard E. (Rik) Faith <faith@valinux.com>
6  * \author Gareth Hughes <gareth@valinux.com>
7  */
8
9 /*
10  * Created: Fri Mar 19 14:30:16 1999 by faith@valinux.com
11  *
12  * Copyright 1999, 2000 Precision Insight, Inc., Cedar Park, Texas.
13  * Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
14  * All Rights Reserved.
15  *
16  * Permission is hereby granted, free of charge, to any person obtaining a
17  * copy of this software and associated documentation files (the "Software"),
18  * to deal in the Software without restriction, including without limitation
19  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
20  * and/or sell copies of the Software, and to permit persons to whom the
21  * Software is furnished to do so, subject to the following conditions:
22  *
23  * The above copyright notice and this permission notice (including the next
24  * paragraph) shall be included in all copies or substantial portions of the
25  * Software.
26  *
27  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
28  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
29  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
30  * VA LINUX SYSTEMS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
31  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
32  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
33  * OTHER DEALINGS IN THE SOFTWARE.
34  */
35
36 #include "drmP.h"
37
38 #include <linux/interrupt.h>    /* For task queue support */
39
40 #include <linux/vgaarb.h>
41 /**
42  * Get interrupt from bus id.
43  *
44  * \param inode device inode.
45  * \param file_priv DRM file private.
46  * \param cmd command.
47  * \param arg user argument, pointing to a drm_irq_busid structure.
48  * \return zero on success or a negative number on failure.
49  *
50  * Finds the PCI device with the specified bus id and gets its IRQ number.
51  * This IOCTL is deprecated, and will now return EINVAL for any busid not equal
52  * to that of the device that this DRM instance attached to.
53  */
54 int drm_irq_by_busid(struct drm_device *dev, void *data,
55                      struct drm_file *file_priv)
56 {
57         struct drm_irq_busid *p = data;
58
59         if (!drm_core_check_feature(dev, DRIVER_HAVE_IRQ))
60                 return -EINVAL;
61
62         if ((p->busnum >> 8) != drm_get_pci_domain(dev) ||
63             (p->busnum & 0xff) != dev->pdev->bus->number ||
64             p->devnum != PCI_SLOT(dev->pdev->devfn) || p->funcnum != PCI_FUNC(dev->pdev->devfn))
65                 return -EINVAL;
66
67         p->irq = dev->pdev->irq;
68
69         DRM_DEBUG("%d:%d:%d => IRQ %d\n", p->busnum, p->devnum, p->funcnum,
70                   p->irq);
71
72         return 0;
73 }
74
75 static void vblank_disable_fn(unsigned long arg)
76 {
77         struct drm_device *dev = (struct drm_device *)arg;
78         unsigned long irqflags;
79         int i;
80
81         if (!dev->vblank_disable_allowed)
82                 return;
83
84         for (i = 0; i < dev->num_crtcs; i++) {
85                 spin_lock_irqsave(&dev->vbl_lock, irqflags);
86                 if (atomic_read(&dev->vblank_refcount[i]) == 0 &&
87                     dev->vblank_enabled[i]) {
88                         DRM_DEBUG("disabling vblank on crtc %d\n", i);
89                         dev->last_vblank[i] =
90                                 dev->driver->get_vblank_counter(dev, i);
91                         dev->driver->disable_vblank(dev, i);
92                         dev->vblank_enabled[i] = 0;
93                 }
94                 spin_unlock_irqrestore(&dev->vbl_lock, irqflags);
95         }
96 }
97
98 void drm_vblank_cleanup(struct drm_device *dev)
99 {
100         /* Bail if the driver didn't call drm_vblank_init() */
101         if (dev->num_crtcs == 0)
102                 return;
103
104         del_timer(&dev->vblank_disable_timer);
105
106         vblank_disable_fn((unsigned long)dev);
107
108         kfree(dev->vbl_queue);
109         kfree(dev->_vblank_count);
110         kfree(dev->vblank_refcount);
111         kfree(dev->vblank_enabled);
112         kfree(dev->last_vblank);
113         kfree(dev->last_vblank_wait);
114         kfree(dev->vblank_inmodeset);
115
116         dev->num_crtcs = 0;
117 }
118
119 int drm_vblank_init(struct drm_device *dev, int num_crtcs)
120 {
121         int i, ret = -ENOMEM;
122
123         setup_timer(&dev->vblank_disable_timer, vblank_disable_fn,
124                     (unsigned long)dev);
125         spin_lock_init(&dev->vbl_lock);
126         dev->num_crtcs = num_crtcs;
127
128         dev->vbl_queue = kmalloc(sizeof(wait_queue_head_t) * num_crtcs,
129                                  GFP_KERNEL);
130         if (!dev->vbl_queue)
131                 goto err;
132
133         dev->_vblank_count = kmalloc(sizeof(atomic_t) * num_crtcs, GFP_KERNEL);
134         if (!dev->_vblank_count)
135                 goto err;
136
137         dev->vblank_refcount = kmalloc(sizeof(atomic_t) * num_crtcs,
138                                        GFP_KERNEL);
139         if (!dev->vblank_refcount)
140                 goto err;
141
142         dev->vblank_enabled = kcalloc(num_crtcs, sizeof(int), GFP_KERNEL);
143         if (!dev->vblank_enabled)
144                 goto err;
145
146         dev->last_vblank = kcalloc(num_crtcs, sizeof(u32), GFP_KERNEL);
147         if (!dev->last_vblank)
148                 goto err;
149
150         dev->last_vblank_wait = kcalloc(num_crtcs, sizeof(u32), GFP_KERNEL);
151         if (!dev->last_vblank_wait)
152                 goto err;
153
154         dev->vblank_inmodeset = kcalloc(num_crtcs, sizeof(int), GFP_KERNEL);
155         if (!dev->vblank_inmodeset)
156                 goto err;
157
158         /* Zero per-crtc vblank stuff */
159         for (i = 0; i < num_crtcs; i++) {
160                 init_waitqueue_head(&dev->vbl_queue[i]);
161                 atomic_set(&dev->_vblank_count[i], 0);
162                 atomic_set(&dev->vblank_refcount[i], 0);
163         }
164
165         dev->vblank_disable_allowed = 0;
166
167         return 0;
168
169 err:
170         drm_vblank_cleanup(dev);
171         return ret;
172 }
173 EXPORT_SYMBOL(drm_vblank_init);
174
175 static void drm_irq_vgaarb_nokms(void *cookie, bool state)
176 {
177         struct drm_device *dev = cookie;
178
179         if (dev->driver->vgaarb_irq) {
180                 dev->driver->vgaarb_irq(dev, state);
181                 return;
182         }
183
184         if (!dev->irq_enabled)
185                 return;
186
187         if (state)
188                 dev->driver->irq_uninstall(dev);
189         else {
190                 dev->driver->irq_preinstall(dev);
191                 dev->driver->irq_postinstall(dev);
192         }
193 }
194
195 /**
196  * Install IRQ handler.
197  *
198  * \param dev DRM device.
199  *
200  * Initializes the IRQ related data. Installs the handler, calling the driver
201  * \c drm_driver_irq_preinstall() and \c drm_driver_irq_postinstall() functions
202  * before and after the installation.
203  */
204 int drm_irq_install(struct drm_device *dev)
205 {
206         int ret = 0;
207         unsigned long sh_flags = 0;
208         char *irqname;
209
210         if (!drm_core_check_feature(dev, DRIVER_HAVE_IRQ))
211                 return -EINVAL;
212
213         if (dev->pdev->irq == 0)
214                 return -EINVAL;
215
216         mutex_lock(&dev->struct_mutex);
217
218         /* Driver must have been initialized */
219         if (!dev->dev_private) {
220                 mutex_unlock(&dev->struct_mutex);
221                 return -EINVAL;
222         }
223
224         if (dev->irq_enabled) {
225                 mutex_unlock(&dev->struct_mutex);
226                 return -EBUSY;
227         }
228         dev->irq_enabled = 1;
229         mutex_unlock(&dev->struct_mutex);
230
231         DRM_DEBUG("irq=%d\n", dev->pdev->irq);
232
233         /* Before installing handler */
234         dev->driver->irq_preinstall(dev);
235
236         /* Install handler */
237         if (drm_core_check_feature(dev, DRIVER_IRQ_SHARED))
238                 sh_flags = IRQF_SHARED;
239
240         if (dev->devname)
241                 irqname = dev->devname;
242         else
243                 irqname = dev->driver->name;
244
245         ret = request_irq(drm_dev_to_irq(dev), dev->driver->irq_handler,
246                           sh_flags, irqname, dev);
247
248         if (ret < 0) {
249                 mutex_lock(&dev->struct_mutex);
250                 dev->irq_enabled = 0;
251                 mutex_unlock(&dev->struct_mutex);
252                 return ret;
253         }
254
255         if (!drm_core_check_feature(dev, DRIVER_MODESET))
256                 vga_client_register(dev->pdev, (void *)dev, drm_irq_vgaarb_nokms, NULL);
257
258         /* After installing handler */
259         ret = dev->driver->irq_postinstall(dev);
260         if (ret < 0) {
261                 mutex_lock(&dev->struct_mutex);
262                 dev->irq_enabled = 0;
263                 mutex_unlock(&dev->struct_mutex);
264         }
265
266         return ret;
267 }
268 EXPORT_SYMBOL(drm_irq_install);
269
270 /**
271  * Uninstall the IRQ handler.
272  *
273  * \param dev DRM device.
274  *
275  * Calls the driver's \c drm_driver_irq_uninstall() function, and stops the irq.
276  */
277 int drm_irq_uninstall(struct drm_device * dev)
278 {
279         unsigned long irqflags;
280         int irq_enabled, i;
281
282         if (!drm_core_check_feature(dev, DRIVER_HAVE_IRQ))
283                 return -EINVAL;
284
285         mutex_lock(&dev->struct_mutex);
286         irq_enabled = dev->irq_enabled;
287         dev->irq_enabled = 0;
288         mutex_unlock(&dev->struct_mutex);
289
290         /*
291          * Wake up any waiters so they don't hang.
292          */
293         spin_lock_irqsave(&dev->vbl_lock, irqflags);
294         for (i = 0; i < dev->num_crtcs; i++) {
295                 DRM_WAKEUP(&dev->vbl_queue[i]);
296                 dev->vblank_enabled[i] = 0;
297                 dev->last_vblank[i] = dev->driver->get_vblank_counter(dev, i);
298         }
299         spin_unlock_irqrestore(&dev->vbl_lock, irqflags);
300
301         if (!irq_enabled)
302                 return -EINVAL;
303
304         DRM_DEBUG("irq=%d\n", dev->pdev->irq);
305
306         if (!drm_core_check_feature(dev, DRIVER_MODESET))
307                 vga_client_register(dev->pdev, NULL, NULL, NULL);
308
309         dev->driver->irq_uninstall(dev);
310
311         free_irq(dev->pdev->irq, dev);
312
313         return 0;
314 }
315 EXPORT_SYMBOL(drm_irq_uninstall);
316
317 /**
318  * IRQ control ioctl.
319  *
320  * \param inode device inode.
321  * \param file_priv DRM file private.
322  * \param cmd command.
323  * \param arg user argument, pointing to a drm_control structure.
324  * \return zero on success or a negative number on failure.
325  *
326  * Calls irq_install() or irq_uninstall() according to \p arg.
327  */
328 int drm_control(struct drm_device *dev, void *data,
329                 struct drm_file *file_priv)
330 {
331         struct drm_control *ctl = data;
332
333         /* if we haven't irq we fallback for compatibility reasons - this used to be a separate function in drm_dma.h */
334
335
336         switch (ctl->func) {
337         case DRM_INST_HANDLER:
338                 if (!drm_core_check_feature(dev, DRIVER_HAVE_IRQ))
339                         return 0;
340                 if (drm_core_check_feature(dev, DRIVER_MODESET))
341                         return 0;
342                 if (dev->if_version < DRM_IF_VERSION(1, 2) &&
343                     ctl->irq != dev->pdev->irq)
344                         return -EINVAL;
345                 return drm_irq_install(dev);
346         case DRM_UNINST_HANDLER:
347                 if (!drm_core_check_feature(dev, DRIVER_HAVE_IRQ))
348                         return 0;
349                 if (drm_core_check_feature(dev, DRIVER_MODESET))
350                         return 0;
351                 return drm_irq_uninstall(dev);
352         default:
353                 return -EINVAL;
354         }
355 }
356
357 /**
358  * drm_vblank_count - retrieve "cooked" vblank counter value
359  * @dev: DRM device
360  * @crtc: which counter to retrieve
361  *
362  * Fetches the "cooked" vblank count value that represents the number of
363  * vblank events since the system was booted, including lost events due to
364  * modesetting activity.
365  */
366 u32 drm_vblank_count(struct drm_device *dev, int crtc)
367 {
368         return atomic_read(&dev->_vblank_count[crtc]);
369 }
370 EXPORT_SYMBOL(drm_vblank_count);
371
372 /**
373  * drm_update_vblank_count - update the master vblank counter
374  * @dev: DRM device
375  * @crtc: counter to update
376  *
377  * Call back into the driver to update the appropriate vblank counter
378  * (specified by @crtc).  Deal with wraparound, if it occurred, and
379  * update the last read value so we can deal with wraparound on the next
380  * call if necessary.
381  *
382  * Only necessary when going from off->on, to account for frames we
383  * didn't get an interrupt for.
384  *
385  * Note: caller must hold dev->vbl_lock since this reads & writes
386  * device vblank fields.
387  */
388 static void drm_update_vblank_count(struct drm_device *dev, int crtc)
389 {
390         u32 cur_vblank, diff;
391
392         /*
393          * Interrupts were disabled prior to this call, so deal with counter
394          * wrap if needed.
395          * NOTE!  It's possible we lost a full dev->max_vblank_count events
396          * here if the register is small or we had vblank interrupts off for
397          * a long time.
398          */
399         cur_vblank = dev->driver->get_vblank_counter(dev, crtc);
400         diff = cur_vblank - dev->last_vblank[crtc];
401         if (cur_vblank < dev->last_vblank[crtc]) {
402                 diff += dev->max_vblank_count;
403
404                 DRM_DEBUG("last_vblank[%d]=0x%x, cur_vblank=0x%x => diff=0x%x\n",
405                           crtc, dev->last_vblank[crtc], cur_vblank, diff);
406         }
407
408         DRM_DEBUG("enabling vblank interrupts on crtc %d, missed %d\n",
409                   crtc, diff);
410
411         atomic_add(diff, &dev->_vblank_count[crtc]);
412 }
413
414 /**
415  * drm_vblank_get - get a reference count on vblank events
416  * @dev: DRM device
417  * @crtc: which CRTC to own
418  *
419  * Acquire a reference count on vblank events to avoid having them disabled
420  * while in use.
421  *
422  * RETURNS
423  * Zero on success, nonzero on failure.
424  */
425 int drm_vblank_get(struct drm_device *dev, int crtc)
426 {
427         unsigned long irqflags;
428         int ret = 0;
429
430         spin_lock_irqsave(&dev->vbl_lock, irqflags);
431         /* Going from 0->1 means we have to enable interrupts again */
432         if (atomic_add_return(1, &dev->vblank_refcount[crtc]) == 1) {
433                 if (!dev->vblank_enabled[crtc]) {
434                         ret = dev->driver->enable_vblank(dev, crtc);
435                         DRM_DEBUG("enabling vblank on crtc %d, ret: %d\n", crtc, ret);
436                         if (ret)
437                                 atomic_dec(&dev->vblank_refcount[crtc]);
438                         else {
439                                 dev->vblank_enabled[crtc] = 1;
440                                 drm_update_vblank_count(dev, crtc);
441                         }
442                 }
443         } else {
444                 if (!dev->vblank_enabled[crtc]) {
445                         atomic_dec(&dev->vblank_refcount[crtc]);
446                         ret = -EINVAL;
447                 }
448         }
449         spin_unlock_irqrestore(&dev->vbl_lock, irqflags);
450
451         return ret;
452 }
453 EXPORT_SYMBOL(drm_vblank_get);
454
455 /**
456  * drm_vblank_put - give up ownership of vblank events
457  * @dev: DRM device
458  * @crtc: which counter to give up
459  *
460  * Release ownership of a given vblank counter, turning off interrupts
461  * if possible.
462  */
463 void drm_vblank_put(struct drm_device *dev, int crtc)
464 {
465         BUG_ON (atomic_read (&dev->vblank_refcount[crtc]) == 0);
466
467         /* Last user schedules interrupt disable */
468         if (atomic_dec_and_test(&dev->vblank_refcount[crtc]))
469                 mod_timer(&dev->vblank_disable_timer, jiffies + 5*DRM_HZ);
470 }
471 EXPORT_SYMBOL(drm_vblank_put);
472
473 void drm_vblank_off(struct drm_device *dev, int crtc)
474 {
475         unsigned long irqflags;
476
477         spin_lock_irqsave(&dev->vbl_lock, irqflags);
478         DRM_WAKEUP(&dev->vbl_queue[crtc]);
479         dev->vblank_enabled[crtc] = 0;
480         dev->last_vblank[crtc] = dev->driver->get_vblank_counter(dev, crtc);
481         spin_unlock_irqrestore(&dev->vbl_lock, irqflags);
482 }
483 EXPORT_SYMBOL(drm_vblank_off);
484
485 /**
486  * drm_vblank_pre_modeset - account for vblanks across mode sets
487  * @dev: DRM device
488  * @crtc: CRTC in question
489  * @post: post or pre mode set?
490  *
491  * Account for vblank events across mode setting events, which will likely
492  * reset the hardware frame counter.
493  */
494 void drm_vblank_pre_modeset(struct drm_device *dev, int crtc)
495 {
496         /*
497          * To avoid all the problems that might happen if interrupts
498          * were enabled/disabled around or between these calls, we just
499          * have the kernel take a reference on the CRTC (just once though
500          * to avoid corrupting the count if multiple, mismatch calls occur),
501          * so that interrupts remain enabled in the interim.
502          */
503         if (!dev->vblank_inmodeset[crtc]) {
504                 dev->vblank_inmodeset[crtc] = 0x1;
505                 if (drm_vblank_get(dev, crtc) == 0)
506                         dev->vblank_inmodeset[crtc] |= 0x2;
507         }
508 }
509 EXPORT_SYMBOL(drm_vblank_pre_modeset);
510
511 void drm_vblank_post_modeset(struct drm_device *dev, int crtc)
512 {
513         unsigned long irqflags;
514
515         if (dev->vblank_inmodeset[crtc]) {
516                 spin_lock_irqsave(&dev->vbl_lock, irqflags);
517                 dev->vblank_disable_allowed = 1;
518                 spin_unlock_irqrestore(&dev->vbl_lock, irqflags);
519
520                 if (dev->vblank_inmodeset[crtc] & 0x2)
521                         drm_vblank_put(dev, crtc);
522
523                 dev->vblank_inmodeset[crtc] = 0;
524         }
525 }
526 EXPORT_SYMBOL(drm_vblank_post_modeset);
527
528 /**
529  * drm_modeset_ctl - handle vblank event counter changes across mode switch
530  * @DRM_IOCTL_ARGS: standard ioctl arguments
531  *
532  * Applications should call the %_DRM_PRE_MODESET and %_DRM_POST_MODESET
533  * ioctls around modesetting so that any lost vblank events are accounted for.
534  *
535  * Generally the counter will reset across mode sets.  If interrupts are
536  * enabled around this call, we don't have to do anything since the counter
537  * will have already been incremented.
538  */
539 int drm_modeset_ctl(struct drm_device *dev, void *data,
540                     struct drm_file *file_priv)
541 {
542         struct drm_modeset_ctl *modeset = data;
543         int ret = 0;
544         unsigned int crtc;
545
546         /* If drm_vblank_init() hasn't been called yet, just no-op */
547         if (!dev->num_crtcs)
548                 goto out;
549
550         crtc = modeset->crtc;
551         if (crtc >= dev->num_crtcs) {
552                 ret = -EINVAL;
553                 goto out;
554         }
555
556         switch (modeset->cmd) {
557         case _DRM_PRE_MODESET:
558                 drm_vblank_pre_modeset(dev, crtc);
559                 break;
560         case _DRM_POST_MODESET:
561                 drm_vblank_post_modeset(dev, crtc);
562                 break;
563         default:
564                 ret = -EINVAL;
565                 break;
566         }
567
568 out:
569         return ret;
570 }
571
572 /**
573  * Wait for VBLANK.
574  *
575  * \param inode device inode.
576  * \param file_priv DRM file private.
577  * \param cmd command.
578  * \param data user argument, pointing to a drm_wait_vblank structure.
579  * \return zero on success or a negative number on failure.
580  *
581  * This function enables the vblank interrupt on the pipe requested, then
582  * sleeps waiting for the requested sequence number to occur, and drops
583  * the vblank interrupt refcount afterwards. (vblank irq disable follows that
584  * after a timeout with no further vblank waits scheduled).
585  */
586 int drm_wait_vblank(struct drm_device *dev, void *data,
587                     struct drm_file *file_priv)
588 {
589         union drm_wait_vblank *vblwait = data;
590         int ret = 0;
591         unsigned int flags, seq, crtc;
592
593         if ((!dev->pdev->irq) || (!dev->irq_enabled))
594                 return -EINVAL;
595
596         if (vblwait->request.type & _DRM_VBLANK_SIGNAL)
597                 return -EINVAL;
598
599         if (vblwait->request.type &
600             ~(_DRM_VBLANK_TYPES_MASK | _DRM_VBLANK_FLAGS_MASK)) {
601                 DRM_ERROR("Unsupported type value 0x%x, supported mask 0x%x\n",
602                           vblwait->request.type,
603                           (_DRM_VBLANK_TYPES_MASK | _DRM_VBLANK_FLAGS_MASK));
604                 return -EINVAL;
605         }
606
607         flags = vblwait->request.type & _DRM_VBLANK_FLAGS_MASK;
608         crtc = flags & _DRM_VBLANK_SECONDARY ? 1 : 0;
609
610         if (crtc >= dev->num_crtcs)
611                 return -EINVAL;
612
613         ret = drm_vblank_get(dev, crtc);
614         if (ret) {
615                 DRM_DEBUG("failed to acquire vblank counter, %d\n", ret);
616                 return ret;
617         }
618         seq = drm_vblank_count(dev, crtc);
619
620         switch (vblwait->request.type & _DRM_VBLANK_TYPES_MASK) {
621         case _DRM_VBLANK_RELATIVE:
622                 vblwait->request.sequence += seq;
623                 vblwait->request.type &= ~_DRM_VBLANK_RELATIVE;
624         case _DRM_VBLANK_ABSOLUTE:
625                 break;
626         default:
627                 ret = -EINVAL;
628                 goto done;
629         }
630
631         if ((flags & _DRM_VBLANK_NEXTONMISS) &&
632             (seq - vblwait->request.sequence) <= (1<<23)) {
633                 vblwait->request.sequence = seq + 1;
634         }
635
636         DRM_DEBUG("waiting on vblank count %d, crtc %d\n",
637                   vblwait->request.sequence, crtc);
638         dev->last_vblank_wait[crtc] = vblwait->request.sequence;
639         DRM_WAIT_ON(ret, dev->vbl_queue[crtc], 3 * DRM_HZ,
640                     (((drm_vblank_count(dev, crtc) -
641                        vblwait->request.sequence) <= (1 << 23)) ||
642                      !dev->irq_enabled));
643
644         if (ret != -EINTR) {
645                 struct timeval now;
646
647                 do_gettimeofday(&now);
648
649                 vblwait->reply.tval_sec = now.tv_sec;
650                 vblwait->reply.tval_usec = now.tv_usec;
651                 vblwait->reply.sequence = drm_vblank_count(dev, crtc);
652                 DRM_DEBUG("returning %d to client\n",
653                           vblwait->reply.sequence);
654         } else {
655                 DRM_DEBUG("vblank wait interrupted by signal\n");
656         }
657
658 done:
659         drm_vblank_put(dev, crtc);
660         return ret;
661 }
662
663 /**
664  * drm_handle_vblank - handle a vblank event
665  * @dev: DRM device
666  * @crtc: where this event occurred
667  *
668  * Drivers should call this routine in their vblank interrupt handlers to
669  * update the vblank counter and send any signals that may be pending.
670  */
671 void drm_handle_vblank(struct drm_device *dev, int crtc)
672 {
673         atomic_inc(&dev->_vblank_count[crtc]);
674         DRM_WAKEUP(&dev->vbl_queue[crtc]);
675 }
676 EXPORT_SYMBOL(drm_handle_vblank);