523 lines
16 KiB
C++
523 lines
16 KiB
C++
/**
|
|
* @file trigger_structure.cpp
|
|
*
|
|
* @date Jan 20, 2014
|
|
* @author Andrey Belomutskiy, (c) 2012-2018
|
|
*
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "main.h"
|
|
#include "trigger_structure.h"
|
|
#include "error_handling.h"
|
|
#include "trigger_decoder.h"
|
|
#include "engine_math.h"
|
|
#include "trigger_universal.h"
|
|
#if EFI_SENSOR_CHART || defined(__DOXYGEN__)
|
|
#include "sensor_chart.h"
|
|
#endif /* EFI_SENSOR_CHART */
|
|
|
|
EXTERN_ENGINE;
|
|
|
|
trigger_shape_helper::trigger_shape_helper() {
|
|
memset(&pinStates, 0, sizeof(pinStates));
|
|
for (int i = 0; i < TRIGGER_CHANNEL_COUNT; i++) {
|
|
waves[i].init(pinStates[i]);
|
|
}
|
|
}
|
|
|
|
TriggerShape::TriggerShape() :
|
|
wave(switchTimesBuffer, NULL) {
|
|
initialize(OM_NONE, false);
|
|
wave.waves = h.waves;
|
|
|
|
memset(triggerIndexByAngle, 0, sizeof(triggerIndexByAngle));
|
|
}
|
|
|
|
void TriggerShape::calculateTriggerSynchPoint(TriggerState *state DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
#if EFI_PROD_CODE || defined(__DOXYGEN__)
|
|
efiAssertVoid(getRemainingStack(chThdGetSelfX()) > 256, "calc s");
|
|
#endif
|
|
trigger_config_s const*triggerConfig = &engineConfiguration->trigger;
|
|
|
|
triggerShapeSynchPointIndex = findTriggerZeroEventIndex(state, this, triggerConfig PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
int length = getLength();
|
|
engine->engineCycleEventCount = length;
|
|
efiAssertVoid(length > 0, "shapeLength=0");
|
|
if (length >= PWM_PHASE_MAX_COUNT) {
|
|
warning(CUSTOM_ERR_TRIGGER_SHAPE_TOO_LONG, "Count above %d", length);
|
|
shapeDefinitionError = true;
|
|
return;
|
|
}
|
|
|
|
float firstAngle = getAngle(triggerShapeSynchPointIndex);
|
|
assertAngleRange(triggerShapeSynchPointIndex, "firstAngle");
|
|
|
|
int frontOnlyIndex = 0;
|
|
|
|
for (int eventIndex = 0; eventIndex < length; eventIndex++) {
|
|
if (eventIndex == 0) {
|
|
// explicit check for zero to avoid issues where logical zero is not exactly zero due to float nature
|
|
eventAngles[0] = 0;
|
|
// this value would be used in case of front-only
|
|
eventAngles[1] = 0;
|
|
frontOnlyIndexes[0] = 0;
|
|
} else {
|
|
assertAngleRange(triggerShapeSynchPointIndex, "triggerShapeSynchPointIndex");
|
|
int triggerDefinitionCoordinate = (triggerShapeSynchPointIndex + eventIndex) % engine->engineCycleEventCount;
|
|
efiAssertVoid(engine->engineCycleEventCount != 0, "zero engineCycleEventCount");
|
|
int triggerDefinitionIndex = triggerDefinitionCoordinate >= size ? triggerDefinitionCoordinate - size : triggerDefinitionCoordinate;
|
|
float angle = getAngle(triggerDefinitionCoordinate) - firstAngle;
|
|
efiAssertVoid(!cisnan(angle), "trgSyncNaN");
|
|
fixAngle(angle, "trgSync");
|
|
if (engineConfiguration->useOnlyRisingEdgeForTrigger) {
|
|
if (isFrontEvent[triggerDefinitionIndex]) {
|
|
frontOnlyIndex += 2;
|
|
eventAngles[frontOnlyIndex] = angle;
|
|
eventAngles[frontOnlyIndex + 1] = angle;
|
|
}
|
|
} else {
|
|
eventAngles[eventIndex] = angle;
|
|
}
|
|
|
|
frontOnlyIndexes[eventIndex] = frontOnlyIndex;
|
|
}
|
|
}
|
|
}
|
|
|
|
void TriggerShape::initialize(operation_mode_e operationMode, bool needSecondTriggerInput) {
|
|
isSynchronizationNeeded = true; // that's default value
|
|
this->needSecondTriggerInput = needSecondTriggerInput;
|
|
memset(dutyCycle, 0, sizeof(dutyCycle));
|
|
memset(eventAngles, 0, sizeof(eventAngles));
|
|
// memset(triggerIndexByAngle, 0, sizeof(triggerIndexByAngle));
|
|
setTriggerSynchronizationGap(2);
|
|
|
|
secondSyncRatioFrom = NAN; // NaN means do not use this ratio
|
|
secondSyncRatioTo = 100000;
|
|
thirdSyncRatioFrom = NAN; // NaN means do not use this ratio
|
|
thirdSyncRatioTo = 100000;
|
|
|
|
|
|
tdcPosition = 0;
|
|
shapeDefinitionError = useOnlyPrimaryForSync = false;
|
|
useRiseEdge = true;
|
|
gapBothDirections = false;
|
|
|
|
invertOnAdd = false;
|
|
|
|
this->operationMode = operationMode;
|
|
size = 0;
|
|
triggerShapeSynchPointIndex = 0;
|
|
memset(initialState, 0, sizeof(initialState));
|
|
memset(switchTimesBuffer, 0, sizeof(switchTimesBuffer));
|
|
memset(expectedEventCount, 0, sizeof(expectedEventCount));
|
|
wave.reset();
|
|
previousAngle = 0;
|
|
memset(frontOnlyIndexes, 0, sizeof(frontOnlyIndexes));
|
|
memset(isFrontEvent, 0, sizeof(isFrontEvent));
|
|
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
|
|
memset(&triggerSignals, 0, sizeof(triggerSignals));
|
|
#endif
|
|
}
|
|
|
|
int TriggerShape::getSize() const {
|
|
return size;
|
|
}
|
|
|
|
int TriggerShape::getTriggerShapeSynchPointIndex() {
|
|
return triggerShapeSynchPointIndex;
|
|
}
|
|
|
|
int multi_wave_s::getChannelState(int channelIndex, int phaseIndex) const {
|
|
return waves[channelIndex].pinStates[phaseIndex];
|
|
}
|
|
|
|
int multi_wave_s::waveIndertionAngle(float angle, int size) const {
|
|
for (int i = size - 1; i >= 0; i--) {
|
|
if (angle > switchTimes[i])
|
|
return i + 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int multi_wave_s::findAngleMatch(float angle, int size) const {
|
|
for (int i = 0; i < size; i++) {
|
|
if (isSameF(switchTimes[i], angle))
|
|
return i;
|
|
}
|
|
return EFI_ERROR_CODE;
|
|
}
|
|
|
|
void multi_wave_s::setSwitchTime(int index, float value) {
|
|
switchTimes[index] = value;
|
|
}
|
|
|
|
TriggerState::TriggerState() {
|
|
reset();
|
|
}
|
|
|
|
void TriggerState::reset() {
|
|
cycleCallback = NULL;
|
|
shaft_is_synchronized = false;
|
|
toothed_previous_time = 0;
|
|
toothed_previous_duration = 0;
|
|
durationBeforePrevious = 0;
|
|
thirdPreviousDuration = 0;
|
|
|
|
totalRevolutionCounter = 0;
|
|
totalTriggerErrorCounter = 0;
|
|
orderingErrorCounter = 0;
|
|
currentDuration = 0;
|
|
curSignal = SHAFT_PRIMARY_FALLING;
|
|
prevSignal = SHAFT_PRIMARY_FALLING;
|
|
startOfCycleNt = 0;
|
|
|
|
resetRunningCounters();
|
|
resetCurrentCycleState();
|
|
memset(expectedTotalTime, 0, sizeof(expectedTotalTime));
|
|
|
|
totalEventCountBase = 0;
|
|
isFirstEvent = true;
|
|
}
|
|
|
|
int TriggerState::getCurrentIndex() {
|
|
return currentCycle.current_index;
|
|
}
|
|
|
|
efitime_t TriggerState::getStartOfRevolutionIndex() {
|
|
return totalEventCountBase;
|
|
}
|
|
|
|
void TriggerState::resetRunningCounters() {
|
|
runningRevolutionCounter = 0;
|
|
runningTriggerErrorCounter = 0;
|
|
runningOrderingErrorCounter = 0;
|
|
}
|
|
|
|
void TriggerState::runtimeStatistics(efitime_t nowNt DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
// empty base implementation
|
|
}
|
|
|
|
TriggerStateWithRunningStatistics::TriggerStateWithRunningStatistics() {
|
|
instantRpm = 0;
|
|
}
|
|
|
|
float TriggerStateWithRunningStatistics::calculateInstantRpm(int *prevIndex, efitime_t nowNt DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
int current_index = currentCycle.current_index; // local copy so that noone changes the value on us
|
|
/**
|
|
* Here we calculate RPM based on last 90 degrees
|
|
*/
|
|
angle_t currentAngle = TRIGGER_SHAPE(eventAngles[current_index]);
|
|
// todo: make this '90' depend on cylinder count or trigger shape?
|
|
angle_t previousAngle = currentAngle - 90;
|
|
fixAngle(previousAngle, "prevAngle");
|
|
// todo: prevIndex should be pre-calculated
|
|
*prevIndex = TRIGGER_SHAPE(triggerIndexByAngle[(int)previousAngle]);
|
|
|
|
// now let's get precise angle for that event
|
|
angle_t prevIndexAngle = TRIGGER_SHAPE(eventAngles[*prevIndex]);
|
|
uint32_t time = nowNt - timeOfLastEvent[*prevIndex];
|
|
angle_t angleDiff = currentAngle - prevIndexAngle;
|
|
// todo: angle diff should be pre-calculated
|
|
fixAngle(angleDiff, "angleDiff");
|
|
|
|
float instantRpm = (60000000.0 / 360 * US_TO_NT_MULTIPLIER) * angleDiff / time;
|
|
instantRpmValue[current_index] = instantRpm;
|
|
timeOfLastEvent[current_index] = nowNt;
|
|
|
|
return instantRpm;
|
|
}
|
|
|
|
void TriggerStateWithRunningStatistics::runtimeStatistics(efitime_t nowNt DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
if (engineConfiguration->debugMode == DBG_INSTANT_RPM) {
|
|
int prevIndex;
|
|
instantRpm = calculateInstantRpm(&prevIndex, nowNt PASS_ENGINE_PARAMETER_SUFFIX);
|
|
}
|
|
if (ENGINE(sensorChartMode) == SC_RPM_ACCEL || ENGINE(sensorChartMode) == SC_DETAILED_RPM) {
|
|
int prevIndex;
|
|
instantRpm = calculateInstantRpm(&prevIndex, nowNt PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
#if EFI_SENSOR_CHART || defined(__DOXYGEN__)
|
|
angle_t currentAngle = TRIGGER_SHAPE(eventAngles[currentCycle.current_index]);
|
|
if (boardConfiguration->sensorChartMode == SC_DETAILED_RPM) {
|
|
scAddData(currentAngle, instantRpm);
|
|
} else {
|
|
scAddData(currentAngle, instantRpm / instantRpmValue[prevIndex]);
|
|
}
|
|
#endif /* EFI_SENSOR_CHART */
|
|
}
|
|
}
|
|
|
|
efitime_t TriggerState::getTotalEventCounter() {
|
|
return totalEventCountBase + currentCycle.current_index;
|
|
}
|
|
|
|
int TriggerState::getTotalRevolutionCounter() {
|
|
return totalRevolutionCounter;
|
|
}
|
|
|
|
|
|
void TriggerState::intTotalEventCounter() {
|
|
totalRevolutionCounter++;
|
|
}
|
|
|
|
bool TriggerState::isEvenRevolution() {
|
|
return totalRevolutionCounter & 1;
|
|
}
|
|
|
|
void TriggerState::resetCurrentCycleState() {
|
|
memset(currentCycle.eventCount, 0, sizeof(currentCycle.eventCount));
|
|
memset(currentCycle.timeOfPreviousEventNt, 0, sizeof(currentCycle.timeOfPreviousEventNt));
|
|
memset(currentCycle.totalTimeNt, 0, sizeof(currentCycle.totalTimeNt));
|
|
currentCycle.current_index = 0;
|
|
}
|
|
|
|
/**
|
|
* physical primary trigger duration
|
|
*/
|
|
angle_t TriggerShape::getCycleDuration() const {
|
|
switch (operationMode) {
|
|
case FOUR_STROKE_SYMMETRICAL_CRANK_SENSOR:
|
|
return 180;
|
|
case FOUR_STROKE_CRANK_SENSOR:
|
|
case TWO_STROKE:
|
|
return 360;
|
|
default:
|
|
return 720;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Trigger event count equals engine cycle event count if we have a cam sensor.
|
|
* Two trigger cycles make one engine cycle in case of a four stroke engine If we only have a cranksensor.
|
|
*/
|
|
uint32_t TriggerShape::getLength() const {
|
|
/**
|
|
* 4 for FOUR_STROKE_SYMMETRICAL_CRANK_SENSOR
|
|
* 2 for FOUR_STROKE_CRANK_SENSOR
|
|
* 1 otherwise
|
|
*/
|
|
int multiplier = getEngineCycle(operationMode) / getCycleDuration();
|
|
return multiplier * getSize();
|
|
}
|
|
|
|
angle_t TriggerShape::getAngle(int index) const {
|
|
// todo: why is this check here? looks like the code below could be used universally
|
|
if (operationMode == FOUR_STROKE_CAM_SENSOR) {
|
|
return getSwitchAngle(index);
|
|
}
|
|
/**
|
|
* FOUR_STROKE_CRANK_SENSOR magic:
|
|
* We have two crank shaft revolutions for each engine cycle
|
|
* See also trigger_central.cpp
|
|
* See also getEngineCycleEventCount()
|
|
*/
|
|
efiAssert(size != 0, "shapeSize=0", NAN);
|
|
int crankCycle = index / size;
|
|
int remainder = index % size;
|
|
|
|
return getCycleDuration() * crankCycle + getSwitchAngle(remainder);
|
|
}
|
|
|
|
void TriggerShape::addEvent2(angle_t angle, trigger_wheel_e const waveIndex, trigger_value_e const stateParam, float filterLeft, float filterRight DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
if (angle > filterLeft && angle < filterRight)
|
|
addEvent2(angle, waveIndex, stateParam PASS_ENGINE_PARAMETER_SUFFIX);
|
|
}
|
|
|
|
operation_mode_e TriggerShape::getOperationMode() {
|
|
return operationMode;
|
|
}
|
|
|
|
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
|
|
extern bool printTriggerDebug;
|
|
#endif
|
|
|
|
void TriggerShape::addEvent2(angle_t angle, trigger_wheel_e const waveIndex, trigger_value_e const stateParam DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
efiAssertVoid(operationMode != OM_NONE, "operationMode not set");
|
|
|
|
efiAssertVoid(waveIndex!= T_SECONDARY || needSecondTriggerInput, "secondary needed or not?");
|
|
|
|
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
|
|
if (printTriggerDebug) {
|
|
printf("addEvent2 %.2f i=%d r/f=%d\r\n", angle, waveIndex, stateParam);
|
|
}
|
|
#endif
|
|
|
|
trigger_value_e state;
|
|
if (invertOnAdd) {
|
|
state = (stateParam == TV_FALL) ? TV_RISE : TV_FALL;
|
|
} else {
|
|
state = stateParam;
|
|
}
|
|
|
|
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
|
|
int signal = waveIndex * 1000 + stateParam;
|
|
triggerSignals[size] = signal;
|
|
#endif
|
|
|
|
float engineCycle = getEngineCycle(operationMode);
|
|
|
|
/**
|
|
* While '720' value works perfectly it has not much sense for crank sensor-only scenario.
|
|
* todo: accept angle as a value in the 0..1 range?
|
|
*/
|
|
angle /= engineCycle;
|
|
|
|
if (!engineConfiguration->useOnlyRisingEdgeForTrigger || stateParam == TV_RISE) {
|
|
expectedEventCount[waveIndex]++;
|
|
}
|
|
|
|
efiAssertVoid(angle > 0, "angle should be positive");
|
|
if (size > 0) {
|
|
if (angle <= previousAngle) {
|
|
warning(CUSTOM_ERR_TRG_ANGLE_ORDER, "invalid angle order: %.2f and %.2f, size=%d", angle, previousAngle, size);
|
|
shapeDefinitionError = true;
|
|
return;
|
|
}
|
|
}
|
|
previousAngle = angle;
|
|
if (size == 0) {
|
|
size = 1;
|
|
for (int i = 0; i < PWM_PHASE_MAX_WAVE_PER_PWM; i++) {
|
|
single_wave_s *wave = &this->wave.waves[i];
|
|
|
|
if (wave->pinStates == NULL) {
|
|
warning(CUSTOM_ERR_STATE_NULL, "wave pinStates is NULL");
|
|
shapeDefinitionError = true;
|
|
return;
|
|
}
|
|
wave->pinStates[0] = initialState[i];
|
|
}
|
|
|
|
isFrontEvent[0] = TV_RISE == stateParam;
|
|
wave.setSwitchTime(0, angle);
|
|
wave.waves[waveIndex].pinStates[0] = state;
|
|
return;
|
|
}
|
|
|
|
int exactMatch = wave.findAngleMatch(angle, size);
|
|
if (exactMatch != EFI_ERROR_CODE) {
|
|
warning(CUSTOM_ERR_SAME_ANGLE, "same angle: not supported");
|
|
shapeDefinitionError = true;
|
|
return;
|
|
}
|
|
|
|
int index = wave.waveIndertionAngle(angle, size);
|
|
|
|
// shifting existing data
|
|
// todo: does this logic actually work? I think it does not!
|
|
for (int i = size - 1; i >= index; i--) {
|
|
for (int j = 0; j < PWM_PHASE_MAX_WAVE_PER_PWM; j++) {
|
|
wave.waves[j].pinStates[i + 1] = wave.getChannelState(j, index);
|
|
}
|
|
wave.setSwitchTime(i + 1, wave.getSwitchTime(i));
|
|
}
|
|
|
|
isFrontEvent[index] = TV_RISE == stateParam;
|
|
|
|
if (index != size) {
|
|
firmwareError(ERROR_TRIGGER_DRAMA, "are we ever here?");
|
|
}
|
|
|
|
// int index = size;
|
|
size++;
|
|
|
|
for (int i = 0; i < PWM_PHASE_MAX_WAVE_PER_PWM; i++) {
|
|
wave.waves[i].pinStates[index] = wave.getChannelState(i, index - 1);
|
|
}
|
|
wave.setSwitchTime(index, angle);
|
|
wave.waves[waveIndex].pinStates[index] = state;
|
|
}
|
|
|
|
angle_t TriggerShape::getSwitchAngle(int index) const {
|
|
return getCycleDuration() * wave.getSwitchTime(index);
|
|
}
|
|
|
|
void multi_wave_s::checkSwitchTimes(int size) {
|
|
checkSwitchTimes2(size, switchTimes);
|
|
}
|
|
|
|
void setVwConfiguration(TriggerShape *s DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
efiAssertVoid(s != NULL, "TriggerShape is NULL");
|
|
operation_mode_e operationMode = FOUR_STROKE_CRANK_SENSOR;
|
|
|
|
s->initialize(operationMode, false);
|
|
|
|
s->isSynchronizationNeeded = true;
|
|
|
|
|
|
int totalTeethCount = 60;
|
|
int skippedCount = 2;
|
|
|
|
float engineCycle = getEngineCycle(operationMode);
|
|
float toothWidth = 0.5;
|
|
|
|
addSkippedToothTriggerEvents(T_PRIMARY, s, 60, 2, toothWidth, 0, engineCycle,
|
|
NO_LEFT_FILTER, 690 PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
float angleDown = engineCycle / totalTeethCount * (totalTeethCount - skippedCount - 1 + (1 - toothWidth) );
|
|
s->addEvent2(0 + angleDown + 12, T_PRIMARY, TV_RISE, NO_LEFT_FILTER, NO_RIGHT_FILTER PASS_ENGINE_PARAMETER_SUFFIX);
|
|
s->addEvent2(0 + engineCycle, T_PRIMARY, TV_FALL, NO_LEFT_FILTER, NO_RIGHT_FILTER PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
s->setTriggerSynchronizationGap2(1.6, 4);
|
|
}
|
|
|
|
void setToothedWheelConfiguration(TriggerShape *s, int total, int skipped,
|
|
operation_mode_e operationMode DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
#if EFI_ENGINE_CONTROL || defined(__DOXYGEN__)
|
|
|
|
s->useRiseEdge = true;
|
|
|
|
initializeSkippedToothTriggerShapeExt(s, total, skipped,
|
|
operationMode PASS_ENGINE_PARAMETER_SUFFIX);
|
|
#endif
|
|
}
|
|
|
|
void TriggerShape::setTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
|
|
isSynchronizationNeeded = true;
|
|
this->syncRatioFrom = syncRatioFrom;
|
|
this->syncRatioTo = syncRatioTo;
|
|
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
|
|
if (printTriggerDebug) {
|
|
printf("setTriggerSynchronizationGap2 %.2f %.2f\r\n", syncRatioFrom, syncRatioTo);
|
|
}
|
|
#endif /* EFI_UNIT_TEST */
|
|
}
|
|
|
|
void TriggerShape::setTriggerSynchronizationGap(float syncRatio) {
|
|
setTriggerSynchronizationGap2(syncRatio * 0.75f, syncRatio * 1.25f);
|
|
}
|
|
|
|
void TriggerShape::setSecondTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
|
|
isSynchronizationNeeded = true;
|
|
this->secondSyncRatioFrom = syncRatioFrom;
|
|
this->secondSyncRatioTo = syncRatioTo;
|
|
}
|
|
|
|
void TriggerShape::setThirdTriggerSynchronizationGap(float syncRatio) {
|
|
setThirdTriggerSynchronizationGap2(syncRatio * 0.75f, syncRatio * 1.25f);
|
|
}
|
|
|
|
void TriggerShape::setThirdTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
|
|
isSynchronizationNeeded = true;
|
|
this->thirdSyncRatioFrom = syncRatioFrom;
|
|
this->thirdSyncRatioTo = syncRatioTo;
|
|
}
|
|
|
|
void TriggerShape::setSecondTriggerSynchronizationGap(float syncRatio) {
|
|
setSecondTriggerSynchronizationGap2(syncRatio * 0.75f, syncRatio * 1.25f);
|
|
}
|