2021-08-03 19:05:01 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2019-09-21 11:33:38 -07:00
|
|
|
#include "adc_subscription.h"
|
2019-09-24 18:11:41 -07:00
|
|
|
|
2020-08-21 16:47:12 -07:00
|
|
|
#include "biquad.h"
|
2019-09-21 11:33:38 -07:00
|
|
|
|
2020-04-05 06:10:08 -07:00
|
|
|
#if EFI_UNIT_TEST
|
|
|
|
|
2021-08-14 05:41:27 -07:00
|
|
|
/*static*/ void AdcSubscription::SubscribeSensor(FunctionalSensor&, adc_channel_e, float, float) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/*static*/ void AdcSubscription::UnsubscribeSensor(FunctionalSensor&) {
|
2020-04-05 06:10:08 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
2019-09-21 11:33:38 -07:00
|
|
|
|
|
|
|
struct AdcSubscriptionEntry {
|
2019-09-25 04:26:56 -07:00
|
|
|
FunctionalSensor *Sensor;
|
2019-09-21 11:33:38 -07:00
|
|
|
float VoltsPerAdcVolt;
|
2021-08-26 23:32:31 -07:00
|
|
|
Biquad Filter;
|
2021-08-27 03:18:05 -07:00
|
|
|
adc_channel_e Channel;
|
2020-08-31 18:05:04 -07:00
|
|
|
bool HasUpdated = false;
|
2019-09-21 11:33:38 -07:00
|
|
|
};
|
|
|
|
|
2021-05-09 11:36:11 -07:00
|
|
|
static AdcSubscriptionEntry s_entries[16];
|
2019-09-21 11:33:38 -07:00
|
|
|
|
2021-08-14 05:41:27 -07:00
|
|
|
static AdcSubscriptionEntry* findEntry(FunctionalSensor* sensor) {
|
|
|
|
for (size_t i = 0; i < efi::size(s_entries); i++) {
|
|
|
|
if (s_entries[i].Sensor == sensor) {
|
|
|
|
return &s_entries[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static AdcSubscriptionEntry* findEntry() {
|
|
|
|
// Find an entry with no sensor set
|
|
|
|
return findEntry(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*static*/ void AdcSubscription::SubscribeSensor(FunctionalSensor &sensor,
|
2019-09-24 18:11:41 -07:00
|
|
|
adc_channel_e channel,
|
2020-08-21 16:47:12 -07:00
|
|
|
float lowpassCutoff,
|
2019-09-24 18:11:41 -07:00
|
|
|
float voltsPerAdcVolt /*= 0.0f*/) {
|
2019-09-21 11:33:38 -07:00
|
|
|
// Don't subscribe null channels
|
2021-01-05 13:02:20 -08:00
|
|
|
if (!isAdcChannelValid(channel)) {
|
2019-09-21 11:33:38 -07:00
|
|
|
return;
|
|
|
|
}
|
2019-09-24 18:11:41 -07:00
|
|
|
|
2021-05-09 11:36:11 -07:00
|
|
|
const char* name = sensor.getSensorName();
|
|
|
|
if (/*type-limited (int)setting < 0 || */(int)channel >= HW_MAX_ADC_INDEX) {
|
|
|
|
firmwareError(CUSTOM_INVALID_ADC, "Invalid ADC setting %s", name);
|
2019-09-21 11:33:38 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-08-14 05:41:27 -07:00
|
|
|
auto entry = findEntry();
|
|
|
|
|
|
|
|
// Ensure that a free entry was found
|
|
|
|
if (!entry) {
|
2021-05-09 11:36:11 -07:00
|
|
|
firmwareError(CUSTOM_INVALID_ADC, "too many ADC subscriptions");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-07-14 19:37:05 -07:00
|
|
|
#if EFI_PROD_CODE
|
2021-05-09 11:36:11 -07:00
|
|
|
// Enable the input pin
|
|
|
|
efiSetPadMode(name, getAdcChannelBrainPin(name, channel), PAL_MODE_INPUT_ANALOG);
|
2021-07-14 19:37:05 -07:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2021-05-09 11:36:11 -07:00
|
|
|
|
2019-09-21 11:33:38 -07:00
|
|
|
// if 0, default to the board's divider coefficient
|
|
|
|
if (voltsPerAdcVolt == 0) {
|
|
|
|
voltsPerAdcVolt = engineConfiguration->analogInputDividerCoefficient;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Populate the entry
|
2021-08-14 05:41:27 -07:00
|
|
|
entry->VoltsPerAdcVolt = voltsPerAdcVolt;
|
|
|
|
entry->Channel = channel;
|
|
|
|
entry->Filter.configureLowpass(SLOW_ADC_RATE, lowpassCutoff);
|
2021-08-24 13:41:16 -07:00
|
|
|
entry->HasUpdated = false;
|
2019-09-21 11:33:38 -07:00
|
|
|
|
2021-08-14 05:41:27 -07:00
|
|
|
// Set the sensor last - it's the field we use to determine whether this entry is in use
|
|
|
|
entry->Sensor = &sensor;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*static*/ void AdcSubscription::UnsubscribeSensor(FunctionalSensor& sensor) {
|
|
|
|
auto entry = findEntry(&sensor);
|
|
|
|
|
|
|
|
if (!entry) {
|
|
|
|
// This sensor wasn't configured, skip it
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE
|
|
|
|
// Release the pin
|
|
|
|
efiSetPadUnused(getAdcChannelBrainPin("adc unsubscribe", entry->Channel));
|
|
|
|
#endif // EFI_PROD_CODE
|
|
|
|
|
2022-10-20 06:33:27 -07:00
|
|
|
sensor.unregister();
|
|
|
|
|
2021-08-14 05:41:27 -07:00
|
|
|
// clear the sensor first to mark this entry not in use
|
|
|
|
entry->Sensor = nullptr;
|
|
|
|
|
|
|
|
entry->VoltsPerAdcVolt = 0;
|
|
|
|
entry->Channel = EFI_ADC_NONE;
|
2019-09-21 11:33:38 -07:00
|
|
|
}
|
|
|
|
|
2020-02-08 14:48:15 -08:00
|
|
|
void AdcSubscription::UpdateSubscribers(efitick_t nowNt) {
|
2019-10-11 17:43:21 -07:00
|
|
|
ScopePerf perf(PE::AdcSubscriptionUpdateSubscribers);
|
|
|
|
|
2021-08-14 05:41:27 -07:00
|
|
|
for (size_t i = 0; i < efi::size(s_entries); i++) {
|
2019-09-24 18:11:41 -07:00
|
|
|
auto &entry = s_entries[i];
|
2019-09-21 11:33:38 -07:00
|
|
|
|
2021-08-14 05:41:27 -07:00
|
|
|
if (!entry.Sensor) {
|
|
|
|
// Skip unconfigured entries
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-09-21 11:33:38 -07:00
|
|
|
float mcuVolts = getVoltage("sensor", entry.Channel);
|
|
|
|
float sensorVolts = mcuVolts * entry.VoltsPerAdcVolt;
|
|
|
|
|
2020-08-31 18:05:04 -07:00
|
|
|
// On the very first update, preload the filter as if we've been
|
|
|
|
// seeing this value for a long time. This prevents a slow ramp-up
|
|
|
|
// towards the correct value just after startup
|
|
|
|
if (!entry.HasUpdated) {
|
|
|
|
entry.Filter.cookSteadyState(sensorVolts);
|
|
|
|
entry.HasUpdated = true;
|
|
|
|
}
|
|
|
|
|
2020-08-21 16:47:12 -07:00
|
|
|
float filtered = entry.Filter.filter(sensorVolts);
|
|
|
|
|
|
|
|
entry.Sensor->postRawValue(filtered, nowNt);
|
2019-09-21 11:33:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-10 04:44:20 -07:00
|
|
|
#if EFI_PROD_CODE
|
|
|
|
void AdcSubscription::PrintInfo() {
|
|
|
|
for (size_t i = 0; i < efi::size(s_entries); i++) {
|
|
|
|
auto& entry = s_entries[i];
|
|
|
|
|
|
|
|
if (!entry.Sensor) {
|
|
|
|
// Skip unconfigured entries
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto name = entry.Sensor->getSensorName();
|
|
|
|
float mcuVolts = getVoltage("sensor", entry.Channel);
|
|
|
|
float sensorVolts = mcuVolts * entry.VoltsPerAdcVolt;
|
|
|
|
auto channel = entry.Channel;
|
|
|
|
|
|
|
|
char pinNameBuffer[16];
|
|
|
|
|
|
|
|
efiPrintf(
|
|
|
|
"%s ADC%d m=%d %s adc=%.2f/input=%.2fv/divider=%.2f",
|
|
|
|
name,
|
|
|
|
channel,
|
|
|
|
getAdcMode(channel),
|
|
|
|
getPinNameByAdcChannel(name, channel, pinNameBuffer),
|
|
|
|
mcuVolts, sensorVolts, entry.VoltsPerAdcVolt
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // EFI_PROD_CODE
|
|
|
|
|
2019-09-24 18:11:41 -07:00
|
|
|
#endif // !EFI_UNIT_TEST
|