|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348 |
-
-
-
-
- #include <Arduino.h>
- #include "EventResponder.h"
-
- EventResponder * EventResponder::firstYield = nullptr;
- EventResponder * EventResponder::lastYield = nullptr;
- EventResponder * EventResponder::firstInterrupt = nullptr;
- EventResponder * EventResponder::lastInterrupt = nullptr;
- bool EventResponder::runningFromYield = false;
-
-
-
- void EventResponder::triggerEventNotImmediate()
- {
- bool irq = disableInterrupts();
- if (_triggered == false) {
-
- if (_type == EventTypeYield) {
-
- if (firstYield == nullptr) {
- _next = nullptr;
- _prev = nullptr;
- firstYield = this;
- lastYield = this;
- } else {
- _next = nullptr;
- _prev = lastYield;
- _prev->_next = this;
- lastYield = this;
- }
- } else if (_type == EventTypeInterrupt) {
-
- if (firstInterrupt == nullptr) {
- _next = nullptr;
- _prev = nullptr;
- firstInterrupt = this;
- lastInterrupt = this;
- } else {
- _next = nullptr;
- _prev = lastInterrupt;
- _prev->_next = this;
- lastInterrupt = this;
- }
- SCB_ICSR = SCB_ICSR_PENDSVSET;
- } else {
-
- }
- _triggered = true;
- }
- enableInterrupts(irq);
- }
-
- extern "C" void pendablesrvreq_isr(void)
- {
- EventResponder::runFromInterrupt();
- }
-
- void EventResponder::runFromInterrupt()
- {
- while (1) {
- bool irq = disableInterrupts();
- EventResponder *first = firstInterrupt;
- if (first) {
- firstInterrupt = first->_next;
- if (firstInterrupt) {
- firstInterrupt->_prev = nullptr;
- } else {
- lastInterrupt = nullptr;
- }
- enableInterrupts(irq);
- first->_triggered = false;
- (*(first->_function))(*first);
- } else {
- enableInterrupts(irq);
- break;
- }
- }
- }
-
- bool EventResponder::clearEvent()
- {
- bool ret = false;
- bool irq = disableInterrupts();
- if (_triggered) {
- if (_type == EventTypeYield) {
- if (_prev) {
- _prev->_next = _next;
- } else {
- firstYield = _next;
- }
- if (_next) {
- _next->_prev = _prev;
- } else {
- lastYield = _prev;
- }
- } else if (_type == EventTypeInterrupt) {
- if (_prev) {
- _prev->_next = _next;
- } else {
- firstInterrupt = _next;
- }
- if (_next) {
- _next->_prev = _prev;
- } else {
- lastInterrupt = _prev;
- }
- }
- _triggered = false;
- ret = true;
- }
- enableInterrupts(irq);
- return ret;
- }
-
-
- void EventResponder::detachNoInterrupts()
- {
- if (_type == EventTypeYield) {
- if (_triggered) {
- if (_prev) {
- _prev->_next = _next;
- } else {
- firstYield = _next;
- }
- if (_next) {
- _next->_prev = _prev;
- } else {
- lastYield = _prev;
- }
- }
- _type = EventTypeDetached;
- } else if (_type == EventTypeInterrupt) {
- if (_triggered) {
- if (_prev) {
- _prev->_next = _next;
- } else {
- firstInterrupt = _next;
- }
- if (_next) {
- _next->_prev = _prev;
- } else {
- lastInterrupt = _prev;
- }
- }
- _type = EventTypeDetached;
- }
- }
-
-
-
-
-
- MillisTimer * MillisTimer::listWaiting = nullptr;
- MillisTimer * MillisTimer::listActive = nullptr;
-
- void MillisTimer::begin(unsigned long milliseconds, EventResponderRef event)
- {
- if (_state != TimerOff) end();
- if (!milliseconds) return;
- _event = &event;
- _ms = (milliseconds > 2)? milliseconds-2 : 0;
- _reload = 0;
- addToWaitingList();
- }
-
- void MillisTimer::beginRepeating(unsigned long milliseconds, EventResponderRef event)
- {
- if (_state != TimerOff) end();
- if (!milliseconds) return;
- _event = &event;
- _ms = (milliseconds > 2)? milliseconds-2 : 0;
- _reload = milliseconds;
- addToWaitingList();
- }
-
- void MillisTimer::addToWaitingList()
- {
- _prev = nullptr;
- bool irq = disableTimerInterrupt();
- _next = listWaiting;
- listWaiting = this;
- _state = TimerWaiting;
- enableTimerInterrupt(irq);
- }
-
- void MillisTimer::addToActiveList()
- {
- if (listActive == nullptr) {
-
- _next = nullptr;
- _prev = nullptr;
- listActive = this;
- } else if (_ms < listActive->_ms) {
-
- _next = listActive;
- _prev = nullptr;
- listActive->_prev = this;
-
- listActive->_ms -= _ms;
- listActive = this;
- } else {
-
- MillisTimer *timer = listActive;
- while (timer->_next) {
- _ms -= timer->_ms;
- timer = timer->_next;
- if (_ms < timer->_ms) {
-
- _next = timer;
- _prev = timer->_prev;
- timer->_prev = this;
- _prev->_next = this;
- timer->_ms -= _ms;
- _state = TimerActive;
- return;
- }
- }
-
- _ms -= timer->_ms;
- _next = nullptr;
- _prev = timer;
- timer->_next = this;
- }
- _state = TimerActive;
- }
-
- void MillisTimer::end()
- {
- bool irq = disableTimerInterrupt();
- TimerStateType s = _state;
- if (s == TimerActive) {
- if (_next) {
- _next->_prev = _prev;
- _next->_ms += _ms;
- }
- if (_prev) {
- _prev->_next = _next;
- } else {
- listActive = _next;
- }
- _state = TimerOff;
- } else if (s == TimerWaiting) {
- if (listWaiting == this) {
- listWaiting = _next;
- } else {
- MillisTimer *timer = listWaiting;
- while (timer) {
- if (timer->_next == this) {
- timer->_next = _next;
- break;
- }
- timer = timer->_next;
- }
- }
- _state = TimerOff;
- }
- enableTimerInterrupt(irq);
- }
-
- void MillisTimer::runFromTimer()
- {
- MillisTimer *timer = listActive;
- while (timer) {
- if (timer->_ms > 0) {
- timer->_ms--;
- break;
- } else {
- MillisTimer *next = timer->_next;
- if (next) next->_prev = nullptr;
- listActive = next;
- timer->_state = TimerOff;
- EventResponderRef event = *(timer->_event);
- event.triggerEvent(0, timer);
- if (timer->_reload) {
- timer->_ms = timer->_reload;
- timer->addToActiveList();
- }
- timer = listActive;
- }
- }
- bool irq = disableTimerInterrupt();
- MillisTimer *waiting = listWaiting;
- listWaiting = nullptr;
- enableTimerInterrupt(irq);
- while (waiting) {
- MillisTimer *next = waiting->_next;
- waiting->addToActiveList();
- waiting = next;
- }
- }
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- extern "C" volatile uint32_t systick_millis_count;
- extern "C" volatile uint32_t systick_cycle_count;
- extern "C" uint32_t systick_safe_read;
- extern "C" void systick_isr(void)
- {
- systick_cycle_count = ARM_DWT_CYCCNT;
- systick_millis_count++;
- MillisTimer::runFromTimer();
- }
-
-
|