time.cpp 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. #include "all-headers.h"
  2. static volatile uint32_t gUptime;
  3. static void startSystick (BOOT_MODE) {
  4. SYST_RVR = CPU_MHZ * 1000 - 1 ; // Underflow every ms
  5. SYST_CVR = 0 ;
  6. SYST_CSR = SYST_CSR_CLKSOURCE | SYST_CSR_ENABLE ;
  7. // Configure and chain PIT0 and PIT1 for 64-bit counting
  8. // Power on PIT
  9. SIM_SCGC6 |= SIM_SCGC6_PIT ;
  10. // Enable PIT module
  11. PIT_MCR = 0 ;
  12. // Disable PIT0 and PIT1
  13. PIT_TCTRL (0) = 0 ;
  14. PIT_TCTRL (1) = 0 ;
  15. // PIT0 and PIT1 down-count: initialize them with all 1's
  16. PIT_LDVAL (0) = UINT32_MAX ;
  17. PIT_LDVAL (1) = UINT32_MAX ;
  18. // Enable PIT0 and PIT1: start counting, chain PI1 to PIT0, no interrupt
  19. PIT_TCTRL (1) = PIT_TCTRL_CHN | PIT_TCTRL_TEN ;
  20. PIT_TCTRL (0) = PIT_TCTRL_TEN ;
  21. }
  22. MACRO_BOOT_ROUTINE (startSystick);
  23. static void activateSystickInterrupt (INIT_MODE) {
  24. SYST_CSR |= SYST_CSR_TICKINT;
  25. }
  26. MACRO_INIT_ROUTINE (activateSystickInterrupt);
  27. void busyWaitDuring_initMode (INIT_MODE_ const uint32_t inDelayMS) {
  28. const uint32_t COUNTFLAG_MASK = 1 << 16 ;
  29. for (uint32_t i=0 ; i<inDelayMS ; i++) {
  30. while ((SYST_CSR & COUNTFLAG_MASK) == 0) {} // Busy wait, polling COUNTFLAG
  31. }
  32. }
  33. void busyWaitDuring (USER_MODE_ const uint32_t inDelayMS) {
  34. busyWaitUntil(MODE_ gUptime + inDelayMS);
  35. }
  36. void busyWaitUntil (USER_MODE_ const uint32_t inDeadLineMS) {
  37. while (inDeadLineMS > gUptime) {}
  38. }
  39. void systickInterruptServiceRoutine (SECTION_MODE) {
  40. const uint32_t newUptime = gUptime + 1 ;
  41. gUptime = newUptime ;
  42. // Run every section routines in real.time.interrupt.routine.array
  43. extern void (* __real_time_interrupt_routine_array_start) (SECTION_MODE_ const uint32_t inUptime) ;
  44. extern void (* __real_time_interrupt_routine_array_end) (SECTION_MODE_ const uint32_t inUptime) ;
  45. void (** ptr) (SECTION_MODE_ const uint32_t) = & __real_time_interrupt_routine_array_start ;
  46. while (ptr != & __real_time_interrupt_routine_array_end) {
  47. (* ptr) (MODE_ newUptime) ;
  48. ptr ++ ;
  49. }
  50. }
  51. uint32_t millis(ANY_MODE) {
  52. return gUptime;
  53. }
  54. uint64_t section_micros(SECTION_MODE) {
  55. // To obtain the correct value, first read LTMR64H and then LTMR64L
  56. uint64_t result = PIT_LTMR64H ;
  57. result <<= 32 ;
  58. result |= PIT_LTMR64L ;
  59. // PIT0 and PIT1 actually downcount
  60. result = ~ result ;
  61. // Divide by the clock frequency in MHz for getting microsecond count
  62. result /= busMHZ () ;
  63. return result;
  64. }
  65. uint32_t systick(ANY_MODE) {
  66. return SYST_CVR;
  67. }