/** * @file engine_math.cpp * @brief * * @date Jul 13, 2013 * @author Andrey Belomutskiy, (c) 2012-2014 * * This file is part of rusEfi - see http://rusefi.com * * rusEfi is free software; you can redistribute it and/or modify it under the terms of * the GNU General Public License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * rusEfi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without * even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along with this program. * If not, see . */ #include "engine_math.h" #include "main.h" #include "engine_configuration.h" #include "interpolation.h" #include "allsensors.h" #include "io_pins.h" #include "OutputSignalList.h" #include "trigger_decoder.h" /* * default Volumetric Efficiency */ //float getDefaultVE(int rpm) { // if (rpm > 5000) // return interpolate(5000, 1.1, 8000, 1, rpm); // return interpolate(500, 0.5, 5000, 1.1, rpm); //} //#define K_AT_MIN_RPM_MIN_TPS 0.25 //#define K_AT_MIN_RPM_MAX_TPS 0.25 //#define K_AT_MAX_RPM_MIN_TPS 0.25 //#define K_AT_MAX_RPM_MAX_TPS 0.9 // //#define rpmMin 500 //#define rpmMax 8000 // //#define tpMin 0 //#define tpMax 100 // // http://rusefi.com/math/t_charge.html // / //float getTCharge(int rpm, int tps, float coolantTemp, float airTemp) { // float minRpmKcurrentTPS = interpolate(tpMin, K_AT_MIN_RPM_MIN_TPS, tpMax, // K_AT_MIN_RPM_MAX_TPS, tps); // float maxRpmKcurrentTPS = interpolate(tpMin, K_AT_MAX_RPM_MIN_TPS, tpMax, // K_AT_MAX_RPM_MAX_TPS, tps); // // float Tcharge_coff = interpolate(rpmMin, minRpmKcurrentTPS, rpmMax, // maxRpmKcurrentTPS, rpm); // // float Tcharge = coolantTemp * (1 - Tcharge_coff) + airTemp * Tcharge_coff; // // return Tcharge; //} #define MAX_STARTING_FUEL 15 #define MIN_STARTING_FUEL 8 /** * @return time needed to rotate crankshaft by one degree, in milliseconds. */ float getOneDegreeTimeMs(int rpm) { return 1000.0 * 60 / 360 / rpm; } /** * @return number of milliseconds in one crankshaft revolution */ float getCrankshaftRevolutionTimeMs(int rpm) { return 360 * getOneDegreeTimeMs(rpm); } /** * @brief Shifts angle into the [0..720) range * TODO: should be 'crankAngleRange' range? */ float fixAngle(float angle) { // I guess this implementation would be faster than 'angle % 720' while (angle < 0) angle += 720; while (angle > 720) angle -= 720; return angle; } /** * @brief Returns engine load according to selected engine_load_mode * */ float getEngineLoadT(engine_configuration_s *engineConfiguration) { switch (engineConfiguration->engineLoadMode) { case LM_MAF: return getMaf(); case LM_MAP: return getMap(); case LM_TPS: return getTPS(); case LM_SPEED_DENSITY: // TODO: real implementation return getMap(); default: firmwareError("Unexpected engine load parameter: %d", engineConfiguration->engineLoadMode); return -1; } } void setSingleCoilDwell(engine_configuration_s *engineConfiguration) { for (int i = 0; i < DWELL_CURVE_SIZE; i++) { engineConfiguration->sparkDwellBins[i] = 0; engineConfiguration->sparkDwell[i] = -1; } engineConfiguration->sparkDwellBins[5] = 1; engineConfiguration->sparkDwell[5] = 4; engineConfiguration->sparkDwellBins[6] = 4500; engineConfiguration->sparkDwell[6] = 4; engineConfiguration->sparkDwellBins[7] = 12500; engineConfiguration->sparkDwell[7] = 0; } int isCrankingRT(engine_configuration_s *engineConfiguration, int rpm) { return rpm > 0 && rpm < engineConfiguration->crankingSettings.crankingRpm; } OutputSignalList ignitionSignals; OutputSignalList injectonSignals; void initializeIgnitionActions(float baseAngle, engine_configuration_s *engineConfiguration, engine_configuration2_s *engineConfiguration2) { chDbgCheck(engineConfiguration->cylindersCount > 0, "cylindersCount"); ignitionSignals.clear(); EventHandlerConfiguration *config = &engineConfiguration2->engineEventConfiguration; resetEventList(&config->ignitionEvents); switch (engineConfiguration->ignitionMode) { case IM_ONE_COIL: for (int i = 0; i < engineConfiguration->cylindersCount; i++) { // todo: extract method float angle = baseAngle + 720.0 * i / engineConfiguration->cylindersCount; registerActuatorEventExt(engineConfiguration, &engineConfiguration2->triggerShape, &config->ignitionEvents, ignitionSignals.add(SPARKOUT_1_OUTPUT), angle); } break; case IM_WASTED_SPARK: for (int i = 0; i < engineConfiguration->cylindersCount; i++) { float angle = baseAngle + 720.0 * i / engineConfiguration->cylindersCount; int wastedIndex = i % (engineConfiguration->cylindersCount / 2); int id = (getCylinderId(engineConfiguration->firingOrder, wastedIndex) - 1); io_pin_e ioPin = (io_pin_e) (SPARKOUT_1_OUTPUT + id); registerActuatorEventExt(engineConfiguration, &engineConfiguration2->triggerShape, &config->ignitionEvents, ignitionSignals.add(ioPin), angle); } break; case IM_INDIVIDUAL_COILS: for (int i = 0; i < engineConfiguration->cylindersCount; i++) { float angle = baseAngle + 720.0 * i / engineConfiguration->cylindersCount; io_pin_e pin = (io_pin_e) ((int) SPARKOUT_1_OUTPUT + getCylinderId(engineConfiguration->firingOrder, i) - 1); registerActuatorEventExt(engineConfiguration, &engineConfiguration2->triggerShape, &config->ignitionEvents, ignitionSignals.add(pin), angle); } break; default: firmwareError("unsupported ignitionMode %d in initializeIgnitionActions()", engineConfiguration->ignitionMode); } } void addFuelEvents(engine_configuration_s const *e, engine_configuration2_s *engineConfiguration2, ActuatorEventList *list, injection_mode_e mode) { resetEventList(list); trigger_shape_s *s = &engineConfiguration2->triggerShape; float baseAngle = e->globalTriggerAngleOffset + e->injectionOffset; switch (mode) { case IM_SEQUENTIAL: for (int i = 0; i < e->cylindersCount; i++) { io_pin_e pin = (io_pin_e) ((int) INJECTOR_1_OUTPUT + getCylinderId(e->firingOrder, i) - 1); float angle = baseAngle + i * 720.0 / e->cylindersCount; registerActuatorEventExt(e, s, list, injectonSignals.add(pin), angle); } break; case IM_SIMULTANEOUS: for (int i = 0; i < e->cylindersCount; i++) { float angle = baseAngle + i * 720.0 / e->cylindersCount; for (int j = 0; j < e->cylindersCount; j++) { io_pin_e pin = (io_pin_e) ((int) INJECTOR_1_OUTPUT + j); registerActuatorEventExt(e, s, list, injectonSignals.add(pin), angle); } } break; case IM_BATCH: for (int i = 0; i < e->cylindersCount; i++) { io_pin_e pin = (io_pin_e) ((int) INJECTOR_1_OUTPUT + (i % 2)); float angle = baseAngle + i * 720.0 / e->cylindersCount; registerActuatorEventExt(e, s, list, injectonSignals.add(pin), angle); } break; default: firmwareError("Unexpected injection mode %d", mode); } } /** * @return Spark dwell time, in milliseconds. */ float getSparkDwellMsT(engine_configuration_s *engineConfiguration, int rpm) { if (isCrankingR(rpm)) { // technically this could be implemented via interpolate2d float angle = engineConfiguration->crankingChargeAngle; return getOneDegreeTimeMs(rpm) * angle; } return interpolate2d(rpm, engineConfiguration->sparkDwellBins, engineConfiguration->sparkDwell, DWELL_CURVE_SIZE); } void registerActuatorEventExt(engine_configuration_s const *engineConfiguration, trigger_shape_s * s, ActuatorEventList *list, OutputSignal *actuator, float angleOffset) { chDbgCheck(s->size > 0, "uninitialized trigger_shape_s"); angleOffset = fixAngle(angleOffset + engineConfiguration->globalTriggerAngleOffset); int triggerIndexOfZeroEvent = s->triggerShapeSynchPointIndex; // todo: migrate to crankAngleRange? float firstAngle = s->wave.getSwitchTime(triggerIndexOfZeroEvent) * 720; // let's find the last trigger angle which is less or equal to the desired angle int i; for (i = 0; i < s->size - 1; i++) { // todo: we need binary search here float angle = fixAngle(s->wave.getSwitchTime((triggerIndexOfZeroEvent + i + 1) % s->size) * 720 - firstAngle); if (angle > angleOffset) break; } // explicit check for zero to avoid issues where logical zero is not exactly zero due to float nature float angle = i == 0 ? 0 : fixAngle(s->wave.getSwitchTime((triggerIndexOfZeroEvent + i) % s->size) * 720 - firstAngle); chDbgCheck(angleOffset >= angle, "angle constraint violation in registerActuatorEventExt()"); registerActuatorEvent(list, i, actuator, angleOffset - angle); } //float getTriggerEventAngle(int triggerEventIndex) { // return 0; //} /** * there is some BS related to isnan in MinGW, so let's have all the issues in one place */ int cisnan(float f) { return *(((int*) (&f))) == 0x7FC00000; } static int order_1_THEN_3_THEN_4_THEN2[] = { 1, 3, 4, 2 }; static int order_1_THEN_5_THEN_3_THEN_6_THEN_2_THEN_4[] = { 1, 5, 3, 6, 2, 4 }; /** * @param index from zero to cylindersCount - 1 * @return cylinderId from one to cylindersCount */ int getCylinderId(firing_order_e firingOrder, int index) { switch (firingOrder) { case FO_ONE_CYLINDER: return 1; case FO_1_THEN_3_THEN_4_THEN2: return order_1_THEN_3_THEN_4_THEN2[index]; case FO_1_THEN_5_THEN_3_THEN_6_THEN_2_THEN_4: return order_1_THEN_5_THEN_3_THEN_6_THEN_2_THEN_4[index]; default: firmwareError("getCylinderId not supported for %d", firingOrder); } return -1; } void prepareOutputSignals(engine_configuration_s *engineConfiguration, engine_configuration2_s *engineConfiguration2) { // todo: move this reset into decoder engineConfiguration2->triggerShape.triggerShapeSynchPointIndex = findTriggerZeroEventIndex( &engineConfiguration2->triggerShape, &engineConfiguration->triggerConfig); injectonSignals.clear(); EventHandlerConfiguration *config = &engineConfiguration2->engineEventConfiguration; addFuelEvents(engineConfiguration, engineConfiguration2, &config->crankingInjectionEvents, engineConfiguration->crankingInjectionMode); addFuelEvents(engineConfiguration, engineConfiguration2, &config->injectionEvents, engineConfiguration->injectionMode); } void setTableBin(float array[], int size, float l, float r) { for (int i = 0; i < size; i++) array[i] = interpolate(0, l, size - 1, r, i); } void setFuelRpmBin(engine_configuration_s *engineConfiguration, float l, float r) { setTableBin(engineConfiguration->fuelRpmBins, FUEL_RPM_COUNT, l, r); } void setFuelLoadBin(engine_configuration_s *engineConfiguration, float l, float r) { setTableBin(engineConfiguration->fuelLoadBins, FUEL_LOAD_COUNT, l, r); } void setTimingRpmBin(engine_configuration_s *engineConfiguration, float l, float r) { setTableBin(engineConfiguration->ignitionRpmBins, IGN_RPM_COUNT, l, r); } void setTimingLoadBin(engine_configuration_s *engineConfiguration, float l, float r) { setTableBin(engineConfiguration->ignitionLoadBins, IGN_LOAD_COUNT, l, r); }