arch_timer.h 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. #ifndef __ASMARM_ARCH_TIMER_H
  2. #define __ASMARM_ARCH_TIMER_H
  3. #include <linux/ioport.h>
  4. #include <linux/clocksource.h>
  5. #include <asm/errno.h>
  6. #define ARCH_TIMER_USR_PCT_ACCESS_EN (1 << 0) /* physical counter */
  7. #define ARCH_TIMER_USR_VCT_ACCESS_EN (1 << 1) /* virtual counter */
  8. #define ARCH_TIMER_VIRT_EVT_EN (1 << 2)
  9. #define ARCH_TIMER_EVT_TRIGGER_SHIFT (4)
  10. #define ARCH_TIMER_EVT_TRIGGER_MASK (0xF << ARCH_TIMER_EVT_TRIGGER_SHIFT)
  11. #define ARCH_TIMER_USR_VT_ACCESS_EN (1 << 8) /* virtual timer registers */
  12. #define ARCH_TIMER_USR_PT_ACCESS_EN (1 << 9) /* physical timer registers */
  13. struct arch_timer {
  14. struct resource res[3];
  15. };
  16. #ifdef CONFIG_ARM_ARCH_TIMER
  17. int arch_timer_register(struct arch_timer *);
  18. int arch_timer_of_register(void);
  19. cycle_t arch_counter_get_cntpct(void);
  20. static inline u32 arch_timer_get_cntkctl(void)
  21. {
  22. u32 cntkctl;
  23. asm volatile("mrc p15, 0, %0, c14, c1, 0" : "=r" (cntkctl));
  24. return cntkctl;
  25. }
  26. static inline void arch_timer_set_cntkctl(u32 cntkctl)
  27. {
  28. asm volatile("mcr p15, 0, %0, c14, c1, 0" : : "r" (cntkctl));
  29. }
  30. static inline void arch_counter_set_user_access(void)
  31. {
  32. u32 cntkctl = arch_timer_get_cntkctl();
  33. /* Disable user access to the timers and the physical counter */
  34. /* Also disable virtual event stream */
  35. cntkctl &= ~(ARCH_TIMER_USR_PT_ACCESS_EN
  36. | ARCH_TIMER_USR_VT_ACCESS_EN
  37. | ARCH_TIMER_VIRT_EVT_EN
  38. | ARCH_TIMER_USR_PCT_ACCESS_EN);
  39. /* Enable user access to the virtual counter */
  40. if (IS_ENABLED(CONFIG_ARM_ARCH_TIMER_VCT_ACCESS))
  41. cntkctl |= ARCH_TIMER_USR_VCT_ACCESS_EN;
  42. else
  43. cntkctl &= ~ARCH_TIMER_USR_VCT_ACCESS_EN;
  44. arch_timer_set_cntkctl(cntkctl);
  45. }
  46. static inline void arch_timer_evtstrm_enable(int divider)
  47. {
  48. u32 cntkctl = arch_timer_get_cntkctl();
  49. cntkctl &= ~ARCH_TIMER_EVT_TRIGGER_MASK;
  50. /* Set the divider and enable virtual event stream */
  51. cntkctl |= (divider << ARCH_TIMER_EVT_TRIGGER_SHIFT)
  52. | ARCH_TIMER_VIRT_EVT_EN;
  53. arch_timer_set_cntkctl(cntkctl);
  54. elf_hwcap |= HWCAP_EVTSTRM;
  55. }
  56. #else
  57. static inline int arch_timer_register(struct arch_timer *at)
  58. {
  59. return -ENXIO;
  60. }
  61. static inline int arch_timer_of_register(void)
  62. {
  63. return -ENXIO;
  64. }
  65. static inline cycle_t arch_counter_get_cntpct(void)
  66. {
  67. return 0;
  68. }
  69. #endif
  70. #endif