2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file event_queue.h
|
|
|
|
*
|
|
|
|
* @date Apr 17, 2014
|
2020-01-13 18:57:43 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "scheduler.h"
|
|
|
|
#include "utlist.h"
|
2023-03-07 05:45:17 -08:00
|
|
|
#include <rusefi/expected.h>
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-01-20 22:40:11 -08:00
|
|
|
#pragma once
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
#define QUEUE_LENGTH_LIMIT 1000
|
|
|
|
|
2019-11-23 14:04:51 -08:00
|
|
|
// templates do not accept field names so we use a macro here
|
2022-12-10 14:07:02 -08:00
|
|
|
#define assertNotInListMethodBody(head, element, field) \
|
2019-11-23 14:04:51 -08:00
|
|
|
/* this code is just to validate state, no functional load*/ \
|
2022-12-10 14:07:02 -08:00
|
|
|
decltype(head) current; \
|
2019-11-23 14:04:51 -08:00
|
|
|
int counter = 0; \
|
|
|
|
LL_FOREACH2(head, current, field) { \
|
|
|
|
if (++counter > QUEUE_LENGTH_LIMIT) { \
|
2023-04-11 17:01:34 -07:00
|
|
|
firmwareError(ObdCode::CUSTOM_ERR_LOOPED_QUEUE, "Looped queue?"); \
|
2019-11-23 14:04:51 -08:00
|
|
|
return false; \
|
|
|
|
} \
|
|
|
|
if (current == element) { \
|
|
|
|
/** \
|
|
|
|
* for example, this might happen in case of sudden RPM change if event \
|
|
|
|
* was not scheduled by angle but was scheduled by time. In case of scheduling \
|
|
|
|
* by time with slow RPM the whole next fast revolution might be within the wait period \
|
|
|
|
*/ \
|
2023-04-11 17:01:34 -07:00
|
|
|
warning(ObdCode::CUSTOM_RE_ADDING_INTO_EXECUTION_QUEUE, "re-adding element into event_queue"); \
|
2019-11-23 14:04:51 -08:00
|
|
|
return true; \
|
|
|
|
} \
|
|
|
|
} \
|
2015-07-10 06:01:56 -07:00
|
|
|
return false;
|
2019-11-23 14:04:51 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2016-01-25 20:01:36 -08:00
|
|
|
/**
|
|
|
|
* Execution sorted linked list
|
|
|
|
*/
|
2015-07-10 06:01:56 -07:00
|
|
|
class EventQueue {
|
|
|
|
public:
|
2024-01-07 13:04:36 -08:00
|
|
|
// See comment in EventQueue::executeAll for info about lateDelay - it sets the
|
2020-06-16 19:13:42 -07:00
|
|
|
// time gap between events for which we will wait instead of rescheduling the next
|
|
|
|
// event in a group of events near one another.
|
2024-01-07 13:21:54 -08:00
|
|
|
explicit EventQueue(efitick_t lateDelay = 0);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
/**
|
2016-01-25 20:01:36 -08:00
|
|
|
* O(size) - linear search in sorted linked list
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2022-09-11 13:08:11 -07:00
|
|
|
bool insertTask(scheduling_s *scheduling, efitick_t timeX, action_s action);
|
2021-08-26 13:03:09 -07:00
|
|
|
void remove(scheduling_s* scheduling);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2022-09-11 13:08:11 -07:00
|
|
|
int executeAll(efitick_t now);
|
|
|
|
bool executeOne(efitick_t now);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2022-09-11 13:08:11 -07:00
|
|
|
expected<efitick_t> getNextEventTime(efitick_t nowUs) const;
|
2015-07-10 06:01:56 -07:00
|
|
|
void clear(void);
|
2019-06-08 06:51:36 -07:00
|
|
|
int size(void) const;
|
2019-12-02 21:20:47 -08:00
|
|
|
scheduling_s *getElementAtIndexForUnitText(int index);
|
2015-07-10 06:01:56 -07:00
|
|
|
scheduling_s * getHead();
|
2024-01-07 13:21:54 -08:00
|
|
|
|
|
|
|
scheduling_s* getFreeScheduling();
|
|
|
|
void tryReturnScheduling(scheduling_s* sched);
|
2015-07-10 06:01:56 -07:00
|
|
|
private:
|
2023-11-17 13:53:57 -08:00
|
|
|
void assertListIsSorted() const;
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* this list is sorted
|
|
|
|
*/
|
2024-01-07 13:04:36 -08:00
|
|
|
scheduling_s *m_head = nullptr;
|
2024-01-07 13:21:54 -08:00
|
|
|
const efitick_t m_lateDelay;
|
|
|
|
|
|
|
|
scheduling_s* m_freelist = nullptr;
|
|
|
|
scheduling_s m_pool[64];
|
2015-07-10 06:01:56 -07:00
|
|
|
};
|
|
|
|
|