aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/char/hpet.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/char/hpet.c')
-rw-r--r--drivers/char/hpet.c164
1 files changed, 113 insertions, 51 deletions
diff --git a/drivers/char/hpet.c b/drivers/char/hpet.c
index a0a1829d3198..34d6a1cab8de 100644
--- a/drivers/char/hpet.c
+++ b/drivers/char/hpet.c
@@ -14,7 +14,6 @@
14#include <linux/interrupt.h> 14#include <linux/interrupt.h>
15#include <linux/module.h> 15#include <linux/module.h>
16#include <linux/kernel.h> 16#include <linux/kernel.h>
17#include <linux/smp_lock.h>
18#include <linux/types.h> 17#include <linux/types.h>
19#include <linux/miscdevice.h> 18#include <linux/miscdevice.h>
20#include <linux/major.h> 19#include <linux/major.h>
@@ -30,13 +29,14 @@
30#include <linux/bcd.h> 29#include <linux/bcd.h>
31#include <linux/seq_file.h> 30#include <linux/seq_file.h>
32#include <linux/bitops.h> 31#include <linux/bitops.h>
32#include <linux/compat.h>
33#include <linux/clocksource.h> 33#include <linux/clocksource.h>
34#include <linux/uaccess.h>
34#include <linux/slab.h> 35#include <linux/slab.h>
36#include <linux/io.h>
35 37
36#include <asm/current.h> 38#include <asm/current.h>
37#include <asm/uaccess.h>
38#include <asm/system.h> 39#include <asm/system.h>
39#include <asm/io.h>
40#include <asm/irq.h> 40#include <asm/irq.h>
41#include <asm/div64.h> 41#include <asm/div64.h>
42 42
@@ -67,6 +67,7 @@
67#define read_counter(MC) readl(MC) 67#define read_counter(MC) readl(MC)
68#endif 68#endif
69 69
70static DEFINE_MUTEX(hpet_mutex); /* replaces BKL */
70static u32 hpet_nhpet, hpet_max_freq = HPET_USER_FREQ; 71static u32 hpet_nhpet, hpet_max_freq = HPET_USER_FREQ;
71 72
72/* This clocksource driver currently only works on ia64 */ 73/* This clocksource driver currently only works on ia64 */
@@ -79,13 +80,11 @@ static cycle_t read_hpet(struct clocksource *cs)
79} 80}
80 81
81static struct clocksource clocksource_hpet = { 82static struct clocksource clocksource_hpet = {
82 .name = "hpet", 83 .name = "hpet",
83 .rating = 250, 84 .rating = 250,
84 .read = read_hpet, 85 .read = read_hpet,
85 .mask = CLOCKSOURCE_MASK(64), 86 .mask = CLOCKSOURCE_MASK(64),
86 .mult = 0, /* to be calculated */ 87 .flags = CLOCK_SOURCE_IS_CONTINUOUS,
87 .shift = 10,
88 .flags = CLOCK_SOURCE_IS_CONTINUOUS,
89}; 88};
90static struct clocksource *hpet_clocksource; 89static struct clocksource *hpet_clocksource;
91#endif 90#endif
@@ -164,11 +163,32 @@ static irqreturn_t hpet_interrupt(int irq, void *data)
164 * This has the effect of treating non-periodic like periodic. 163 * This has the effect of treating non-periodic like periodic.
165 */ 164 */
166 if ((devp->hd_flags & (HPET_IE | HPET_PERIODIC)) == HPET_IE) { 165 if ((devp->hd_flags & (HPET_IE | HPET_PERIODIC)) == HPET_IE) {
167 unsigned long m, t; 166 unsigned long m, t, mc, base, k;
167 struct hpet __iomem *hpet = devp->hd_hpet;
168 struct hpets *hpetp = devp->hd_hpets;
168 169
169 t = devp->hd_ireqfreq; 170 t = devp->hd_ireqfreq;
170 m = read_counter(&devp->hd_timer->hpet_compare); 171 m = read_counter(&devp->hd_timer->hpet_compare);
171 write_counter(t + m, &devp->hd_timer->hpet_compare); 172 mc = read_counter(&hpet->hpet_mc);
173 /* The time for the next interrupt would logically be t + m,
174 * however, if we are very unlucky and the interrupt is delayed
175 * for longer than t then we will completely miss the next
176 * interrupt if we set t + m and an application will hang.
177 * Therefore we need to make a more complex computation assuming
178 * that there exists a k for which the following is true:
179 * k * t + base < mc + delta
180 * (k + 1) * t + base > mc + delta
181 * where t is the interval in hpet ticks for the given freq,
182 * base is the theoretical start value 0 < base < t,
183 * mc is the main counter value at the time of the interrupt,
184 * delta is the time it takes to write the a value to the
185 * comparator.
186 * k may then be computed as (mc - base + delta) / t .
187 */
188 base = mc % t;
189 k = (mc - base + hpetp->hp_delta) / t;
190 write_counter(t * (k + 1) + base,
191 &devp->hd_timer->hpet_compare);
172 } 192 }
173 193
174 if (devp->hd_flags & HPET_SHARED_IRQ) 194 if (devp->hd_flags & HPET_SHARED_IRQ)
@@ -250,7 +270,7 @@ static int hpet_open(struct inode *inode, struct file *file)
250 if (file->f_mode & FMODE_WRITE) 270 if (file->f_mode & FMODE_WRITE)
251 return -EINVAL; 271 return -EINVAL;
252 272
253 lock_kernel(); 273 mutex_lock(&hpet_mutex);
254 spin_lock_irq(&hpet_lock); 274 spin_lock_irq(&hpet_lock);
255 275
256 for (devp = NULL, hpetp = hpets; hpetp && !devp; hpetp = hpetp->hp_next) 276 for (devp = NULL, hpetp = hpets; hpetp && !devp; hpetp = hpetp->hp_next)
@@ -264,7 +284,7 @@ static int hpet_open(struct inode *inode, struct file *file)
264 284
265 if (!devp) { 285 if (!devp) {
266 spin_unlock_irq(&hpet_lock); 286 spin_unlock_irq(&hpet_lock);
267 unlock_kernel(); 287 mutex_unlock(&hpet_mutex);
268 return -EBUSY; 288 return -EBUSY;
269 } 289 }
270 290
@@ -272,7 +292,7 @@ static int hpet_open(struct inode *inode, struct file *file)
272 devp->hd_irqdata = 0; 292 devp->hd_irqdata = 0;
273 devp->hd_flags |= HPET_OPEN; 293 devp->hd_flags |= HPET_OPEN;
274 spin_unlock_irq(&hpet_lock); 294 spin_unlock_irq(&hpet_lock);
275 unlock_kernel(); 295 mutex_unlock(&hpet_mutex);
276 296
277 hpet_timer_set_irq(devp); 297 hpet_timer_set_irq(devp);
278 298
@@ -429,22 +449,6 @@ static int hpet_release(struct inode *inode, struct file *file)
429 return 0; 449 return 0;
430} 450}
431 451
432static int hpet_ioctl_common(struct hpet_dev *, int, unsigned long, int);
433
434static long hpet_ioctl(struct file *file, unsigned int cmd,
435 unsigned long arg)
436{
437 struct hpet_dev *devp;
438 int ret;
439
440 devp = file->private_data;
441 lock_kernel();
442 ret = hpet_ioctl_common(devp, cmd, arg, 0);
443 unlock_kernel();
444
445 return ret;
446}
447
448static int hpet_ioctl_ieon(struct hpet_dev *devp) 452static int hpet_ioctl_ieon(struct hpet_dev *devp)
449{ 453{
450 struct hpet_timer __iomem *timer; 454 struct hpet_timer __iomem *timer;
@@ -479,6 +483,21 @@ static int hpet_ioctl_ieon(struct hpet_dev *devp)
479 if (irq) { 483 if (irq) {
480 unsigned long irq_flags; 484 unsigned long irq_flags;
481 485
486 if (devp->hd_flags & HPET_SHARED_IRQ) {
487 /*
488 * To prevent the interrupt handler from seeing an
489 * unwanted interrupt status bit, program the timer
490 * so that it will not fire in the near future ...
491 */
492 writel(readl(&timer->hpet_config) & ~Tn_TYPE_CNF_MASK,
493 &timer->hpet_config);
494 write_counter(read_counter(&hpet->hpet_mc),
495 &timer->hpet_compare);
496 /* ... and clear any left-over status. */
497 isr = 1 << (devp - devp->hd_hpets->hp_dev);
498 writel(isr, &hpet->hpet_isr);
499 }
500
482 sprintf(devp->hd_name, "hpet%d", (int)(devp - hpetp->hp_dev)); 501 sprintf(devp->hd_name, "hpet%d", (int)(devp - hpetp->hp_dev));
483 irq_flags = devp->hd_flags & HPET_SHARED_IRQ 502 irq_flags = devp->hd_flags & HPET_SHARED_IRQ
484 ? IRQF_SHARED : IRQF_DISABLED; 503 ? IRQF_SHARED : IRQF_DISABLED;
@@ -553,7 +572,8 @@ static inline unsigned long hpet_time_div(struct hpets *hpets,
553} 572}
554 573
555static int 574static int
556hpet_ioctl_common(struct hpet_dev *devp, int cmd, unsigned long arg, int kernel) 575hpet_ioctl_common(struct hpet_dev *devp, int cmd, unsigned long arg,
576 struct hpet_info *info)
557{ 577{
558 struct hpet_timer __iomem *timer; 578 struct hpet_timer __iomem *timer;
559 struct hpet __iomem *hpet; 579 struct hpet __iomem *hpet;
@@ -594,23 +614,14 @@ hpet_ioctl_common(struct hpet_dev *devp, int cmd, unsigned long arg, int kernel)
594 break; 614 break;
595 case HPET_INFO: 615 case HPET_INFO:
596 { 616 {
597 struct hpet_info info; 617 memset(info, 0, sizeof(*info));
598
599 if (devp->hd_ireqfreq) 618 if (devp->hd_ireqfreq)
600 info.hi_ireqfreq = 619 info->hi_ireqfreq =
601 hpet_time_div(hpetp, devp->hd_ireqfreq); 620 hpet_time_div(hpetp, devp->hd_ireqfreq);
602 else 621 info->hi_flags =
603 info.hi_ireqfreq = 0;
604 info.hi_flags =
605 readq(&timer->hpet_config) & Tn_PER_INT_CAP_MASK; 622 readq(&timer->hpet_config) & Tn_PER_INT_CAP_MASK;
606 info.hi_hpet = hpetp->hp_which; 623 info->hi_hpet = hpetp->hp_which;
607 info.hi_timer = devp - hpetp->hp_dev; 624 info->hi_timer = devp - hpetp->hp_dev;
608 if (kernel)
609 memcpy((void *)arg, &info, sizeof(info));
610 else
611 if (copy_to_user((void __user *)arg, &info,
612 sizeof(info)))
613 err = -EFAULT;
614 break; 625 break;
615 } 626 }
616 case HPET_EPI: 627 case HPET_EPI:
@@ -636,7 +647,7 @@ hpet_ioctl_common(struct hpet_dev *devp, int cmd, unsigned long arg, int kernel)
636 devp->hd_flags &= ~HPET_PERIODIC; 647 devp->hd_flags &= ~HPET_PERIODIC;
637 break; 648 break;
638 case HPET_IRQFREQ: 649 case HPET_IRQFREQ:
639 if (!kernel && (arg > hpet_max_freq) && 650 if ((arg > hpet_max_freq) &&
640 !capable(CAP_SYS_RESOURCE)) { 651 !capable(CAP_SYS_RESOURCE)) {
641 err = -EACCES; 652 err = -EACCES;
642 break; 653 break;
@@ -653,12 +664,63 @@ hpet_ioctl_common(struct hpet_dev *devp, int cmd, unsigned long arg, int kernel)
653 return err; 664 return err;
654} 665}
655 666
667static long
668hpet_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
669{
670 struct hpet_info info;
671 int err;
672
673 mutex_lock(&hpet_mutex);
674 err = hpet_ioctl_common(file->private_data, cmd, arg, &info);
675 mutex_unlock(&hpet_mutex);
676
677 if ((cmd == HPET_INFO) && !err &&
678 (copy_to_user((void __user *)arg, &info, sizeof(info))))
679 err = -EFAULT;
680
681 return err;
682}
683
684#ifdef CONFIG_COMPAT
685struct compat_hpet_info {
686 compat_ulong_t hi_ireqfreq; /* Hz */
687 compat_ulong_t hi_flags; /* information */
688 unsigned short hi_hpet;
689 unsigned short hi_timer;
690};
691
692static long
693hpet_compat_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
694{
695 struct hpet_info info;
696 int err;
697
698 mutex_lock(&hpet_mutex);
699 err = hpet_ioctl_common(file->private_data, cmd, arg, &info);
700 mutex_unlock(&hpet_mutex);
701
702 if ((cmd == HPET_INFO) && !err) {
703 struct compat_hpet_info __user *u = compat_ptr(arg);
704 if (put_user(info.hi_ireqfreq, &u->hi_ireqfreq) ||
705 put_user(info.hi_flags, &u->hi_flags) ||
706 put_user(info.hi_hpet, &u->hi_hpet) ||
707 put_user(info.hi_timer, &u->hi_timer))
708 err = -EFAULT;
709 }
710
711 return err;
712}
713#endif
714
656static const struct file_operations hpet_fops = { 715static const struct file_operations hpet_fops = {
657 .owner = THIS_MODULE, 716 .owner = THIS_MODULE,
658 .llseek = no_llseek, 717 .llseek = no_llseek,
659 .read = hpet_read, 718 .read = hpet_read,
660 .poll = hpet_poll, 719 .poll = hpet_poll,
661 .unlocked_ioctl = hpet_ioctl, 720 .unlocked_ioctl = hpet_ioctl,
721#ifdef CONFIG_COMPAT
722 .compat_ioctl = hpet_compat_ioctl,
723#endif
662 .open = hpet_open, 724 .open = hpet_open,
663 .release = hpet_release, 725 .release = hpet_release,
664 .fasync = hpet_fasync, 726 .fasync = hpet_fasync,
@@ -781,7 +843,7 @@ int hpet_alloc(struct hpet_data *hdp)
781 struct hpets *hpetp; 843 struct hpets *hpetp;
782 size_t siz; 844 size_t siz;
783 struct hpet __iomem *hpet; 845 struct hpet __iomem *hpet;
784 static struct hpets *last = NULL; 846 static struct hpets *last;
785 unsigned long period; 847 unsigned long period;
786 unsigned long long temp; 848 unsigned long long temp;
787 u32 remainder; 849 u32 remainder;
@@ -891,9 +953,7 @@ int hpet_alloc(struct hpet_data *hdp)
891 if (!hpet_clocksource) { 953 if (!hpet_clocksource) {
892 hpet_mctr = (void __iomem *)&hpetp->hp_hpet->hpet_mc; 954 hpet_mctr = (void __iomem *)&hpetp->hp_hpet->hpet_mc;
893 CLKSRC_FSYS_MMIO_SET(clocksource_hpet.fsys_mmio, hpet_mctr); 955 CLKSRC_FSYS_MMIO_SET(clocksource_hpet.fsys_mmio, hpet_mctr);
894 clocksource_hpet.mult = clocksource_hz2mult(hpetp->hp_tick_freq, 956 clocksource_register_hz(&clocksource_hpet, hpetp->hp_tick_freq);
895 clocksource_hpet.shift);
896 clocksource_register(&clocksource_hpet);
897 hpetp->hp_clocksource = &clocksource_hpet; 957 hpetp->hp_clocksource = &clocksource_hpet;
898 hpet_clocksource = &clocksource_hpet; 958 hpet_clocksource = &clocksource_hpet;
899 } 959 }
@@ -970,6 +1030,8 @@ static int hpet_acpi_add(struct acpi_device *device)
970 return -ENODEV; 1030 return -ENODEV;
971 1031
972 if (!data.hd_address || !data.hd_nirqs) { 1032 if (!data.hd_address || !data.hd_nirqs) {
1033 if (data.hd_address)
1034 iounmap(data.hd_address);
973 printk("%s: no address or irqs in _CRS\n", __func__); 1035 printk("%s: no address or irqs in _CRS\n", __func__);
974 return -ENODEV; 1036 return -ENODEV;
975 } 1037 }