On Tue 6.May'08 at 9:21:20 -0700, Daniel Walker wrote:
So I wanted to check the accuracy of div_sc versus clocksource_hz2mult
for the computations in arch/x86/kernel/hpet.c, as you made me curious
about it.
First of all, note that hpet_clockevent.mult can not be computed
using clocksource_hz2mult (as you suggested in the first reply) due
to the different definitions for .mult in clockevents.h and
clocksource.h, so I will skip this one and discuss
clocksource_hpet.mult instead.
I applied the test patch below to implement the different ways of
computing it and this is what I got in the dmesg:
clocksource_hpet.mult orig = 292935555
clocksource_hpet.mult walker = 292935555
clocksource_hpet.mult my patch = 292935555 hpet_period = 69841279
so there is no difference at all at this precision level!
Out of curiosity I computed the "exact" value by hand and
got 292935555.9 so we can't even talk about "error" because
the difference is in the extra digit not considered with
the precision I used to print the results.
And if you look at the patch to do computation using
clocksource_hz2mult() you will see that it is uglier
than the simple call to div_sc(), as you have to
consider an extra variable hpet_freq and do an extra
do_div() together with an extra rounding.
So I think you will be ok with the patch now, right? :-)
Thanks,
Carlos
PS: I also computed the "exact" value for hpet_clockevent.mult
by hand and got
hpet_clockevent.mult = 61496114.58 (exact)
hpet_clockevent.mult = 61496114 (my patch)
hpet_clockevent.mult = 61496110 (original)
so my patch in fact improves the accuracy (due to the fact that
it removes the extra do_div() to compute the frequency, which
originaly did not have the rounding trick.)
diff --git a/arch/x86/kernel/hpet.c b/arch/x86/kernel/hpet.c
index 9b5cfcd..ee07694 100644
--- a/arch/x86/kernel/hpet.c
+++ b/arch/x86/kernel/hpet.c
@@ -326,6 +326,7 @@ static int hpet_clocksource_register(void)
{
u64 tmp, start, now;
cycle_t t1;
+ uint64_t hpet_freq;
/* Start the counter */
hpet_start_counter();
@@ -366,6 +367,21 @@ static int hpet_clocksource_register(void)
tmp = (u64)hpet_period << HPET_SHIFT;
do_div(tmp, FSEC_PER_NSEC);
clocksource_hpet.mult = (u32)tmp;
+ printk(KERN_INFO "clocksource_hpet.mult orig = %u\n",
+ clocksource_hpet.mult);
+
+ clocksource_hpet.mult = 0;
+ hpet_freq = 1000000000000000ULL;
+ hpet_freq += hpet_period/2; /* round for do_div */
+ do_div(hpet_freq, hpet_period);
+ clocksource_hpet.mult = clocksource_hz2mult((u32)hpet_freq, (u32)HPET_SHIFT);
+ printk(KERN_INFO "clocksource_hpet.mult walker = %u\n",
+ clocksource_hpet.mult);
+
+ clocksource_hpet.mult = 0;
+ clocksource_hpet.mult = div_sc(hpet_period, FSEC_PER_NSEC, HPET_SHIFT);
+ printk(KERN_INFO "clocksource_hpet.mult my patch = %u hpet_period = %lu\n",
+ clocksource_hpet.mult, hpet_period);
clocksource_register(&clocksource_hpet);
--