On 5/25/2011 9:40 PM, Graeme Russ wrote: > On Thu, May 26, 2011 at 2:19 PM, Reinhard Meyer > <u-b...@emk-elektronik.de> wrote: >> Dear Graeme Russ, >>> On closer inspection, some do, some don't. All ARMv7 (OMAP, S5P, Tegra2) >>> do. at91 is odd - It looks like it uses interrupts, but get_timer() and >>> udelay() both end up calling get_timer_raw() (with udelay only having >>> millisecond resolution it seems). Some others can be configured to >>> increment the timer using an interrupt. ARM is, quite frankly, a complete >>> mess - It has a mass of *_timer_masked() functions which the core timer >>> functions are 'wafer thin' wrapper around, udelay() silently resets >>> the timebase trashing get_timer() loops etc. >> Please look at current master for at91. >> >> http://git.denx.de/?p=u-boot.git;a=blob;f=arch/arm/cpu/arm926ejs/at91/timer.c;h=a0876879d3907af553d832bea187a062a22b9bd4;hb=5d1ee00b1fe1180503f6dfc10e87a6c6e74778f3 >> >> AT91 uses a 32 bit hardware register that by means of a prescaler is made >> to increment at a rate in the low megahertz range. > Yes, I see that now > >> This results in a wrap approximately every 1000 seconds. >> Actually this would be sufficient for all known uses of udelay() and >> get_timer() >> timeout loops. However, this hardware register is extended to 64 bits by >> software >> every time it is read (by detecting rollovers). > Which makes it 100% compatible with my proposed solution - The software > prescaler will trigger the 64-bit extension and rollover detection > >> Since a wrap of that 64 bit "tick" would occur after the earth has ended, >> it is simple to obtain milliseconds from it by doing a 64 bit division. > Which would be done in the common prescaler in /lib/ > > Currently, most ARM specific utilisations of get_timer() enforce a reset > of the tick counter by calling reset_timer() - Subsequent calls to > get_timer() then assume a start time of zero. Provided the internal timer > rolls over currectly, the initial call of get_timer(0) will reset the ms > timer and remove and 'glitch' present due to not calling the 'extender' > function between 32-bit rollovers which makes the reset_timer() call > unneccessary - I believe at91 behaves correctly in this regard. > > In any case, the underlying assumption made by the ARM timer interface > (call reset_timer() first always) is inherently broken as not all users > of the timer API do this - They assume a sane behaviour of: > > start = get_timer(0); > elapsed_time = get_timer(start); > > Add to this udelay() resetting the timer make the following very broken: > > start = get_timer(0); > while(condition) { > udelay(delay); > } > elapsed_time = get_timer(start); > > NOTE: In this case, if udelay() also calls the prescaler then no interrupt > triggered every 1000s would be required in the above example to get > correct elapsed_time even if the loop ran for several hours (provided > udelay() is called at least every 1000s > > However, to allow timing of independent events with no intervening > udelay() or get_timer() calls, an 1000s interrupt to kick the prescaler is > all that is needed to make this particular implementation behave correctly. Hi All, True, if the processor supports timer interrupts. The problem is that the existing u-boots in many cases do not. I think that is really the crux of the problem. So what are we going to do? I am open to ideas here.
Best Regards, Bill Campbell > Of course disabling interruts and not calling get_timer() or udelay() will > break the timer - But there is nothing that can be done about that) > > Regards, > > Graeme > > _______________________________________________ U-Boot mailing list U-Boot@lists.denx.de http://lists.denx.de/mailman/listinfo/u-boot