From: Torgeir Veimo <torgeir@pobox.com>
To: linux-fbdev-devel@lists.sourceforge.net
Subject: [patch] radeonfb: FB_WAITFORVSYNC implementation
Date: Sat, 12 Mar 2005 14:06:45 +0000 [thread overview]
Message-ID: <1110636406.5997.86.camel@atlantis.netenviron.com> (raw)
[-- Attachment #1: Type: text/plain, Size: 250 bytes --]
This is an implementation of the FB_WAITFORVSYNC ioctl for the radeonfb.
A small test application is attached at the end. This patch is against
vanilla 2.6.11.
Signed-off-by: Torgeir Veimo <torgeir@pobox.com>
--
Torgeir Veimo <torgeir@pobox.com>
[-- Attachment #2: vsynctest.c --]
[-- Type: text/x-csrc, Size: 1342 bytes --]
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <strings.h>
#include <fcntl.h>
#include <sys/time.h>
#include <sys/ioctl.h>
#define __u32 u_int32_t
#ifndef FBIO_WAITFORVSYNC
#define FBIO_WAITFORVSYNC _IOW('F', 0x20, __u32)
#endif
/* check functionality of VSYNC ioctl & determine approximate vertical refresh frequency */
/* compile with gcc -o vsynctest vsynctest.c */
/* March 12th, 2005, torgeir at nothome dot com. */
int main(int argc, char** argv) {
int retval = 0;
int fd = 0;
int zero = 0;
int i;
struct timeval start;
struct timeval end;
long total; float secs; float rate;
printf("testing refresh rate using FB_WAITFORVSYNC ioctl, standby while i loop 500 times..\n");
if ((fd = open( "/dev/fb0", O_RDWR )) < 0) {
printf("unable to access /dev/fb0, exiting..\n");
exit(1);
}
gettimeofday(&start, NULL);
for (i = 0; i < 500; i++) {
retval = ioctl(fd, FBIO_WAITFORVSYNC, &zero);
if (retval != 0) {
printf("ioctl failed, aborting...\n");
exit(1);
}
}
gettimeofday(&end, NULL);
total = (end.tv_sec - start.tv_sec) * 1000000 + (end.tv_usec - start.tv_usec);
secs = total / 1000000.0;
rate = 500000000.0 / total;
printf("time elapsed: %2.2f secs, refresh rate is %2.2f Hz\n", secs, rate);
close( fd );
}
[-- Attachment #3: irq.patch --]
[-- Type: text/x-patch, Size: 5518 bytes --]
--- linux-2.6.11-orig/include/video/radeon.h 2005-03-02 07:38:33.000000000 +0000
+++ linux-2.6.11/include/video/radeon.h 2005-03-12 13:50:35.000000000 +0000
@@ -541,6 +541,14 @@
#define CRTC_EN (1 << 25)
#define CRTC_DISP_REQ_EN_B (1 << 26)
+/* GEN_INT_CNTL bit constants */
+#define CRTC_VBLANK_MASK (1 << 0)
+#define CRTC_VBLANK_INT_EN (1 << 0)
+
+/* GEN_INT_STATUS bit constants */
+#define CRTC_VBLANK_STAT (1 << 0)
+#define CRTC_VBLANK_STAT_ACK (1 << 0)
+
/* CRTC_STATUS bit constants */
#define CRTC_VBLANK 0x00000001
--- linux-2.6.11-orig/drivers/video/aty/radeonfb.h 2005-03-02 07:38:37.000000000 +0000
+++ linux-2.6.11/drivers/video/aty/radeonfb.h 2005-03-03 17:09:23.000000000 +0000
@@ -269,6 +269,11 @@
radeon_pm_off = 0x00000002, /* Can resume from D3 cold */
};
+struct radeon_interrupt {
+ wait_queue_head_t wait;
+ unsigned int count;
+};
+
struct radeonfb_info {
struct fb_info *info;
@@ -344,6 +349,11 @@
struct timer_list lvds_timer;
u32 pending_lvds_gen_cntl;
+ int open;
+ struct radeon_interrupt vblank;
+ unsigned long irq_flags;
+ spinlock_t int_lock;
+
#ifdef CONFIG_FB_RADEON_I2C
struct radeon_i2c_chan i2c[4];
#endif
--- linux-2.6.11-orig/drivers/video/aty/radeon_base.c 2005-03-02 07:37:54.000000000 +0000
+++ linux-2.6.11/drivers/video/aty/radeon_base.c 2005-03-12 13:53:32.000000000 +0000
@@ -68,6 +68,9 @@
#include <linux/ioport.h>
#include <linux/init.h>
#include <linux/pci.h>
+#include <linux/interrupt.h>
+#include <linux/spinlock.h>
+#include <linux/wait.h>
#include <linux/vmalloc.h>
#include <linux/device.h>
#include <linux/i2c.h>
@@ -863,6 +866,97 @@
}
+static irqreturn_t radeon_irq(int irq, void *dev_id, struct pt_regs *fp)
+{
+ struct radeonfb_info *rinfo = dev_id;
+ u32 int_cntl = 0;
+ u32 stat = 0;
+
+ int_cntl = INREG(GEN_INT_CNTL);
+ stat = INREG(GEN_INT_STATUS) & (CRTC_VBLANK_MASK);
+ if (!stat)
+ return IRQ_NONE;
+ if (stat) {
+ /* clear interrupt */
+ OUTREG(GEN_INT_STATUS, stat);
+
+ (rinfo->vblank.count)++;
+ wake_up_interruptible(&rinfo->vblank.wait);
+ }
+ return IRQ_HANDLED;
+}
+
+static int radeon_enable_irq(struct radeonfb_info *rinfo, int reenable)
+{
+ u32 int_cntl;
+
+ if (!test_and_set_bit(0, &rinfo->irq_flags)) {
+ if (request_irq(rinfo->pdev->irq, radeon_irq, SA_SHIRQ,
+ "radeonfb", rinfo)) {
+ printk("radeonfb: request_irq failed..\n");
+ clear_bit(0, &rinfo->irq_flags);
+ return -EINVAL;
+ }
+ int_cntl = INREG(GEN_INT_CNTL) & CRTC_VBLANK_MASK;
+ /* clear interrupt */
+ //OUTREG(GEN_INT_CNTL, int_cntl | CRTC_VBLANK_STAT_ACK);
+ /* enable interrupt */
+ OUTREG(GEN_INT_CNTL, int_cntl | CRTC_VBLANK_INT_EN);
+ printk("radeonfb: enabled IRQ\n");
+ } else if (reenable) {
+ int_cntl = INREG(GEN_INT_CNTL) & CRTC_VBLANK_MASK;
+ if (!(int_cntl & CRTC_VBLANK_INT_EN)) {
+ printk("radeonfb: someone disabled IRQ [%08x]\n", int_cntl);
+ /* re-enable interrupt */
+ OUTREG(GEN_INT_CNTL, int_cntl | CRTC_VBLANK_INT_EN);
+ }
+ }
+
+ return 0;
+}
+
+static void radeon_disable_irq(struct radeonfb_info *rinfo)
+{
+ u32 int_cntl;
+
+ if (test_and_clear_bit(0, &rinfo->irq_flags)) {
+ int_cntl = INREG(GEN_INT_CNTL) & CRTC_VBLANK_MASK;
+ /* disable interrupt */
+ OUTREG(GEN_INT_CNTL, int_cntl & ~CRTC_VBLANK_INT_EN);
+ free_irq(rinfo->pdev->irq, rinfo);
+ printk("radeonfb: disabled IRQ\n");
+ }
+}
+
+static int radeon_waitforvblank(struct radeonfb_info *rinfo)
+{
+ struct radeon_interrupt *vbl;
+ unsigned int count;
+ int ret;
+
+ vbl = &rinfo->vblank;
+
+ ret = radeon_enable_irq(rinfo, 0);
+ if (ret) {
+ return ret;
+ }
+
+ count = vbl->count;
+ ret = wait_event_interruptible_timeout(vbl->wait, count != vbl->count, HZ/10);
+ if (ret < 0) {
+ return ret;
+ }
+ if (ret == 0) {
+ radeon_enable_irq(rinfo, 1);
+ return -ETIMEDOUT;
+ }
+ return 0;
+}
+
+#ifndef FBIO_WAITFORVSYNC
+#define FBIO_WAITFORVSYNC _IOW('F', 0x20, __u32)
+#endif
+
static int radeonfb_ioctl (struct inode *inode, struct file *file, unsigned int cmd,
unsigned long arg, struct fb_info *info)
{
@@ -927,6 +1021,8 @@
value |= 0x02;
return put_user(value, (__u32 __user *)arg);
+ case FBIO_WAITFORVSYNC:
+ return radeon_waitforvblank(rinfo);
default:
return -EINVAL;
}
@@ -1464,6 +1560,30 @@
RTRACE("ppll_div_3 = 0x%x\n", regs->ppll_div_3);
}
+static int radeonfb_open(struct fb_info *info, int user)
+{
+ struct radeonfb_info *rinfo = (struct radeonfb_info *) info->par;
+ if (user) {
+ rinfo->open++;
+ }
+ return (0);
+}
+
+static int radeonfb_release(struct fb_info *info, int user)
+{
+ struct radeonfb_info *rinfo = (struct radeonfb_info *) info->par;
+ if (user) {
+ rinfo->open--;
+ mdelay(1);
+ //wait_for_idle(rinfo);
+ if (!rinfo->open) {
+ radeon_disable_irq(rinfo);
+ }
+ }
+ return (0);
+}
+
+
static int radeonfb_set_par(struct fb_info *info)
{
struct radeonfb_info *rinfo = info->par;
@@ -1788,6 +1908,8 @@
static struct fb_ops radeonfb_ops = {
.owner = THIS_MODULE,
+ .fb_open = radeonfb_open,
+ .fb_release = radeonfb_release,
.fb_check_var = radeonfb_check_var,
.fb_set_par = radeonfb_set_par,
.fb_setcolreg = radeonfb_setcolreg,
@@ -2338,6 +2460,10 @@
radeon_create_i2c_busses(rinfo);
#endif
+ rinfo->irq_flags = 0;
+ init_waitqueue_head(&rinfo->vblank.wait);
+ spin_lock_init(&(rinfo->int_lock));
+
/* set all the vital stuff */
radeon_set_fbinfo (rinfo);
next reply other threads:[~2005-03-12 14:06 UTC|newest]
Thread overview: 27+ messages / expand[flat|nested] mbox.gz Atom feed top
2005-03-12 14:06 Torgeir Veimo [this message]
2005-03-12 15:13 ` [patch] radeonfb: FB_WAITFORVSYNC implementation Ville Syrjälä
2005-03-12 16:59 ` Torgeir Veimo
2005-03-12 17:10 ` Michel Dänzer
2005-03-12 17:12 ` Torgeir Veimo
2005-03-12 19:20 ` Michel Dänzer
2005-03-12 19:35 ` Jon Smirl
2005-03-12 23:33 ` Benjamin Herrenschmidt
2005-03-12 17:33 ` Ville Syrjälä
2005-03-16 1:28 ` Torgeir Veimo
2005-03-16 1:59 ` Ville Syrjälä
2005-03-16 6:19 ` Michel Dänzer
2005-03-12 15:33 ` Jon Smirl
2005-03-12 15:51 ` Ville Syrjälä
2005-03-12 16:00 ` Jon Smirl
2005-03-12 16:10 ` Ville Syrjälä
2005-03-12 16:21 ` Jon Smirl
2005-03-12 23:23 ` Benjamin Herrenschmidt
2005-03-12 17:03 ` Torgeir Veimo
2005-03-12 17:21 ` Torgeir Veimo
2005-03-12 17:56 ` Jon Smirl
2005-03-12 23:21 ` Benjamin Herrenschmidt
2005-03-12 23:14 ` Benjamin Herrenschmidt
2005-03-13 3:15 ` Torgeir Veimo
2005-03-12 23:30 ` Benjamin Herrenschmidt
2005-03-16 1:28 ` Torgeir Veimo
[not found] <20050313012923.60373.qmail@web14926.mail.yahoo.com>
2005-03-13 1:37 ` Benjamin Herrenschmidt
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=1110636406.5997.86.camel@atlantis.netenviron.com \
--to=torgeir@pobox.com \
--cc=linux-fbdev-devel@lists.sourceforge.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).