// Copyright (C) 2019 BogDan Vatra // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #ifndef SENSOREVENTQUEUE_H #define SENSOREVENTQUEUE_H #include "sensormanager.h" #include #include #include #include class FunctionEvent : public QEvent { public: using Function = std::function; enum { FunctionEventType = User + 1 }; explicit FunctionEvent(const Function &function) : QEvent(Type(FunctionEventType)) , m_function(function) {} void callFunction() const { m_function(); } private: Function m_function; }; class ThreadSafeSensorBackend : public QSensorBackend { public: ThreadSafeSensorBackend(QSensor *sensor, QObject *parent = nullptr) : QSensorBackend(sensor, parent) , m_sensorManager(SensorManager::instance()) {} // QObject interface bool event(QEvent *event) override { if (event->type() == FunctionEvent::FunctionEventType) { static_cast(event)->callFunction(); event->accept(); return true; } return QSensorBackend::event(event); } protected: QSharedPointer m_sensorManager; }; template class SensorEventQueue : public ThreadSafeSensorBackend { public: explicit SensorEventQueue(int androidSensorType, QSensor *sensor, QObject *parent = nullptr) : ThreadSafeSensorBackend(sensor, parent) { setReading(&m_reader); m_sensorEventQueue = ASensorManager_createEventQueue(m_sensorManager->manager(), m_sensorManager->looper(), -1, &looperCallback, this); setSensorType(androidSensorType); } ~SensorEventQueue() override { stop(); ASensorManager_destroyEventQueue(m_sensorManager->manager(), m_sensorEventQueue); } void setSensorType(int type) { bool started = m_started; if (started) stop(); m_sensor = ASensorManager_getDefaultSensor(m_sensorManager->manager(), type); setDescription(m_sensorManager->description(m_sensor)); if (started) start(); } // QSensorBackend interface void start() override { ASensorEventQueue_enableSensor(m_sensorEventQueue, m_sensor); // The QSensor datarate is in Hz (frequency), whereas the setEventRate expects microseconds if (sensor()->dataRate() > 0) ASensorEventQueue_setEventRate(m_sensorEventQueue, m_sensor, std::max(ASensor_getMinDelay(m_sensor), 1000000/sensor()->dataRate())); m_started = true; } void stop() override { ASensorEventQueue_disableSensor(m_sensorEventQueue, m_sensor); m_started = false; } bool isFeatureSupported(QSensor::Feature feature) const override { switch (feature) { case QSensor::SkipDuplicates: return true; default: return false; } } protected: virtual void dataReceived(const ASensorEvent &event) = 0; static int looperCallback(int /*fd*/, int /*events*/, void* data) { ASensorEvent sensorEvent; auto self = reinterpret_cast(data); while (ASensorEventQueue_getEvents(self->m_sensorEventQueue, &sensorEvent, 1)) QCoreApplication::postEvent(self, new FunctionEvent{[=]{self->dataReceived(sensorEvent);}}); return 1; // 1 means keep receiving events } protected: T m_reader; const ASensor *m_sensor = nullptr; ASensorEventQueue* m_sensorEventQueue = nullptr; bool m_started = false; }; #endif // SENSOREVENTQUEUE_H