Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

148 lines
4.5KB

  1. /* Teensyduino Core Library
  2. * http://www.pjrc.com/teensy/
  3. * Copyright (c) 2017 PJRC.COM, LLC.
  4. *
  5. * Permission is hereby granted, free of charge, to any person obtaining
  6. * a copy of this software and associated documentation files (the
  7. * "Software"), to deal in the Software without restriction, including
  8. * without limitation the rights to use, copy, modify, merge, publish,
  9. * distribute, sublicense, and/or sell copies of the Software, and to
  10. * permit persons to whom the Software is furnished to do so, subject to
  11. * the following conditions:
  12. *
  13. * 1. The above copyright notice and this permission notice shall be
  14. * included in all copies or substantial portions of the Software.
  15. *
  16. * 2. If the Software is incorporated into a build system that allows
  17. * selection among a list of target devices, then similar target
  18. * devices manufactured by PJRC.COM must be included in the list of
  19. * target devices and selectable in the same manner.
  20. *
  21. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  22. * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  23. * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  24. * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
  25. * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
  26. * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
  27. * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  28. * SOFTWARE.
  29. */
  30. #ifndef __INTERVALTIMER_H__
  31. #define __INTERVALTIMER_H__
  32. #include "kinetis.h"
  33. #ifdef __cplusplus
  34. extern "C" {
  35. #endif
  36. class IntervalTimer {
  37. private:
  38. static const uint32_t MAX_PERIOD = UINT32_MAX / (F_BUS / 1000000.0);
  39. public:
  40. IntervalTimer() {
  41. channel = NULL;
  42. nvic_priority = 128;
  43. }
  44. ~IntervalTimer() {
  45. end();
  46. }
  47. bool begin(void (*funct)(), unsigned int microseconds) {
  48. if (microseconds == 0 || microseconds > MAX_PERIOD) return false;
  49. uint32_t cycles = (F_BUS / 1000000) * microseconds - 1;
  50. if (cycles < 36) return false;
  51. return beginCycles(funct, cycles);
  52. }
  53. bool begin(void (*funct)(), int microseconds) {
  54. if (microseconds < 0) return false;
  55. return begin(funct, (unsigned int)microseconds);
  56. }
  57. bool begin(void (*funct)(), unsigned long microseconds) {
  58. return begin(funct, (unsigned int)microseconds);
  59. }
  60. bool begin(void (*funct)(), long microseconds) {
  61. return begin(funct, (int)microseconds);
  62. }
  63. bool begin(void (*funct)(), float microseconds) {
  64. if (microseconds <= 0 || microseconds > MAX_PERIOD) return false;
  65. uint32_t cycles = (float)(F_BUS / 1000000) * microseconds - 0.5;
  66. if (cycles < 36) return false;
  67. return beginCycles(funct, cycles);
  68. }
  69. bool begin(void (*funct)(), double microseconds) {
  70. return begin(funct, (float)microseconds);
  71. }
  72. void update(unsigned int microseconds) {
  73. if (microseconds == 0 || microseconds > MAX_PERIOD) return;
  74. uint32_t cycles = (F_BUS / 1000000) * microseconds - 1;
  75. if (cycles < 36) return;
  76. if (channel) channel->LDVAL = cycles;
  77. }
  78. void update(int microseconds) {
  79. if (microseconds < 0) return;
  80. return update((unsigned int)microseconds);
  81. }
  82. void update(unsigned long microseconds) {
  83. return update((unsigned int)microseconds);
  84. }
  85. void update(long microseconds) {
  86. return update((int)microseconds);
  87. }
  88. void update(float microseconds) {
  89. if (microseconds <= 0 || microseconds > MAX_PERIOD) return;
  90. uint32_t cycles = (float)(F_BUS / 1000000) * microseconds - 0.5;
  91. if (cycles < 36) return;
  92. if (channel) channel->LDVAL = cycles;
  93. }
  94. void update(double microseconds) {
  95. return update((float)microseconds);
  96. }
  97. void end();
  98. void priority(uint8_t n) {
  99. nvic_priority = n;
  100. #if defined(KINETISK)
  101. if (channel) {
  102. int index = channel - KINETISK_PIT_CHANNELS;
  103. NVIC_SET_PRIORITY(IRQ_PIT_CH0 + index, nvic_priority);
  104. }
  105. #elif defined(KINETISL)
  106. if (channel) {
  107. int index = channel - KINETISK_PIT_CHANNELS;
  108. nvic_priorites[index] = nvic_priority;
  109. if (nvic_priorites[0] <= nvic_priorites[1]) {
  110. NVIC_SET_PRIORITY(IRQ_PIT, nvic_priorites[0]);
  111. } else {
  112. NVIC_SET_PRIORITY(IRQ_PIT, nvic_priorites[1]);
  113. }
  114. }
  115. #endif
  116. }
  117. operator IRQ_NUMBER_t() {
  118. if (channel) {
  119. #if defined(KINETISK)
  120. int index = channel - KINETISK_PIT_CHANNELS;
  121. return (IRQ_NUMBER_t)(IRQ_PIT_CH0 + index);
  122. #elif defined(KINETISL)
  123. return IRQ_PIT;
  124. #endif
  125. }
  126. return (IRQ_NUMBER_t)NVIC_NUM_INTERRUPTS;
  127. }
  128. private:
  129. KINETISK_PIT_CHANNEL_t *channel;
  130. uint8_t nvic_priority;
  131. #if defined(KINETISL)
  132. static uint8_t nvic_priorites[2];
  133. #endif
  134. bool beginCycles(void (*funct)(), uint32_t cycles);
  135. };
  136. #ifdef __cplusplus
  137. }
  138. #endif
  139. #endif