/** * @file sensor.h * @brief Base class for sensors. Inherit this class to implement a new type of sensor. * * This file defines the basis for all sensor inputs to the ECU, and provides a registry * so that consumers may be agnostic to how each sensor may work. * * HOW TO ADD A NEW SENSOR TYPE: * * 1. Add an entry to the enum in sensor_type.h. Be sure to add it ABOVE the placeholder * at the end of the list. * * 2. In the init/sensor folder, create/modify logic to create an instance of the new sensor, * configure it if necessary, and call its Register() function if it should be enabled. * See init_oil_pressure.cpp for a minimal example. * * 3. Consume the new sensor with instance(s) of SensorConsumer * * Consumers: * * tl;dr: Use a SensorConsumer. See sensor_consumer.h * * All a consumer does is look up whether a particular sensor is present in the table, * and if so, asks it for the current reading. This could synchronously perform sensor * acquisition and conversion (not recommended), or use a previously stored value (recommended!). * This functionality is implemented in sensor_consumer.h, and sensor.cpp. * * Providers: * Instantiate a subclass of Sensor, and implement the Get() function. * Call Register() to install the new sensor in the registry, preparing it for use. * * Mocking: * The sensor table supports mocking each sensors value. Call Sensor::SetMockValue to * set a mock value for a particular sensor, and Sensor::ResetMockValue or * Sensor::ResetAllMocks to reset one or all stored mock values. * * Composite Sensors: * Some sensors may be implemented as composite sensors, ie sensors that depend on other * sensors to determine their reading. For example, the throttle pedal may have a pair of * potentiometers that provide redundancy for the pedal's position. Each one may be its * own sensor, then with one "master" sensors that combines the results of the others, and * provides validation of whether the readings agree. * * @date September 12, 2019 * @author Matthew Kennedy, (c) 2019 */ #pragma once #include "sensor_type.h" #include struct SensorResult { const bool Valid; const float Value; }; // Fwd declare - nobody outside of Sensor.cpp needs to see inside this type struct SensorRegistryEntry; class Sensor { public: // Register this sensor in the sensor registry. // Returns true if registration succeeded, or false if // another sensor of the same type is already registered. // The return value should not be ignored: no error handling/reporting is // done internally! [[nodiscard]] bool Register(); // Remove all sensors from the sensor registry - tread carefully if you use this outside of a unit test static void resetRegistry(); /* * Static helper for sensor lookup */ static const Sensor *getSensorOfType(SensorType type); /* * Get a reading from the specified sensor. */ static SensorResult get(SensorType type); /* * Mock a value for a particular sensor. */ static void setMockValue(SensorType type, float value); /* * Mock a value for a particular sensor. */ static void setMockValue(int type, float value); /* * Reset mock for a particular sensor. */ static void resetMockValue(SensorType type); /* * Reset mocking for all sensors. */ static void resetAllMocks(); protected: // Protected constructor - only subclasses call this explicit Sensor(SensorType type) : m_type(type) {} private: // Retrieve the current reading from the sensor. // // Override this in a particular sensor's implementation. As reading sensors is in many hot paths, // it is unwise to synchronously read the sensor or do anything otherwise costly here. At the most, // this should be field lookup and simple math. virtual SensorResult get() const = 0; SensorType m_type; // Get this sensor's index in the list constexpr size_t getIndex() { return getIndex(m_type); } // Get the index in the list for a sensor of particular type static constexpr size_t getIndex(SensorType type) { return static_cast(type); } /* * Static helper for sensor lookup */ static SensorRegistryEntry *getEntryForType(SensorType type); };