2024-06-20 06:22:47 -07:00
|
|
|
/**
|
|
|
|
* @file adc_inputs_onchip.cpp
|
|
|
|
* @brief Low level ADC code
|
|
|
|
*
|
2024-07-06 20:15:51 -07:00
|
|
|
* rusEfi uses two ADC devices on the same 16 pins at the moment. Two ADC devices are used in order to distinguish between
|
2024-06-20 06:22:47 -07:00
|
|
|
* fast and slow devices. The idea is that but only having few channels in 'fast' mode we can sample those faster?
|
|
|
|
*
|
|
|
|
* At the moment rusEfi does not allow to have more than 16 ADC channels combined. At the moment there is no flexibility to use
|
|
|
|
* any ADC pins, only the hardcoded choice of 16 pins.
|
|
|
|
*
|
|
|
|
* Slow ADC group is used for IAT, CLT, AFR, VBATT etc - this one is currently sampled at 500Hz
|
|
|
|
*
|
|
|
|
* Fast ADC group is used for MAP, MAF HIP - this one is currently sampled at 10KHz
|
|
|
|
* We need frequent MAP for map_averaging.cpp
|
|
|
|
*
|
|
|
|
* 10KHz equals one measurement every 3.6 degrees at 6000 RPM
|
|
|
|
*
|
|
|
|
* @date Jan 14, 2013
|
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "pch.h"
|
|
|
|
|
|
|
|
#if HAL_USE_ADC
|
|
|
|
|
|
|
|
#include "adc_subscription.h"
|
|
|
|
#include "AdcDevice.h"
|
|
|
|
#include "mpu_util.h"
|
|
|
|
#include "periodic_thread_controller.h"
|
|
|
|
#include "protected_gpio.h"
|
|
|
|
|
|
|
|
// Board voltage, with divider coefficient accounted for
|
|
|
|
float getVoltageDivided(const char *msg, adc_channel_e hwChannel) {
|
|
|
|
return getVoltage(msg, hwChannel) * getAnalogInputDividerCoefficient(hwChannel);
|
|
|
|
}
|
|
|
|
|
2024-07-06 19:46:41 -07:00
|
|
|
float BOARD_WEAK boardAdjustVoltage(float voltage, adc_channel_e hwChannel) {
|
|
|
|
// a hack useful when we do not trust voltage just after board EN was turned on. is this just hiding electrical design flaws?
|
|
|
|
return voltage;
|
|
|
|
}
|
|
|
|
|
2024-06-20 06:22:47 -07:00
|
|
|
// voltage in MCU universe, from zero to VDD
|
|
|
|
float getVoltage(const char *msg, adc_channel_e hwChannel) {
|
2024-07-06 19:46:41 -07:00
|
|
|
float voltage = adcToVolts(getAdcValue(msg, hwChannel));
|
|
|
|
return boardAdjustVoltage(voltage, hwChannel);
|
2024-06-20 06:22:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#if EFI_USE_FAST_ADC
|
|
|
|
|
2024-06-25 05:04:11 -07:00
|
|
|
// is there a reason to have this configurable at runtime?
|
|
|
|
#ifndef ADC_FAST_DEVICE
|
|
|
|
#define ADC_FAST_DEVICE ADCD2
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Depth of the conversion buffer, channels are sampled X times each
|
2024-06-20 06:22:47 -07:00
|
|
|
#ifndef ADC_BUF_DEPTH_FAST
|
2024-06-25 05:04:11 -07:00
|
|
|
#define ADC_BUF_DEPTH_FAST 4
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// See https://github.com/rusefi/rusefi/issues/976 for discussion on this value
|
|
|
|
#ifndef ADC_SAMPLING_FAST
|
|
|
|
#define ADC_SAMPLING_FAST ADC_SAMPLE_28
|
2024-06-20 06:22:47 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
AdcDevice::AdcDevice(ADCDriver *p_adcp, ADCConversionGroup* p_hwConfig, volatile adcsample_t *p_buf, size_t p_depth) {
|
|
|
|
adcp = p_adcp;
|
|
|
|
depth = p_depth;
|
|
|
|
hwConfig = p_hwConfig;
|
|
|
|
samples = p_buf;
|
|
|
|
|
|
|
|
hwConfig->sqr1 = 0;
|
|
|
|
hwConfig->sqr2 = 0;
|
|
|
|
hwConfig->sqr3 = 0;
|
|
|
|
#if ADC_MAX_CHANNELS_COUNT > 16
|
|
|
|
hwConfig->sqr4 = 0;
|
|
|
|
hwConfig->sqr5 = 0;
|
|
|
|
#endif /* ADC_MAX_CHANNELS_COUNT */
|
|
|
|
memset(internalAdcIndexByHardwareIndex, 0xFF, sizeof(internalAdcIndexByHardwareIndex));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fastAdcDoneCB(ADCDriver *adcp);
|
|
|
|
static void fastAdcErrorCB(ADCDriver *, adcerror_t err);
|
|
|
|
|
|
|
|
static ADCConversionGroup adcgrpcfgFast = {
|
|
|
|
.circular = FALSE,
|
|
|
|
.num_channels = 0,
|
|
|
|
.end_cb = fastAdcDoneCB,
|
|
|
|
.error_cb = fastAdcErrorCB,
|
|
|
|
/* HW dependent part.*/
|
|
|
|
.cr1 = 0,
|
|
|
|
.cr2 = ADC_CR2_SWSTART,
|
|
|
|
/**
|
|
|
|
* here we configure all possible channels for fast mode. Some channels would not actually
|
|
|
|
* be used hopefully that's fine to configure all possible channels.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
// sample times for channels 10...18
|
|
|
|
.smpr1 =
|
|
|
|
ADC_SMPR1_SMP_AN10(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR1_SMP_AN11(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR1_SMP_AN12(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR1_SMP_AN13(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR1_SMP_AN14(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR1_SMP_AN15(ADC_SAMPLING_FAST),
|
|
|
|
// In this field must be specified the sample times for channels 0...9
|
|
|
|
.smpr2 =
|
|
|
|
ADC_SMPR2_SMP_AN0(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN1(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN2(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN3(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN4(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN5(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN6(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN7(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN8(ADC_SAMPLING_FAST) |
|
|
|
|
ADC_SMPR2_SMP_AN9(ADC_SAMPLING_FAST),
|
|
|
|
.htr = 0,
|
|
|
|
.ltr = 0,
|
|
|
|
.sqr1 = 0, // Conversion group sequence 13...16 + sequence length
|
|
|
|
.sqr2 = 0, // Conversion group sequence 7...12
|
|
|
|
.sqr3 = 0, // Conversion group sequence 1...6
|
|
|
|
#if ADC_MAX_CHANNELS_COUNT > 16
|
|
|
|
.sqr4 = 0, // Conversion group sequence 19...24
|
|
|
|
.sqr5 = 0 // Conversion group sequence 25...30
|
|
|
|
#endif /* ADC_MAX_CHANNELS_COUNT */
|
|
|
|
};
|
|
|
|
|
|
|
|
static volatile NO_CACHE adcsample_t fastAdcSampleBuf[ADC_BUF_DEPTH_FAST * ADC_MAX_CHANNELS_COUNT];
|
|
|
|
|
|
|
|
AdcDevice fastAdc(&ADC_FAST_DEVICE, &adcgrpcfgFast, fastAdcSampleBuf, ADC_BUF_DEPTH_FAST);
|
|
|
|
|
|
|
|
static void fastAdcDoneCB(ADCDriver *adcp) {
|
|
|
|
// State may not be complete if we get a callback for "half done"
|
|
|
|
if (adcp->state == ADC_COMPLETE) {
|
|
|
|
fastAdc.conversionCount++;
|
|
|
|
onFastAdcComplete(adcp->samples);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static volatile adcerror_t fastAdcLastError;
|
|
|
|
|
2024-07-11 06:37:06 -07:00
|
|
|
static void fastAdcErrorCB(ADCDriver *, adcerror_t err) {
|
2024-06-20 06:22:47 -07:00
|
|
|
fastAdcLastError = err;
|
2024-07-11 06:37:06 -07:00
|
|
|
engine->outputChannels.fastAdcErrorCallbackCount++;
|
2024-06-20 06:22:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void fastAdcTrigger(GPTDriver*) {
|
|
|
|
#if EFI_INTERNAL_ADC
|
|
|
|
/*
|
|
|
|
* Starts an asynchronous ADC conversion operation, the conversion
|
|
|
|
* will be executed in parallel to the current PWM cycle and will
|
|
|
|
* terminate before the next PWM cycle.
|
|
|
|
*/
|
|
|
|
fastAdc.startConversionI();
|
|
|
|
#endif /* EFI_INTERNAL_ADC */
|
|
|
|
}
|
|
|
|
|
|
|
|
static GPTConfig fast_adc_config = {
|
|
|
|
.frequency = GPT_FREQ_FAST,
|
|
|
|
.callback = fastAdcTrigger,
|
|
|
|
.cr2 = 0,
|
|
|
|
.dier = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
int AdcDevice::size() const {
|
|
|
|
return channelCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AdcDevice::init(void) {
|
|
|
|
hwConfig->num_channels = size();
|
|
|
|
/* driver does this internally */
|
|
|
|
//hwConfig->sqr1 += ADC_SQR1_NUM_CH(size());
|
|
|
|
|
|
|
|
gptStart(EFI_INTERNAL_FAST_ADC_GPT, &fast_adc_config);
|
|
|
|
gptStartContinuous(EFI_INTERNAL_FAST_ADC_GPT, GPT_PERIOD_FAST);
|
|
|
|
}
|
|
|
|
|
|
|
|
int AdcDevice::enableChannel(adc_channel_e hwChannel) {
|
|
|
|
if ((channelCount + 1) >= ADC_MAX_CHANNELS_COUNT) {
|
|
|
|
criticalError("Too many ADC channels configured");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int logicChannel = channelCount++;
|
|
|
|
|
|
|
|
/* TODO: following is correct for STM32 ADC1/2.
|
|
|
|
* ADC3 has another input to gpio mapping
|
|
|
|
* and should be handled separately */
|
|
|
|
size_t channelAdcIndex = hwChannel - EFI_ADC_0;
|
|
|
|
|
|
|
|
internalAdcIndexByHardwareIndex[hwChannel] = logicChannel;
|
|
|
|
if (logicChannel < 6) {
|
|
|
|
hwConfig->sqr3 |= channelAdcIndex << (5 * logicChannel);
|
|
|
|
} else if (logicChannel < 12) {
|
|
|
|
hwConfig->sqr2 |= channelAdcIndex << (5 * (logicChannel - 6));
|
|
|
|
} else if (logicChannel < 18) {
|
|
|
|
hwConfig->sqr1 |= channelAdcIndex << (5 * (logicChannel - 12));
|
|
|
|
}
|
|
|
|
#if ADC_MAX_CHANNELS_COUNT > 16
|
|
|
|
else if (logicChannel < 24) {
|
|
|
|
hwConfig->sqr4 |= channelAdcIndex << (5 * (logicChannel - 18));
|
|
|
|
}
|
|
|
|
else if (logicChannel < 30) {
|
|
|
|
hwConfig->sqr5 |= channelAdcIndex << (5 * (logicChannel - 24));
|
|
|
|
}
|
|
|
|
#endif /* ADC_MAX_CHANNELS_COUNT */
|
|
|
|
|
|
|
|
return channelAdcIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AdcDevice::startConversionI()
|
|
|
|
{
|
|
|
|
chSysLockFromISR();
|
|
|
|
if ((ADC_FAST_DEVICE.state != ADC_READY) &&
|
|
|
|
(ADC_FAST_DEVICE.state != ADC_COMPLETE) &&
|
|
|
|
(ADC_FAST_DEVICE.state != ADC_ERROR)) {
|
|
|
|
engine->outputChannels.fastAdcErrorsCount++;
|
|
|
|
// todo: when? why? criticalError("ADC fast not ready?");
|
|
|
|
// see notes at https://github.com/rusefi/rusefi/issues/6399
|
|
|
|
} else {
|
|
|
|
/* drop volatile type qualifier - this is safe */
|
|
|
|
adcStartConversionI(adcp, hwConfig, (adcsample_t *)samples, depth);
|
|
|
|
}
|
|
|
|
chSysUnlockFromISR();
|
|
|
|
}
|
|
|
|
|
|
|
|
adcsample_t AdcDevice::getAvgAdcValue(adc_channel_e hwChannel) {
|
|
|
|
uint32_t result = 0;
|
|
|
|
int numChannels = size();
|
|
|
|
int index = fastAdc.internalAdcIndexByHardwareIndex[hwChannel];
|
|
|
|
|
|
|
|
for (size_t i = 0; i < depth; i++) {
|
|
|
|
adcsample_t sample = samples[index];
|
|
|
|
// if (sample > 0x1FFF) {
|
|
|
|
// // 12bit ADC expected right now, make this configurable one day
|
|
|
|
// criticalError("fast ADC unexpected sample %d", sample);
|
|
|
|
// } else
|
|
|
|
if (sample > ADC_MAX_VALUE) {
|
|
|
|
if (!engineConfiguration->skipADC12bitAssert) {
|
2024-07-10 15:03:28 -07:00
|
|
|
criticalError("ADC unexpected sample %d at %ld uptime. Please report and use skipADC12bitAssert to disable.",
|
|
|
|
sample,
|
|
|
|
(uint32_t)getTimeNowS());
|
2024-06-20 06:22:47 -07:00
|
|
|
}
|
|
|
|
engine->outputChannels.unexpectedAdcSample = sample;
|
|
|
|
// sad hack which works around https://github.com/rusefi/rusefi/issues/6376 which we do not understand
|
|
|
|
sample = sample & ADC_MAX_VALUE;
|
|
|
|
engine->outputChannels.adc13bitCounter++;
|
|
|
|
}
|
|
|
|
result += sample;
|
|
|
|
index += numChannels;
|
|
|
|
}
|
|
|
|
|
|
|
|
// this truncation is guaranteed to not be lossy - the average can't be larger than adcsample_t
|
|
|
|
return static_cast<adcsample_t>(result / depth);
|
|
|
|
}
|
|
|
|
|
|
|
|
adc_channel_e AdcDevice::getAdcChannelByInternalIndex(int hwChannel) const {
|
|
|
|
for (size_t idx = EFI_ADC_0; idx < EFI_ADC_TOTAL_CHANNELS; idx++) {
|
|
|
|
if (internalAdcIndexByHardwareIndex[idx] == hwChannel) {
|
|
|
|
return (adc_channel_e)idx;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return EFI_ADC_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
AdcToken AdcDevice::getAdcChannelToken(adc_channel_e hwChannel) {
|
|
|
|
return fastAdc.internalAdcIndexByHardwareIndex[hwChannel];
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // EFI_USE_FAST_ADC
|
|
|
|
|
|
|
|
#endif
|