/* * @file vw_b6.cpp * * @date Dec 26, 2019 * @author Andrey Belomutskiy, (c) 2012-2020 */ #include "pch.h" #include "vw_b6.h" #include "custom_engine.h" #include "table_helper.h" #include "electronic_throttle_impl.h" #include "mre_meta.h" #include "proteus_meta.h" static void commonPassatB6() { setCrankOperationMode(); engineConfiguration->trigger.type = TT_TOOTHED_WHEEL_60_2; engineConfiguration->vvtMode[0] = VVT_BOSCH_QUICK_START; engineConfiguration->map.sensor.type = MT_BOSCH_2_5; engineConfiguration->etbIdleThrottleRange = 10; engineConfiguration->idlePidRpmDeadZone = 500; engineConfiguration->idleMode = IM_AUTO; engineConfiguration->specs.cylindersCount = 4; engineConfiguration->specs.firingOrder = FO_1_3_4_2; engineConfiguration->isPhaseSyncRequiredForIgnition = true; engineConfiguration->disableEtbWhenEngineStopped = true; for (int i = 4; i < MAX_CYLINDER_COUNT;i++) { engineConfiguration->injectionPins[i] = Gpio::Unassigned; engineConfiguration->ignitionPins[i] = Gpio::Unassigned; } // engineConfiguration->canNbcType = CAN_BUS_NBC_VAG; engineConfiguration->enableAemXSeries = true; engineConfiguration->afr.hwChannel = EFI_ADC_4; // Injectors flow 1214 cc/min at 100 bar pressure engineConfiguration->injector.flow = 1214; // Use high pressure sensor engineConfiguration->injectorPressureType = IPT_High; // Automatic compensation of injector flow based on rail pressure engineConfiguration->injectorCompensationMode = ICM_SensedRailPressure; // Reference rail pressure is 10 000 kPa = 100 bar engineConfiguration->fuelReferencePressure = 10000; //setting "flat" 0.2 ms injector's lag time setArrayValues(engineConfiguration->injector.battLagCorr, 0.2); strcpy(engineConfiguration->engineMake, ENGINE_MAKE_VAG); strcpy(engineConfiguration->engineCode, "BPY"); strcpy(engineConfiguration->vehicleName, "test"); engineConfiguration->throttlePedalUpVoltage = 0.36; engineConfiguration->throttlePedalWOTVoltage = 2.13; engineConfiguration->throttlePedalSecondaryUpVoltage = 0.73; engineConfiguration->throttlePedalSecondaryWOTVoltage = 4.30; engineConfiguration->invertCamVVTSignal = true; engineConfiguration->vvtCamSensorUseRise = true; /** * PSS-140 */ // todo: calibration engineConfiguration->highPressureFuel.v1 = 0.5; /* volts */; engineConfiguration->highPressureFuel.value1 = 0; engineConfiguration->highPressureFuel.v2 = 4.5; /* volts */; engineConfiguration->highPressureFuel.value2 = BAR2KPA(140); engineConfiguration->lowPressureFuel.v1 = 0.5; /* volts */; engineConfiguration->lowPressureFuel.value1 = PSI2KPA(0); engineConfiguration->lowPressureFuel.v2 = 4.5; /* volts */; // todo: what's the proper calibration of this Bosch sensor? is it really 200psi? engineConfiguration->lowPressureFuel.value2 = PSI2KPA(200); gppwm_channel *lowPressureFuelPumpControl = &engineConfiguration->gppwm[1]; strcpy(engineConfiguration->gpPwmNote[1], "LPFP"); lowPressureFuelPumpControl->pwmFrequency = 20; lowPressureFuelPumpControl->loadAxis = GPPWM_FuelLoad; lowPressureFuelPumpControl->dutyIfError = 50; setTable(lowPressureFuelPumpControl->table, (uint8_t)50); gppwm_channel *coolantControl = &engineConfiguration->gppwm[0]; strcpy(engineConfiguration->gpPwmNote[0], "Rad Fan"); coolantControl->loadAxis = GPPWM_Clt; coolantControl->pwmFrequency = 25; coolantControl->loadAxis = GPPWM_FuelLoad; // Volkswage wants 10% for fan to be OFF, between pull-up and low side control we need to invert that value // todo system lua for duty driven by CLT? (3, Gpio::E0, "0.15 90 coolant 120 min max 90 - 30 / 0.8 * +", 25); int value = 100 - 10; coolantControl->dutyIfError = value; setTable(coolantControl->table, (uint8_t)value); // for now I just want to stop radiator whine // todo: enable cooling! /* for (int load = 0; load < GPPWM_LOAD_COUNT; load++) { for (int r = 0; r < GPPWM_RPM_COUNT; r++) { engineConfiguration->gppwm[0].table[load][r] = value; } } */ engineConfiguration->hpfpCamLobes = 3; engineConfiguration->hpfpPumpVolume = 0.290; engineConfiguration->hpfpMinAngle = 10; engineConfiguration->hpfpActivationAngle = 30; engineConfiguration->hpfpTargetDecay = 2000; engineConfiguration->hpfpPidP = 0.01; engineConfiguration->hpfpPidI = 0.0003; engineConfiguration->hpfpPeakPos = 10; setTable(config->veTable, 55); setBoschVAGETB(); // random number just to take position away from zero engineConfiguration->vvtOffsets[0] = 180; // https://rusefi.com/forum/viewtopic.php?p=38235#p38235 engineConfiguration->injector.flow = 1200; engineConfiguration->idle.solenoidPin = Gpio::Unassigned; engineConfiguration->fanPin = Gpio::Unassigned; engineConfiguration->useETBforIdleControl = true; engineConfiguration->injectionMode = IM_SEQUENTIAL; engineConfiguration->crankingInjectionMode = IM_SEQUENTIAL; } /** * set engine_type 39 */ void setProteusVwPassatB6() { #if HW_PROTEUS commonPassatB6(); engineConfiguration->triggerInputPins[0] = PROTEUS_VR_1; engineConfiguration->camInputs[0] = PROTEUS_DIGITAL_2; engineConfiguration->lowPressureFuel.hwChannel = PROTEUS_IN_ANALOG_VOLT_5; engineConfiguration->highPressureFuel.hwChannel = PROTEUS_IN_ANALOG_VOLT_4; gppwm_channel *coolantControl = &engineConfiguration->gppwm[0]; coolantControl->pin = PROTEUS_LS_5; engineConfiguration->mainRelayPin = PROTEUS_LS_6; gppwm_channel *lowPressureFuelPumpControl = &engineConfiguration->gppwm[1]; lowPressureFuelPumpControl->pin = PROTEUS_LS_7; //engineConfiguration->boostControlPin = PROTEUS_LS_8; engineConfiguration->vvtPins[0] = PROTEUS_LS_9; engineConfiguration->hpfpValvePin = PROTEUS_LS_15; engineConfiguration->tps1_2AdcChannel = PROTEUS_IN_TPS1_2; engineConfiguration->throttlePedalPositionAdcChannel = PROTEUS_IN_ANALOG_VOLT_9; engineConfiguration->throttlePedalPositionSecondAdcChannel = PROTEUS_IN_PPS2; strncpy(config->luaScript, R"( AIRBAG = 0x050 GRA = 0x388 TCU_1 = 0x440 TCU_2 = 0x540 BRAKE_2 = 0x5A0 MOTOR_1 = 0x280 MOTOR_3 = 0x380 MOTOR_INFO = 0x580 MOTOR_5 = 0x480 MOTOR_6 = 0x488 MOTOR_7 = 0x588 canRxAdd(AIRBAG) canRxAdd(TCU_1) canRxAdd(TCU_2) canRxAdd(BRAKE_2) fuelCounter = 0 function setBitRange(data, totalBitIndex, bitWidth, value) local byteIndex = totalBitIndex >> 3 local bitInByteIndex = totalBitIndex - byteIndex * 8 if (bitInByteIndex + bitWidth > 8) then bitsToHandleNow = 8 - bitInByteIndex setBitRange(data, totalBitIndex + bitsToHandleNow, bitWidth - bitsToHandleNow, value >> bitsToHandleNow) bitWidth = bitsToHandleNow end mask = (1 << bitWidth) - 1 data[1 + byteIndex] = data[1 + byteIndex] & (~(mask << bitInByteIndex)) maskedValue = value & mask shiftedValue = maskedValue << bitInByteIndex data[1 + byteIndex] = data[1 + byteIndex] | shiftedValue end function setTwoBytes(data, offset, value) data[offset + 1] = value % 255 data[offset + 2] = (value >> 8) % 255 end shallSleep = Timer.new() -- we want to turn on with hardware switch while ignition key is off hadIgnitionEvent = false function onCanRx(bus, id, dlc, data) if id == AIRBAG then -- looks like we have ignition key do not sleep! shallSleep : reset() hadIgnitionEvent = true else print('got CAN id=' ..id ..' dlc=' ..dlc) end end function setTwoBytes(data, offset, value) value = math.floor(value) data[offset + 2] = value >> 8 data[offset + 1] = value & 0xff end function xorChecksum(data, targetIndex) local index = 1 local result = 0 while data[index] ~= nil do if index ~= targetIndex then result = result ~ data[index] end index = index + 1 end data[targetIndex] = result return result end function getBitRange(data, bitIndex, bitWidth) byteIndex = bitIndex >> 3 shift = bitIndex - byteIndex * 8 value = data[1 + byteIndex] if (shift + bitWidth > 8) then value = value + data[2 + byteIndex] * 256 end mask = (1 << bitWidth) - 1 return (value >> shift) & mask end canMotor1 = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } canMotorInfo = { 0x00, 0x00, 0x00, 0x14, 0x1C, 0x93, 0x48, 0x14 } canMotor3 = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } motor5Data = { 0x1C, 0x08, 0xF3, 0x55, 0x19, 0x00, 0x00, 0xAD } canMotor6 = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } canMotor7 = { 0x1A, 0x66, 0x7E, 0x00, 0x00, 0x00, 0x00, 0x00 } setTickRate(100) everySecondTimer = Timer.new() canMotorInfoCounter = 0 counter = 0 function onTick() counter = (counter + 1) % 16 rpm = getSensor("RPM") or 0 clt = getSensor("CLT") or 0 iat = getSensor("IAT") or 0 tps = getSensor("TPS1") or 0 vbat = getSensor("BatteryVoltage") or 0 fakeTorque = interpolate(0, 6, 100, 60, tps) engineTorque = fakeTorque innerTorqWithoutExt = fakeTorque torqueLoss = 10 requestedTorque = fakeTorque canMotor1[2] = engineTorque / 0.39 canMotor1[5] = innerTorqWithoutExt / 0.4 canMotor1[6] = tps / 0.4 canMotor1[7] = torqueLoss / 0.39 canMotor1[8] = requestedTorque / 0.39 setTwoBytes(canMotor1, 2, 4 * rpm) txCan(1, MOTOR_1, 0, canMotor1) desired_wheel_torque = fakeTorque canMotor3[2] = (iat + 48) / 0.75 canMotor3[3] = tps / 0.4 canMotor3[5] = 0x20 setBitRange(canMotor3, 24, 12, math.floor(desired_wheel_torque / 0.39)) canMotor3[8] = tps / 0.4 txCan(1, MOTOR_3, 0, canMotor3) setBitRange(motor5Data, 5, 9, fuelCounter) xorChecksum(motor5Data, 8) txCan(1, MOTOR_5, 0, motor5Data) txCan(1, MOTOR_7, 0, canMotor7) local timeToTurnOff = shallSleep : getElapsedSeconds() > 2 local connectedToUsb = vbat < 4 if hadIgnitionEvent and timeToTurnOff then -- looks like ignition key was removed mcu_standby() end if everySecondTimer : getElapsedSeconds() > 1 then everySecondTimer : reset() fuelCounter = fuelCounter + 20 canMotorInfoCounter = (canMotorInfoCounter + 1) % 8 canMotorInfo[1] = 0x90 + (canMotorInfoCounter * 2) txCan(1, MOTOR_INFO, 0, canMotorInfo) end end )", efi::size(config->luaScript)); #endif } /** * set engine_type 62 * VW_B6 * has to be microRusEFI 0.5.2 */ void setMreVwPassatB6() { #if HW_MICRO_RUSEFI commonPassatB6(); engineConfiguration->tps1_2AdcChannel = MRE_IN_ANALOG_VOLT_9; // EFI_ADC_7: "31 - AN volt 3" - PA7 engineConfiguration->throttlePedalPositionAdcChannel = MRE_IN_ANALOG_VOLT_3; // 36 - AN volt 8 engineConfiguration->throttlePedalPositionSecondAdcChannel = MRE_IN_ANALOG_VOLT_8; // "26 - AN volt 2" engineConfiguration->highPressureFuel.hwChannel = MRE_IN_ANALOG_VOLT_2; // "19 - AN volt 4" engineConfiguration->lowPressureFuel.hwChannel = EFI_ADC_12; engineConfiguration->isSdCardEnabled = false; engineConfiguration->mc33816spiDevice = SPI_DEVICE_3; // RED engineConfiguration->spi3mosiPin = Gpio::C12; // YELLOW engineConfiguration->spi3misoPin = Gpio::C11; // BROWN engineConfiguration->spi3sckPin = Gpio::C10; engineConfiguration->sdCardCsPin = Gpio::Unassigned; engineConfiguration->is_enabled_spi_3 = true; // J8 orange engineConfiguration->mc33816_cs = Gpio::B8; // J8 Grey engineConfiguration->mc33816_rstb = Gpio::A15; // J8 Dark BLUE engineConfiguration->mc33816_driven = Gpio::B9; // J9 violet engineConfiguration->mc33816_flag0 = Gpio::C13; // J10 Dark BLUE engineConfiguration->injectionPins[0] = Gpio::E6; // J11 green engineConfiguration->injectionPins[1] = Gpio::E5; // J18 grey engineConfiguration->injectionPins[2] = Gpio::B7; // J6 white engineConfiguration->injectionPins[3] = Gpio::E0; gppwm_channel *lowPressureFuelPumpControl = &engineConfiguration->gppwm[1]; // "42 - Injector 4", somehow GP4 did not work? not enough current? not happy with diode? lowPressureFuelPumpControl->pin = Gpio::TLE8888_PIN_4; gppwm_channel *coolantControl = &engineConfiguration->gppwm[0]; coolantControl->pin = Gpio::TLE8888_PIN_5; // "3 - Lowside 2" // "7 - Lowside 1" //engineConfiguration->hpfpValvePin = MRE_LS_1; engineConfiguration->disablePrimaryUart = true; engineConfiguration->hpfpValvePin = Gpio::B10; // AUX J13 #endif /* HW_MICRO_RUSEFI */ }