diff options
| author | Michele Bini <michele.bini@gmail.com> | 2022-03-30 20:50:42 (GMT) |
|---|---|---|
| committer | Michele Bini <michele.bini@gmail.com> | 2022-03-30 20:50:42 (GMT) |
| commit | 5db030547eeb0aae422447ddb63f0bd2f9d5f384 (patch) | |
| tree | 648bdc9654a3d826eaef1cefaa4b329495016569 /src/heartratetask | |
| parent | e1cb4f64097a7d084f178f762546cecb5bd3c6be (diff) | |
Revert "sans heart"ultraredux-heart
This reverts commit 6ef420d2407a4685b56a233f6b0f849e90c6cf49.
Diffstat (limited to 'src/heartratetask')
| -rw-r--r-- | src/heartratetask/HeartRateTask.cpp | 100 | ||||
| -rw-r--r-- | src/heartratetask/HeartRateTask.h | 40 |
2 files changed, 140 insertions, 0 deletions
diff --git a/src/heartratetask/HeartRateTask.cpp b/src/heartratetask/HeartRateTask.cpp new file mode 100644 index 0000000..2f689b9 --- /dev/null +++ b/src/heartratetask/HeartRateTask.cpp @@ -0,0 +1,100 @@ +#include "heartratetask/HeartRateTask.h" +#include <drivers/Hrs3300.h> +#include <components/heartrate/HeartRateController.h> +#include <nrf_log.h> + +using namespace Pinetime::Applications; + +HeartRateTask::HeartRateTask(Drivers::Hrs3300& heartRateSensor, Controllers::HeartRateController& controller) + : heartRateSensor {heartRateSensor}, controller {controller}, ppg{} { +} + +void HeartRateTask::Start() { + messageQueue = xQueueCreate(10, 1); + controller.SetHeartRateTask(this); + + if (pdPASS != xTaskCreate(HeartRateTask::Process, "Heartrate", 500, this, 0, &taskHandle)) + APP_ERROR_HANDLER(NRF_ERROR_NO_MEM); +} + +void HeartRateTask::Process(void* instance) { + auto* app = static_cast<HeartRateTask*>(instance); + app->Work(); +} + +void HeartRateTask::Work() { + int lastBpm = 0; + while (true) { + Messages msg; + uint32_t delay; + if (state == States::Running) { + if (measurementStarted) + delay = 40; + else + delay = 100; + } else + delay = portMAX_DELAY; + + if (xQueueReceive(messageQueue, &msg, delay)) { + switch (msg) { + case Messages::GoToSleep: + StopMeasurement(); + state = States::Idle; + break; + case Messages::WakeUp: + state = States::Running; + if (measurementStarted) { + lastBpm = 0; + StartMeasurement(); + } + break; + case Messages::StartMeasurement: + if (measurementStarted) + break; + lastBpm = 0; + StartMeasurement(); + measurementStarted = true; + break; + case Messages::StopMeasurement: + if (!measurementStarted) + break; + StopMeasurement(); + measurementStarted = false; + break; + } + } + + if (measurementStarted) { + ppg.Preprocess(static_cast<float>(heartRateSensor.ReadHrs())); + auto bpm = ppg.HeartRate(); + + if (lastBpm == 0 && bpm == 0) + controller.Update(Controllers::HeartRateController::States::NotEnoughData, 0); + if (bpm != 0) { + lastBpm = bpm; + controller.Update(Controllers::HeartRateController::States::Running, lastBpm); + } + } + } +} + +void HeartRateTask::PushMessage(HeartRateTask::Messages msg) { + BaseType_t xHigherPriorityTaskWoken; + xHigherPriorityTaskWoken = pdFALSE; + xQueueSendFromISR(messageQueue, &msg, &xHigherPriorityTaskWoken); + if (xHigherPriorityTaskWoken) { + /* Actual macro used here is port specific. */ + // TODO : should I do something here? + } +} + +void HeartRateTask::StartMeasurement() { + heartRateSensor.Enable(); + vTaskDelay(100); + ppg.SetOffset(static_cast<float>(heartRateSensor.ReadHrs())); +} + +void HeartRateTask::StopMeasurement() { + heartRateSensor.Disable(); + vTaskDelay(100); +} diff --git a/src/heartratetask/HeartRateTask.h b/src/heartratetask/HeartRateTask.h new file mode 100644 index 0000000..0796dc7 --- /dev/null +++ b/src/heartratetask/HeartRateTask.h @@ -0,0 +1,40 @@ +#pragma once +#include <FreeRTOS.h> +#include <task.h> +#include <queue.h> +#include <components/heartrate/Ppg.h> + +namespace Pinetime { + namespace Drivers { + class Hrs3300; + } + namespace Controllers { + class HeartRateController; + } + namespace Applications { + class HeartRateTask { + public: + enum class Messages : uint8_t { GoToSleep, WakeUp, StartMeasurement, StopMeasurement }; + enum class States { Idle, Running }; + + explicit HeartRateTask(Drivers::Hrs3300& heartRateSensor, Controllers::HeartRateController& controller); + void Start(); + void Work(); + void PushMessage(Messages msg); + + private: + static void Process(void* instance); + void StartMeasurement(); + void StopMeasurement(); + + TaskHandle_t taskHandle; + QueueHandle_t messageQueue; + States state = States::Running; + Drivers::Hrs3300& heartRateSensor; + Controllers::HeartRateController& controller; + Controllers::Ppg ppg; + bool measurementStarted = false; + }; + + } +} |
