timer.c 1.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  1. #include "amt630h.h"
  2. #include "timer.h"
  3. #define APBTMR_CONTROL_ENABLE (1 << 0)
  4. /* 1: periodic, 0:free running. */
  5. #define APBTMR_CONTROL_MODE_PERIODIC (1 << 1)
  6. #define APBTMR_CONTROL_INT_DISABLE (1 << 2)
  7. #define TIMER_CLK CLK_24MHZ
  8. #define TIMER_PERIOD 1000000
  9. #define TIMER_LOAD_VAL 0xffffffff
  10. static unsigned long timestamp;
  11. static unsigned long lastdec;
  12. void timer_init(void)
  13. {
  14. rTIMER0_CONTROL = 0;
  15. rTIMER0_LOAD_COUNT = TIMER_LOAD_VAL;
  16. /* No timer interrupt */
  17. rTIMER0_CONTROL = APBTMR_CONTROL_INT_DISABLE |
  18. APBTMR_CONTROL_MODE_PERIODIC | APBTMR_CONTROL_ENABLE;
  19. }
  20. void udelay(unsigned long usec)
  21. {
  22. long tmo = usec * (TIMER_CLK / TIMER_PERIOD);
  23. unsigned long last = rTIMER0_CURRENT_VALUE;
  24. unsigned long now;
  25. while (tmo > 0) {
  26. now = rTIMER0_CURRENT_VALUE;
  27. if (last >= now)
  28. tmo -= last - now;
  29. else
  30. tmo -= last + (TIMER_LOAD_VAL - now);
  31. last = now;
  32. }
  33. }
  34. void mdelay(unsigned long msec)
  35. {
  36. udelay(1000*msec);
  37. }
  38. #if 0
  39. void reset_timer_masked (void)
  40. {
  41. /* reset time */
  42. lastdec = rTIMER0_CURRENT_VALUE; /* capure current decrementer value time */
  43. timestamp = 0; /* start "advancing" time stamp from 0 */
  44. }
  45. #endif
  46. ULONG get_timer_masked (void)
  47. {
  48. unsigned long now = rTIMER0_CURRENT_VALUE; /* current tick value */
  49. if (lastdec >= now) { /* normal mode (non roll) */
  50. /* normal mode */
  51. timestamp += lastdec - now; /* move stamp fordward with absoulte diff ticks */
  52. } else { /* we have overflow of the count down timer */
  53. /* nts = ts + ld + (TLV - now)
  54. * ts=old stamp, ld=time that passed before passing through -1
  55. * (TLV-now) amount of time after passing though -1
  56. * nts = new "advancing time stamp"...it could also roll and cause problems.
  57. */
  58. timestamp += lastdec + (TIMER_LOAD_VAL - now);
  59. }
  60. lastdec = now;
  61. return timestamp / DIV_ROUND_UP(TIMER_CLK, TIMER_PERIOD) / 1000;
  62. }
  63. ULONG get_timer(ULONG base)
  64. {
  65. return get_timer_masked () - base;
  66. }