2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2020-08-28 18:13:50 -07:00
|
|
|
#include "biquad.h"
|
|
|
|
#include "thread_controller.h"
|
2021-04-04 15:13:21 -07:00
|
|
|
#include "knock_logic.h"
|
2020-08-28 18:13:50 -07:00
|
|
|
#include "software_knock.h"
|
2021-07-06 18:44:59 -07:00
|
|
|
#include "peak_detect.h"
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
#if EFI_SOFTWARE_KNOCK
|
|
|
|
|
|
|
|
#include "knock_config.h"
|
2021-03-25 15:12:17 -07:00
|
|
|
#include "ch.hpp"
|
2020-08-28 18:13:50 -07:00
|
|
|
|
2021-07-06 18:44:59 -07:00
|
|
|
static NO_CACHE adcsample_t sampleBuffer[2000];
|
|
|
|
static int8_t currentCylinderIndex = 0;
|
|
|
|
static efitick_t lastKnockSampleTime = 0;
|
|
|
|
static Biquad knockFilter;
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
static volatile bool knockIsSampling = false;
|
|
|
|
static volatile bool knockNeedsProcess = false;
|
|
|
|
static volatile size_t sampleCount = 0;
|
|
|
|
|
2021-03-25 15:12:17 -07:00
|
|
|
chibios_rt::BinarySemaphore knockSem(/* taken =*/ true);
|
2020-08-28 18:13:50 -07:00
|
|
|
|
2021-01-19 12:20:35 -08:00
|
|
|
static void completionCallback(ADCDriver* adcp) {
|
2020-08-28 18:13:50 -07:00
|
|
|
palClearPad(GPIOD, 2);
|
|
|
|
|
|
|
|
if (adcp->state == ADC_COMPLETE) {
|
|
|
|
knockNeedsProcess = true;
|
|
|
|
|
|
|
|
// Notify the processing thread that it's time to process this sample
|
|
|
|
chSysLockFromISR();
|
2021-03-25 15:12:17 -07:00
|
|
|
knockSem.signalI();
|
2020-08-28 18:13:50 -07:00
|
|
|
chSysUnlockFromISR();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-23 06:43:16 -08:00
|
|
|
static void errorCallback(ADCDriver*, adcerror_t) {
|
2020-08-28 18:13:50 -07:00
|
|
|
}
|
|
|
|
|
2020-10-03 22:54:29 -07:00
|
|
|
static const uint32_t smpr1 =
|
|
|
|
ADC_SMPR1_SMP_AN10(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR1_SMP_AN11(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR1_SMP_AN12(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR1_SMP_AN13(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR1_SMP_AN14(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR1_SMP_AN15(KNOCK_SAMPLE_TIME);
|
|
|
|
|
|
|
|
static const uint32_t smpr2 =
|
|
|
|
ADC_SMPR2_SMP_AN0(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN1(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN2(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN3(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN4(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN5(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN6(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN7(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN8(KNOCK_SAMPLE_TIME) |
|
|
|
|
ADC_SMPR2_SMP_AN9(KNOCK_SAMPLE_TIME);
|
|
|
|
|
2020-09-02 04:19:02 -07:00
|
|
|
static const ADCConversionGroup adcConvGroupCh1 = { FALSE, 1, &completionCallback, &errorCallback,
|
2020-08-28 18:13:50 -07:00
|
|
|
0,
|
|
|
|
ADC_CR2_SWSTART,
|
2020-10-03 22:54:29 -07:00
|
|
|
// sample times for channels 10...18
|
|
|
|
smpr1,
|
|
|
|
// sample times for channels 0...9
|
|
|
|
smpr2,
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
0, // htr
|
|
|
|
0, // ltr
|
|
|
|
|
|
|
|
0, // sqr1
|
|
|
|
0, // sqr2
|
|
|
|
ADC_SQR3_SQ1_N(KNOCK_ADC_CH1)
|
|
|
|
};
|
|
|
|
|
2020-09-02 04:19:02 -07:00
|
|
|
// Not all boards have a second channel - configure it if it exists
|
|
|
|
#if KNOCK_HAS_CH2
|
|
|
|
static const ADCConversionGroup adcConvGroupCh2 = { FALSE, 1, &completionCallback, &errorCallback,
|
|
|
|
0,
|
|
|
|
ADC_CR2_SWSTART,
|
2020-10-03 22:54:29 -07:00
|
|
|
// sample times for channels 10...18
|
|
|
|
smpr1,
|
|
|
|
// sample times for channels 0...9
|
|
|
|
smpr2,
|
2020-09-02 04:19:02 -07:00
|
|
|
|
|
|
|
0, // htr
|
|
|
|
0, // ltr
|
|
|
|
|
|
|
|
0, // sqr1
|
|
|
|
0, // sqr2
|
|
|
|
ADC_SQR3_SQ1_N(KNOCK_ADC_CH2)
|
|
|
|
};
|
2020-11-23 06:43:16 -08:00
|
|
|
#endif // KNOCK_HAS_CH2
|
|
|
|
|
2020-09-02 04:19:02 -07:00
|
|
|
const ADCConversionGroup* getConversionGroup(uint8_t cylinderIndex) {
|
|
|
|
#if KNOCK_HAS_CH2
|
2021-04-04 15:13:21 -07:00
|
|
|
if (getCylinderKnockBank(cylinderIndex)) {
|
2020-09-02 04:19:02 -07:00
|
|
|
return &adcConvGroupCh2;
|
|
|
|
}
|
2021-01-11 05:49:20 -08:00
|
|
|
#else
|
|
|
|
(void)cylinderIndex;
|
2020-09-02 04:19:02 -07:00
|
|
|
#endif // KNOCK_HAS_CH2
|
|
|
|
|
|
|
|
return &adcConvGroupCh1;
|
|
|
|
}
|
|
|
|
|
2020-08-28 18:13:50 -07:00
|
|
|
void startKnockSampling(uint8_t cylinderIndex) {
|
|
|
|
if (!CONFIG(enableSoftwareKnock)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-08-31 18:05:33 -07:00
|
|
|
if (!engine->rpmCalculator.isRunning()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-08-28 18:13:50 -07:00
|
|
|
// Cancel if ADC isn't ready
|
|
|
|
if (!((KNOCK_ADC.state == ADC_READY) ||
|
|
|
|
(KNOCK_ADC.state == ADC_COMPLETE) ||
|
|
|
|
(KNOCK_ADC.state == ADC_ERROR))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there's pending processing, skip this event
|
|
|
|
if (knockNeedsProcess) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-08-09 14:07:38 -07:00
|
|
|
// Sample for XX degrees
|
|
|
|
float samplingSeconds = ENGINE(rpmCalculator).oneDegreeUs * CONFIG(knockSamplingDuration) / US_PER_SECOND_F;
|
2020-08-28 18:13:50 -07:00
|
|
|
constexpr int sampleRate = KNOCK_SAMPLE_RATE;
|
|
|
|
sampleCount = 0xFFFFFFFE & static_cast<size_t>(clampF(100, samplingSeconds * sampleRate, efi::size(sampleBuffer)));
|
|
|
|
|
2020-09-02 04:19:02 -07:00
|
|
|
// Select the appropriate conversion group - it will differ depending on which sensor this cylinder should listen on
|
|
|
|
auto conversionGroup = getConversionGroup(cylinderIndex);
|
|
|
|
|
|
|
|
// Stash the current cylinder's index so we can store the result appropriately
|
2020-08-31 18:05:33 -07:00
|
|
|
currentCylinderIndex = cylinderIndex;
|
2020-09-02 04:19:02 -07:00
|
|
|
|
|
|
|
adcStartConversionI(&KNOCK_ADC, conversionGroup, sampleBuffer, sampleCount);
|
2021-07-06 18:44:59 -07:00
|
|
|
lastKnockSampleTime = getTimeNowNt();
|
2020-08-28 18:13:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
class KnockThread : public ThreadController<256> {
|
|
|
|
public:
|
2021-02-28 04:30:45 -08:00
|
|
|
KnockThread() : ThreadController("knock", PRIO_KNOCK_PROCESS) {}
|
2020-08-28 18:13:50 -07:00
|
|
|
void ThreadTask() override;
|
|
|
|
};
|
|
|
|
|
2020-09-02 04:19:02 -07:00
|
|
|
static KnockThread kt;
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
void initSoftwareKnock() {
|
|
|
|
if (CONFIG(enableSoftwareKnock)) {
|
|
|
|
knockFilter.configureBandpass(KNOCK_SAMPLE_RATE, 1000 * CONFIG(knockBandCustom), 3);
|
|
|
|
adcStart(&KNOCK_ADC, nullptr);
|
|
|
|
|
|
|
|
efiSetPadMode("knock ch1", KNOCK_PIN_CH1, PAL_MODE_INPUT_ANALOG);
|
2020-09-22 00:48:17 -07:00
|
|
|
#if KNOCK_HAS_CH2
|
2020-08-28 18:13:50 -07:00
|
|
|
efiSetPadMode("knock ch2", KNOCK_PIN_CH2, PAL_MODE_INPUT_ANALOG);
|
2020-09-22 00:48:17 -07:00
|
|
|
#endif
|
2020-08-28 18:13:50 -07:00
|
|
|
kt.Start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-06 18:44:59 -07:00
|
|
|
using PD = PeakDetect<float, MS2NT(100)>;
|
|
|
|
static PD peakDetectors[12];
|
|
|
|
static PD allCylinderPeakDetector;
|
|
|
|
|
2020-08-28 18:13:50 -07:00
|
|
|
void processLastKnockEvent() {
|
|
|
|
if (!knockNeedsProcess) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
float sumSq = 0;
|
|
|
|
|
2021-01-14 19:34:06 -08:00
|
|
|
// todo: reduce magic constants. engineConfiguration->adcVcc?
|
2020-08-28 18:13:50 -07:00
|
|
|
constexpr float ratio = 3.3f / 4095.0f;
|
|
|
|
|
|
|
|
size_t localCount = sampleCount;
|
|
|
|
|
|
|
|
// Prepare the steady state at vcc/2 so that there isn't a step
|
|
|
|
// when samples begin
|
2021-01-14 19:34:06 -08:00
|
|
|
// todo: reduce magic constants. engineConfiguration->adcVcc?
|
2020-08-28 18:13:50 -07:00
|
|
|
knockFilter.cookSteadyState(3.3f / 2);
|
|
|
|
|
|
|
|
// Compute the sum of squares
|
2021-01-14 19:34:06 -08:00
|
|
|
for (size_t i = 0; i < localCount; i++) {
|
2020-08-28 18:13:50 -07:00
|
|
|
float volts = ratio * sampleBuffer[i];
|
|
|
|
|
|
|
|
float filtered = knockFilter.filter(volts);
|
2021-08-09 14:07:38 -07:00
|
|
|
if (i == localCount - 1 && engineConfiguration->debugMode == DBG_KNOCK) {
|
|
|
|
tsOutputChannels.debugFloatField1 = volts;
|
|
|
|
tsOutputChannels.debugFloatField2 = filtered;
|
|
|
|
}
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
sumSq += filtered * filtered;
|
|
|
|
}
|
|
|
|
|
2021-07-06 18:44:59 -07:00
|
|
|
// take a local copy
|
|
|
|
auto lastKnockTime = lastKnockSampleTime;
|
|
|
|
|
|
|
|
// We're done with inspecting the buffer, another sample can be taken
|
|
|
|
knockNeedsProcess = false;
|
|
|
|
|
2020-08-28 18:13:50 -07:00
|
|
|
// mean of squares (not yet root)
|
|
|
|
float meanSquares = sumSq / localCount;
|
|
|
|
|
|
|
|
// RMS
|
|
|
|
float db = 10 * log10(meanSquares);
|
|
|
|
|
2021-07-06 18:44:59 -07:00
|
|
|
// clamp to reasonable range
|
|
|
|
db = clampF(-100, db, 100);
|
|
|
|
|
|
|
|
// Pass through peak detector
|
|
|
|
float cylPeak = peakDetectors[currentCylinderIndex].detect(db, lastKnockTime);
|
|
|
|
|
|
|
|
tsOutputChannels.knockLevels[currentCylinderIndex] = roundf(cylPeak);
|
|
|
|
tsOutputChannels.knockLevel = allCylinderPeakDetector.detect(db, lastKnockTime);
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void KnockThread::ThreadTask() {
|
2021-01-14 19:34:06 -08:00
|
|
|
while (1) {
|
2021-03-25 15:12:17 -07:00
|
|
|
knockSem.wait();
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
ScopePerf perf(PE::SoftwareKnockProcess);
|
|
|
|
processLastKnockEvent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // EFI_SOFTWARE_KNOCK
|