2020-03-31 20:21:05 -07:00
|
|
|
/**
|
|
|
|
* @file can_sensor.h
|
|
|
|
*
|
|
|
|
* @date March 31, 2020
|
|
|
|
* @author Matthew Kennedy, (c) 2020
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "stored_value_sensor.h"
|
|
|
|
#include "scaled_channel.h"
|
|
|
|
#include "hal.h"
|
2020-08-31 19:47:33 -07:00
|
|
|
#include "can_msg_tx.h"
|
|
|
|
#include "obd2.h"
|
2020-09-06 20:39:25 -07:00
|
|
|
#include "can.h"
|
2020-03-31 20:21:05 -07:00
|
|
|
|
2020-08-29 11:36:23 -07:00
|
|
|
/**
|
|
|
|
* Sensor which reads it's value from CAN
|
|
|
|
*/
|
2020-03-31 20:21:05 -07:00
|
|
|
class CanSensorBase : public StoredValueSensor {
|
|
|
|
public:
|
|
|
|
CanSensorBase(uint32_t eid, SensorType type, efitick_t timeout)
|
|
|
|
: StoredValueSensor(type, timeout)
|
|
|
|
, m_eid(eid)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-08-31 01:39:27 -07:00
|
|
|
virtual CanSensorBase* request() {
|
|
|
|
return m_next;
|
|
|
|
}
|
|
|
|
|
2020-03-31 20:21:05 -07:00
|
|
|
void showInfo(Logging* logger, const char* sensorName) const override;
|
|
|
|
|
|
|
|
CanSensorBase* processFrame(const CANRxFrame& frame, efitick_t nowNt) {
|
|
|
|
if (frame.EID == m_eid) {
|
|
|
|
decodeFrame(frame, nowNt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_next;
|
|
|
|
}
|
|
|
|
|
2020-08-31 19:47:33 -07:00
|
|
|
uint32_t getEid() {
|
|
|
|
return m_eid;
|
|
|
|
}
|
|
|
|
|
2020-03-31 20:21:05 -07:00
|
|
|
void setNext(CanSensorBase* next) {
|
|
|
|
m_next = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual void decodeFrame(const CANRxFrame& frame, efitick_t nowNt) = 0;
|
2020-08-31 19:47:33 -07:00
|
|
|
CanSensorBase* m_next = nullptr;
|
2020-03-31 20:21:05 -07:00
|
|
|
|
|
|
|
private:
|
|
|
|
const uint32_t m_eid;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename TStorage, int TScale>
|
|
|
|
class CanSensor : public CanSensorBase {
|
|
|
|
public:
|
|
|
|
CanSensor(uint32_t eid, uint8_t offset, SensorType type, efitick_t timeout)
|
|
|
|
: CanSensorBase(eid, type, timeout)
|
|
|
|
, m_offset(offset)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void decodeFrame(const CANRxFrame& frame, efitick_t nowNt) override {
|
|
|
|
// Compute the location of our data within the frame
|
|
|
|
const uint8_t* dataLocation = &frame.data8[m_offset];
|
|
|
|
|
|
|
|
// Reinterpret as a scaled_channel - it already has the logic for decoding a scaled integer to a float
|
|
|
|
const auto scaler = reinterpret_cast<const scaled_channel<TStorage, TScale>*>(dataLocation);
|
|
|
|
|
|
|
|
// Actually do the conversion
|
|
|
|
float value = *scaler;
|
|
|
|
setValidValue(value, nowNt);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const uint8_t m_offset;
|
|
|
|
};
|
2020-08-31 01:39:27 -07:00
|
|
|
|
2020-09-06 20:39:25 -07:00
|
|
|
template <int Size, int Offset>
|
2020-08-31 19:47:33 -07:00
|
|
|
class ObdCanSensor: public CanSensorBase {
|
|
|
|
public:
|
2020-09-06 20:39:25 -07:00
|
|
|
ObdCanSensor(int PID, float Scale, SensorType type) :
|
2020-09-07 17:06:13 -07:00
|
|
|
CanSensorBase(OBD_TEST_RESPONSE, type, /* timeout, never expire */ 0) {
|
2020-09-06 19:36:31 -07:00
|
|
|
this->PID = PID;
|
2020-09-06 20:39:25 -07:00
|
|
|
this->Scale = Scale;
|
2020-08-31 19:47:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void decodeFrame(const CANRxFrame& frame, efitick_t nowNt) override {
|
2020-09-06 19:36:31 -07:00
|
|
|
if (frame.data8[2] != PID) {
|
|
|
|
return;
|
|
|
|
}
|
2020-08-31 19:47:33 -07:00
|
|
|
|
2020-09-06 20:19:32 -07:00
|
|
|
int iValue;
|
|
|
|
if (Size == 2) {
|
|
|
|
iValue = frame.data8[3] * 256 + frame.data8[4];
|
|
|
|
} else {
|
|
|
|
iValue = frame.data8[3];
|
|
|
|
}
|
2020-08-31 19:47:33 -07:00
|
|
|
|
2020-09-06 20:39:25 -07:00
|
|
|
float fValue = (1.0 * iValue / Scale) - Offset;
|
2020-09-06 20:19:32 -07:00
|
|
|
setValidValue(fValue, nowNt);
|
2020-08-31 19:47:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
CanSensorBase* request() override {
|
|
|
|
{
|
|
|
|
CanTxMessage msg(OBD_TEST_REQUEST);
|
|
|
|
msg[0] = _OBD_2;
|
|
|
|
msg[1] = OBD_CURRENT_DATA;
|
2020-09-06 20:19:32 -07:00
|
|
|
msg[2] = PID;
|
2020-08-31 19:47:33 -07:00
|
|
|
}
|
2020-09-07 13:25:00 -07:00
|
|
|
// let's sleep on write update after each OBD request, this would give read thread a chance to read response
|
|
|
|
// todo: smarter logic of all this with with semaphore not just sleep
|
|
|
|
chThdSleepMilliseconds(300);
|
2020-08-31 19:47:33 -07:00
|
|
|
return m_next;
|
|
|
|
}
|
2020-09-06 19:36:31 -07:00
|
|
|
|
|
|
|
int PID;
|
2020-09-06 20:39:25 -07:00
|
|
|
float Scale;
|
2020-08-31 01:39:27 -07:00
|
|
|
};
|
|
|
|
|