custom-board-bundle-sample-.../firmware/controllers/sensors/sensor.h

176 lines
5.2 KiB
C++

/**
* @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<SensorType::MyNewSensor>
*
* 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 <cstddef>
struct SensorResult {
const bool Valid;
const float Value;
// Implicit conversion operator to bool, so you can do things like if (myResult) { ... }
constexpr explicit operator bool() const {
return Valid;
}
// Easy default value handling
constexpr float value_or(float valueIfInvalid) const {
return Valid ? Value : valueIfInvalid;
}
};
// Fwd declare - nobody outside of Sensor.cpp needs to see inside this type
struct SensorRegistryEntry;
class Logging;
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();
// Print information about this sensor
virtual void showInfo(Logging* logger, const char* sensorName) const = 0;
// Print information about all sensors
static void showAllSensorInfo(Logging* logger);
// 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);
/*
* Get a raw (unconverted) value from the sensor, if available.
*/
static float getRaw(SensorType type);
/*
* Query whether there is a sensor of a particular type currently registered.
*/
static bool hasSensor(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();
/*
* Get a friendly name for the sensor.
* For example, CLT, IAT, Throttle Position 2, etc.
*/
const char* getSensorName() { return getSensorName(m_type); }
protected:
// Protected constructor - only subclasses call this
explicit Sensor(SensorType type)
: m_type(type) {}
static const char* getSensorName(SensorType 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;
/*
* Get an unconverted value from the sensor, if available.
*/
virtual float getRaw() const {
return 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<size_t>(type);
}
/*
* Static helper for sensor lookup
*/
static SensorRegistryEntry *getEntryForType(SensorType type);
};