From ee539e3234b7002bec7431484d89e725f76bff9c Mon Sep 17 00:00:00 2001 From: peternewell Date: Sun, 2 Jun 2024 16:50:52 +1000 Subject: [PATCH 01/25] Initial addition of changes for oil burning steam locomotive. --- .../RollingStocks/MSTSLocomotive.cs | 9 + .../RollingStocks/MSTSSteamLocomotive.cs | 319 +++++++++++++----- 2 files changed, 235 insertions(+), 93 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs index e4c197ba5..706c14d80 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs @@ -478,6 +478,15 @@ public float OdometerM public float ThrottleIntervention = -1; public float DynamicBrakeIntervention = -1; + public enum SteamLocomotiveFuelTypes + { + Unknown, + Oil, + Wood, // not used at the moment + Coal, // defaults to coal + } + + public SteamLocomotiveFuelTypes SteamLocomotiveFuelType; public enum TractionMotorTypes { DC, diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 320855a87..c7dd80b16 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -225,7 +225,6 @@ public class MSTSSteamLocomotive : MSTSLocomotive float BoilerSteamDensityLBpFT3; // Steam Density based on current boiler pressure float BoilerWaterDensityLBpFT3; // Water Density based on current boiler pressure float BoilerWaterTempK; - public float FuelBurnRateSmoothedKGpS; float FuelFeedRateKGpS; float DesiredChange; // Amount of change to increase fire mass, clamped to range 0.0 - 1.0 public float CylinderSteamUsageLBpS; @@ -291,10 +290,14 @@ public class MSTSSteamLocomotive : MSTSLocomotive float PressureRatio = 0.001f; // Ratio to control burn rate - based upon boiler pressure float BurnRateRawKGpS; // Raw combustion (burn) rate float MaxCombustionRateKgpS; + SmoothedData FuelRateOil = new SmoothedData(3); // Oil Stoker is more responsive and only takes x seconds to fully react to changing needs. SmoothedData FuelRateStoker = new SmoothedData(15); // Stoker is more responsive and only takes x seconds to fully react to changing needs. - SmoothedData FuelRate = new SmoothedData(45); // Automatic fireman takes x seconds to fully react to changing needs. - SmoothedData BurnRateSmoothKGpS = new SmoothedData(150); // Changes in BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. - float FuelRateSmoothed = 0.0f; // Smoothed Fuel Rate + SmoothedData FuelRateCoal = new SmoothedData(45); // Automatic fireman takes x seconds to fully react to changing needs for coal firing. + SmoothedData CoalBurnRateSmoothKGpS = new SmoothedData(150); // Changes in BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. + SmoothedData OilBurnRateSmoothKGpS = new SmoothedData(3); // Changes in Oil BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. Oil faster then steam + float FuelFeedRateSmoothedKGpS = 0.0f; // Smoothed Fuel feedd Rate + public float FuelBurnRateSmoothedKGpS; // Smoothed fuel burning rate + float OilSpecificGravity = 0.9659f; // Assume a mid range of API for this value, say API = 15. int NumberofTractiveForceValues = 36; float[,] TractiveForceAverageN = new float[5, 37]; @@ -964,6 +967,19 @@ public override void Parse(string lowercasetoken, STFReader stf) STFException.TraceWarning(stf, "Assumed unknown engine type " + steamengineType); } break; + case "engine(steamlocomotivefueltype": + stf.MustMatch("("); + var steamLocomotiveFuelType = stf.ReadString(); + try + { + SteamLocomotiveFuelType = (SteamLocomotiveFuelTypes)Enum.Parse(typeof(SteamLocomotiveFuelTypes), steamLocomotiveFuelType); + } + catch + { + if (Simulator.Settings.VerboseConfigurationMessages) + STFException.TraceWarning(stf, "Assumed unknown engine type " + steamLocomotiveFuelType); + } + break; case "engine(ortssteamboilertype": stf.MustMatch("("); string typeString1 = stf.ReadString(); @@ -1057,6 +1073,7 @@ public override void Copy(MSTSWagon copy) SteamGearRatioHigh = locoCopy.SteamGearRatioHigh; MaxSteamGearPistonRateFtpM = locoCopy.MaxSteamGearPistonRateFtpM; SteamEngineType = locoCopy.SteamEngineType; + SteamLocomotiveFuelType = locoCopy.SteamLocomotiveFuelType; IsSaturated = locoCopy.IsSaturated; IsTenderRequired = locoCopy.IsTenderRequired; HasSuperheater = locoCopy.HasSuperheater; @@ -1130,7 +1147,7 @@ public override void Save(BinaryWriter outf) ControllerFactory.Save(LargeEjectorController, outf); outf.Write(FuelBurnRateSmoothedKGpS); outf.Write(BoilerHeatSmoothedBTU); - outf.Write(FuelRateSmoothed); + outf.Write(FuelFeedRateSmoothedKGpS); base.Save(outf); } @@ -1193,11 +1210,12 @@ public override void Restore(BinaryReader inf) ControllerFactory.Restore(SmallEjectorController, inf); ControllerFactory.Restore(LargeEjectorController, inf); FuelBurnRateSmoothedKGpS = inf.ReadSingle(); - BurnRateSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + CoalBurnRateSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + OilBurnRateSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); BoilerHeatSmoothedBTU = inf.ReadSingle(); BoilerHeatSmoothBTU.ForceSmoothValue(BoilerHeatSmoothedBTU); - FuelRateSmoothed = inf.ReadSingle(); - FuelRate.ForceSmoothValue(FuelRateSmoothed); + FuelFeedRateSmoothedKGpS = inf.ReadSingle(); + FuelRateCoal.ForceSmoothValue(FuelFeedRateSmoothedKGpS); base.Restore(inf); } @@ -1292,6 +1310,16 @@ public override void Initialize() } + // Type of fuel selected + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Unknown) + { + SteamLocomotiveFuelType = SteamLocomotiveFuelTypes.Coal; + + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceInformation("SteamLocomotiveFuelType set to Default value of {0}", SteamLocomotiveFuelType); + + } + // Calculate Grate Limit // Rule of thumb indicates that Grate limit occurs when the Boiler Efficiency is equal to 50% of the BE at zero firing rate // http://5at.co.uk/index.php/definitions/terrms-and-definitions/grate-limit.html @@ -3377,7 +3405,7 @@ private void UpdateFX(float elapsedClockSeconds) Variable2_Booster = Variable2; } - Variable3 = FuelRateSmoothed * 100; + Variable3 = FuelFeedRateSmoothedKGpS * 100; const int rotations = 2; const int fullLoop = 10 * rotations; @@ -3696,77 +3724,84 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) else // *********** AI Fireman ***************** { - if (PreviousTotalSteamUsageLBpS > TheoreticalMaxSteamOutputLBpS) - { - FiringSteamUsageRateLBpS = TheoreticalMaxSteamOutputLBpS; // hold usage rate if steam usage rate exceeds boiler max output - } - else +// if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) +// { +// BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG); +// } +// else { - FiringSteamUsageRateLBpS = PreviousTotalSteamUsageLBpS; // Current steam usage rate - } - - AIFiremanBurnFactorExceed = HeatRatio * PressureRatio; // Firing rate for AI fireman if firemass drops, and fireman needs to exceed normal capacity - AIFiremanBurnFactor = HeatRatio * PressureRatio * FullBoilerHeatRatio * MaxBoilerHeatRatio; // Firing rate factor under normal conditions - float AIFiremanStartingBurnFactor = 4.0f; + if (PreviousTotalSteamUsageLBpS > TheoreticalMaxSteamOutputLBpS) + { + FiringSteamUsageRateLBpS = TheoreticalMaxSteamOutputLBpS; // hold usage rate if steam usage rate exceeds boiler max output + } + else + { + FiringSteamUsageRateLBpS = PreviousTotalSteamUsageLBpS; // Current steam usage rate + } - if (ShovelAnyway && BoilerHeatBTU < MaxBoilerHeatBTU) // will force fire burn rate to increase even though boiler heat seems excessive - { - // burnrate will be the radiation loss @ rest & then related to heat usage as a factor of the maximum boiler output - // ignores total bolier heat to allow burn rate to increase if boiler heat usage is exceeding input - BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactorExceed; - } - else - { + AIFiremanBurnFactorExceed = HeatRatio * PressureRatio; // Firing rate for AI fireman if firemass drops, and fireman needs to exceed normal capacity + AIFiremanBurnFactor = HeatRatio * PressureRatio * FullBoilerHeatRatio * MaxBoilerHeatRatio; // Firing rate factor under normal conditions + float AIFiremanStartingBurnFactor = 4.0f; - if (BoilerHeatBTU > MaxBoilerHeatBTU && BoilerHeatBTU <= MaxBoilerSafetyPressHeatBTU && throttle > 0.1 && cutoff > 0.1 && BoilerHeatInBTUpS < PreviousBoilerHeatOutBTUpS && FullMaxPressBoilerHeat) - // This allows for situation where boiler heat has gone beyond safety valve heat, and is reducing, but steam will be required shortly so don't allow fire to go too low + if (ShovelAnyway && BoilerHeatBTU < MaxBoilerHeatBTU) // will force fire burn rate to increase even though boiler heat seems excessive { - // Burn Rate rate if Boiler Heat is too high - BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanStartingBurnFactor; // Calculate the amount of coal that should be burnt based upon heat used by boiler + // burnrate will be the radiation loss @ rest & then related to heat usage as a factor of the maximum boiler output + // ignores total bolier heat to allow burn rate to increase if boiler heat usage is exceeding input + BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactorExceed; } else { - // Burn Rate rate if Boiler Heat is "normal" - BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactor; - } - } - // AIFireOverride flag set to challenge driver if boiler AI fireman is overriden - ie steam safety valves will be set and blow if pressure is excessive - if (SetFireOn || SetFireOff) // indicate that AI fireman override is in use - { - AIFireOverride = true; // Set whenever SetFireOn or SetFireOff are selected - } - else if (BoilerPressurePSI < MaxBoilerPressurePSI && BoilerHeatSmoothedBTU < MaxBoilerHeatBTU && BoilerHeatBTU < MaxBoilerSafetyPressHeatBTU) - { - AIFireOverride = false; // Reset if pressure and heat back to "normal" - } + if (BoilerHeatBTU > MaxBoilerHeatBTU && BoilerHeatBTU <= MaxBoilerSafetyPressHeatBTU && throttle > 0.1 && cutoff > 0.1 && BoilerHeatInBTUpS < PreviousBoilerHeatOutBTUpS && FullMaxPressBoilerHeat) + // This allows for situation where boiler heat has gone beyond safety valve heat, and is reducing, but steam will be required shortly so don't allow fire to go too low + { + // Burn Rate rate if Boiler Heat is too high + BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanStartingBurnFactor; // Calculate the amount of coal that should be burnt based upon heat used by boiler + } + else + { + // Burn Rate rate if Boiler Heat is "normal" + BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactor; + } + } - if (SetFireReset) // Check FireReset Override command - resets fireoff and fireon override - { - SetFireOff = false; - SetFireOn = false; - SetFireReset = false; - } + // AIFireOverride flag set to challenge driver if boiler AI fireman is overriden - ie steam safety valves will be set and blow if pressure is excessive + if (SetFireOn || SetFireOff) // indicate that AI fireman override is in use + { + AIFireOverride = true; // Set whenever SetFireOn or SetFireOff are selected + } + else if (BoilerPressurePSI < MaxBoilerPressurePSI && BoilerHeatSmoothedBTU < MaxBoilerHeatBTU && BoilerHeatBTU < MaxBoilerSafetyPressHeatBTU) + { + AIFireOverride = false; // Reset if pressure and heat back to "normal" + } - // Check FireOff Override command - allows player to force fire low in preparation for a station stop - if (SetFireOff) - { - if (BoilerPressurePSI < MaxBoilerPressurePSI - 20.0f || BoilerHeatSmoothedBTU < 0.90f || (absSpeedMpS < 0.01f && throttle < 0.01f)) + if (SetFireReset) // Check FireReset Override command - resets fireoff and fireon override { - SetFireOff = false; // Disable FireOff if bolierpressure too low + SetFireOff = false; + SetFireOn = false; + SetFireReset = false; } - BurnRateRawKGpS = 0.0035f; - } + // Check FireOff Override command - allows player to force fire low in preparation for a station stop + if (SetFireOff) + { + if (BoilerPressurePSI < MaxBoilerPressurePSI - 20.0f || BoilerHeatSmoothedBTU < 0.90f || (absSpeedMpS < 0.01f && throttle < 0.01f)) + { + SetFireOff = false; // Disable FireOff if bolierpressure too low + } - // Check FireOn Override command - allows player to force the fire up in preparation for a station departure - if (SetFireOn) - { - if ((BoilerHeatSmoothedBTU > 0.995f * MaxBoilerHeatBTU && absSpeedMpS > 10.0f) || BoilerPressurePSI > MaxBoilerPressurePSI || absSpeedMpS <= 10.0f && (BoilerHeatSmoothedBTU > MaxBoilerHeatBTU || BoilerHeatBTU > 1.1f * MaxBoilerSafetyPressHeatBTU)) + BurnRateRawKGpS = 0.0035f; + } + + // Check FireOn Override command - allows player to force the fire up in preparation for a station departure + if (SetFireOn) { - SetFireOn = false; // Disable FireOn if bolierpressure and boilerheat back to "normal" + if ((BoilerHeatSmoothedBTU > 0.995f * MaxBoilerHeatBTU && absSpeedMpS > 10.0f) || BoilerPressurePSI > MaxBoilerPressurePSI || absSpeedMpS <= 10.0f && (BoilerHeatSmoothedBTU > MaxBoilerHeatBTU || BoilerHeatBTU > 1.1f * MaxBoilerSafetyPressHeatBTU)) + { + SetFireOn = false; // Disable FireOn if bolierpressure and boilerheat back to "normal" + } + BurnRateRawKGpS = 0.9f * pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); // AI fire on goes to approx 100% of fire needed to maintain full boiler steam generation } - BurnRateRawKGpS = 0.9f * pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); // AI fire on goes to approx 100% of fire needed to maintain full boiler steam generation } } @@ -3824,9 +3859,21 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) { BurnRateRawKGpS = 0.0f; // Drop fire due to melting of fusible plug and steam quenching fire, change later to allow graduate ramp down. } + + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + // var oilburnrate = throttle * MaxFuelBurnGrateKGpS; + // OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); + + OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, BurnRateRawKGpS); + FuelBurnRateSmoothedKGpS = OilBurnRateSmoothKGpS.SmoothedValue; + } + else + { + CoalBurnRateSmoothKGpS.Update(elapsedClockSeconds, BurnRateRawKGpS); + FuelBurnRateSmoothedKGpS = CoalBurnRateSmoothKGpS.SmoothedValue; + } - BurnRateSmoothKGpS.Update(elapsedClockSeconds, BurnRateRawKGpS); - FuelBurnRateSmoothedKGpS = BurnRateSmoothKGpS.SmoothedValue; FuelBurnRateSmoothedKGpS = MathHelper.Clamp(FuelBurnRateSmoothedKGpS, 0.0f, MaxFuelBurnGrateKGpS); // clamp burnrate to max fuel that can be burnt within grate limit #endregion @@ -3834,22 +3881,37 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) if (FiringIsManual) { - FuelRateSmoothed = CoalIsExhausted ? 0 : FiringRateController.CurrentValue; - FuelFeedRateKGpS = MaxFiringRateKGpS * FuelRateSmoothed; + FuelFeedRateSmoothedKGpS = CoalIsExhausted ? 0 : FiringRateController.CurrentValue; + FuelFeedRateKGpS = MaxFiringRateKGpS * FuelFeedRateSmoothedKGpS; } else if (elapsedClockSeconds > 0.001 && MaxFiringRateKGpS > 0.001) { // Automatic fireman, ish. - DesiredChange = MathHelper.Clamp(((IdealFireMassKG - FireMassKG) + FuelBurnRateSmoothedKGpS) / MaxFiringRateKGpS, 0.001f, 1); - if (StokerIsMechanical) // if a stoker is fitted expect a quicker response to fuel feeding + + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) { + + // var OilDesiredRate = MathHelper.Clamp(FuelBurnRateSmoothedKGpS / MaxFiringRateKGpS, 0.001f, 1); + + var OilDesiredRate = BurnRateRawKGpS; + + FuelRateOil.Update(elapsedClockSeconds, OilDesiredRate); // faster fuel feed rate for stoker + FuelFeedRateSmoothedKGpS = CoalIsExhausted ? 0 : FuelRateOil.SmoothedValue; // If tender coal is empty stop fuelrate (feeding coal onto fire). + DisplayMaxFiringRateKGpS = MaxFiringRateKGpS; // Reset display max firing rate to new figure + + } + else if (StokerIsMechanical) // if a stoker is fitted expect a quicker response to fuel feeding + { + + DesiredChange = MathHelper.Clamp(((IdealFireMassKG - FireMassKG) + FuelBurnRateSmoothedKGpS) / MaxFiringRateKGpS, 0.001f, 1); FuelRateStoker.Update(elapsedClockSeconds, DesiredChange); // faster fuel feed rate for stoker - FuelRateSmoothed = CoalIsExhausted ? 0 : FuelRateStoker.SmoothedValue; // If tender coal is empty stop fuelrate (feeding coal onto fire). + FuelFeedRateSmoothedKGpS = CoalIsExhausted ? 0 : FuelRateStoker.SmoothedValue; // If tender coal is empty stop fuelrate (feeding coal onto fire). } else { - FuelRate.Update(elapsedClockSeconds, DesiredChange); // slower fuel feed rate for fireman - FuelRateSmoothed = CoalIsExhausted ? 0 : FuelRate.SmoothedValue; // If tender coal is empty stop fuelrate (feeding coal onto fire). + DesiredChange = MathHelper.Clamp(((IdealFireMassKG - FireMassKG) + FuelBurnRateSmoothedKGpS) / MaxFiringRateKGpS, 0.001f, 1); + FuelRateCoal.Update(elapsedClockSeconds, DesiredChange); // slower fuel feed rate for AI fireman + FuelFeedRateSmoothedKGpS = CoalIsExhausted ? 0 : FuelRateCoal.SmoothedValue; // If tender coal is empty stop fuelrate (feeding coal onto fire). } float CurrentFireLevelfactor = 0.95f; // factor representing the how low firemass has got compared to ideal firemass @@ -3882,15 +3944,17 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) if (FuelBoost && !FuelBoostReset) // if fuel boost is still on, and hasn't been reset - needs further refinement as this shouldn't be able to be maintained indefinitely { DisplayMaxFiringRateKGpS = MaxTheoreticalFiringRateKgpS; // Set display value with temporary higher shovelling level - FuelFeedRateKGpS = MaxTheoreticalFiringRateKgpS * FuelRateSmoothed; // At times of heavy burning allow AI fireman to overfuel + FuelFeedRateKGpS = MaxTheoreticalFiringRateKgpS * FuelFeedRateSmoothedKGpS; // At times of heavy burning allow AI fireman to overfuel FuelBoostOnTimerS += elapsedClockSeconds; // Time how long to fuel boost for } else { - DisplayMaxFiringRateKGpS = MaxFiringRateKGpS; // Rest display max firing rate to new figure - FuelFeedRateKGpS = MaxFiringRateKGpS * FuelRateSmoothed; + DisplayMaxFiringRateKGpS = MaxFiringRateKGpS; // Reset display max firing rate to new figure + FuelFeedRateKGpS = MaxFiringRateKGpS * FuelFeedRateSmoothedKGpS; } + } + // Calculate update to firemass as a result of adding fuel to the fire FireMassKG += elapsedClockSeconds * (FuelFeedRateKGpS - FuelBurnRateSmoothedKGpS); FireMassKG = MathHelper.Clamp(FireMassKG, 0, MaxFireMassKG); @@ -7767,7 +7831,25 @@ public override string GetDebugStatus() #endif status.AppendFormat("\n\t\t === {0} === \n", Simulator.Catalog.GetString("Fireman")); - status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t\t{5}\t{6:N0}/{13}\t\t{7}\t{8:N0}/{13}\t\t{9}\t{10:N0}/{13}\t\t{11}\t{12}/{14}{13}\t{15}\t{16}/{18}{17}\t\t{19}\t{20:N0}\n", + + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + status.AppendFormat("{0}\t{1}\t{2:N0}/{7}\t\t{3}\t{4:N0}/{7}\t\t{5}\t{6:N0}/{7}\n", + Simulator.Catalog.GetString("Fire:"), + Simulator.Catalog.GetString("MaxFireR"), + FormatStrings.FormatFuelVolume(pS.TopH(L.FromGUK(OilSpecificGravity * (Kg.ToLb(DisplayMaxFiringRateKGpS) / WaterLBpUKG))), IsMetric, IsUK), + Simulator.Catalog.GetString("FeedRate"), + FormatStrings.FormatFuelVolume(pS.TopH(L.FromGUK(OilSpecificGravity * (Kg.ToLb(FuelFeedRateSmoothedKGpS) / WaterLBpUKG))), IsMetric, IsUK), + Simulator.Catalog.GetString("BurnRate"), + FormatStrings.FormatFuelVolume(pS.TopH(L.FromGUK(OilSpecificGravity * (Kg.ToLb(FuelBurnRateSmoothedKGpS) / WaterLBpUKG))), IsMetric, IsUK), + FormatStrings.h + ); + + } + else + { + + status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t\t{5}\t{6:N0}/{13}\t\t{7}\t{8:N0}/{13}\t\t{9}\t{10:N0}/{13}\t\t{11}\t{12}/{14}{13}\t{15}\t{16}/{18}{17}\t\t{19}\t{20:N0}\n", Simulator.Catalog.GetString("Fire:"), Simulator.Catalog.GetString("Ideal"), FormatStrings.FormatMass(IdealFireMassKG, IsMetric), @@ -7776,7 +7858,7 @@ public override string GetDebugStatus() Simulator.Catalog.GetString("MaxFireR"), FormatStrings.FormatMass(pS.TopH(DisplayMaxFiringRateKGpS), IsMetric), Simulator.Catalog.GetString("FeedRate"), - FormatStrings.FormatMass(pS.TopH(FuelFeedRateKGpS), IsMetric), + FormatStrings.FormatMass(pS.TopH(FuelFeedRateSmoothedKGpS), IsMetric), Simulator.Catalog.GetString("BurnRate"), FormatStrings.FormatMass(pS.TopH(FuelBurnRateSmoothedKGpS), IsMetric), Simulator.Catalog.GetString("Combust"), @@ -7790,6 +7872,8 @@ public override string GetDebugStatus() Simulator.Catalog.GetString("MaxBurn"), MaxFiringRateLbpH ); + } + #if DEBUG_LOCO_BURN_AI status.AppendFormat("{0}\t{1}\t{2}\t{3}\t{4:N2}\t{5}\t{6:N2}\t{7}\t{8:N2}\t{9}\t{10:N2}\t{11}\t{12:N0}\t{13}\t{14:N0}\t{15}\t{16:N0}\t{17}\t{18:N0}\t{19}\t{20:N0}\t{21}\t{22}\t{23}\t{24}\t{25}\t{26}\n {27}\t{28}\t{29:N3}\n", @@ -7825,7 +7909,7 @@ public override string GetDebugStatus() FireHeatLossPercent); #endif - status.AppendFormat("{0}\t{1}\t{6}/{12}\t\t({7:N0} {13})\t{2}\t{8}/{12}\t\t{3}\t{9}\t\t{4}\t{10}/{12}\t\t{5}\t{11}\n", + status.AppendFormat("{0}\t{1}\t{6}/{12}\t\t({7:N0} {13})\t{2}\t{8}/{12}\t\t{3}\t{9}\t\t{4}\t{10}/{12}\t\t{5}\t{11}\n", Simulator.Catalog.GetString("Injector:"), Simulator.Catalog.GetString("Max"), Simulator.Catalog.GetString("Inj1"), @@ -7863,22 +7947,69 @@ public override string GetDebugStatus() } else { - status.AppendFormat("{0}\t{1}\t{2}\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", - Simulator.Catalog.GetString("Tender:"), - Simulator.Catalog.GetString("Coal"), - FormatStrings.FormatMass(TenderCoalMassKG, IsMetric), - TenderCoalMassKG / MaxTenderCoalMassKG * 100, - Simulator.Catalog.GetString("Water"), - FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), - CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, - Simulator.Catalog.GetString("Steam"), - FormatStrings.FormatMass(Kg.FromLb(CumulativeCylinderSteamConsumptionLbs), IsMetric), - Simulator.Catalog.GetString("TotSteam"), - FormatStrings.FormatMass(Kg.FromLb(CummulativeTotalSteamConsumptionLbs), IsMetric) - ); + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + status.AppendFormat("{0}\t{1}\t{2:N0}\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", + Simulator.Catalog.GetString("Tender:"), + Simulator.Catalog.GetString("Oil"), + FormatStrings.FormatFuelVolume(L.FromGUK(OilSpecificGravity * (Kg.ToLb(TenderCoalMassKG) / WaterLBpUKG)), IsMetric, IsUK), + TenderCoalMassKG / MaxTenderCoalMassKG * 100, + Simulator.Catalog.GetString("Water"), + FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), + CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, + Simulator.Catalog.GetString("Steam"), + FormatStrings.FormatMass(Kg.FromLb(CumulativeCylinderSteamConsumptionLbs), IsMetric), + Simulator.Catalog.GetString("TotSteam"), + FormatStrings.FormatMass(Kg.FromLb(CummulativeTotalSteamConsumptionLbs), IsMetric) + ); + } + else // default to coal + { + status.AppendFormat("{0}\t{1}\t{2}\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", + Simulator.Catalog.GetString("Tender:"), + Simulator.Catalog.GetString("Coal"), + FormatStrings.FormatMass(TenderCoalMassKG, IsMetric), + TenderCoalMassKG / MaxTenderCoalMassKG * 100, + Simulator.Catalog.GetString("Water"), + FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), + CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, + Simulator.Catalog.GetString("Steam"), + FormatStrings.FormatMass(Kg.FromLb(CumulativeCylinderSteamConsumptionLbs), IsMetric), + Simulator.Catalog.GetString("TotSteam"), + FormatStrings.FormatMass(Kg.FromLb(CummulativeTotalSteamConsumptionLbs), IsMetric) + ); + } + } + + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + + status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t{5}\t{6}\t{7}\t{8}\t{9}\t{10}\t{11}\t{12}\t{13}\t{14}\t{15}\t{16}\t{17}\t{18}\n", + Simulator.Catalog.GetString("Status:"), + Simulator.Catalog.GetString("OilOut"), + CoalIsExhausted ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("WaterOut"), + WaterIsExhausted ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("FireOut"), + FireIsExhausted ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("Stoker"), + StokerIsMechanical ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("Boost"), + FuelBoost ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("GrLimit"), + IsGrateLimit ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("FireOn"), + SetFireOn ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("FireOff"), + SetFireOff ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("AIOR"), + AIFireOverride ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No") + ); } + else + { - status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t{5}\t{6}\t{7}\t{8}\t{9}\t{10}\t{11}\t{12}\t{13}\t{14}\t{15}\t{16}\t{17}\t{18}\n", + status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t{5}\t{6}\t{7}\t{8}\t{9}\t{10}\t{11}\t{12}\t{13}\t{14}\t{15}\t{16}\t{17}\t{18}\n", Simulator.Catalog.GetString("Status:"), Simulator.Catalog.GetString("CoalOut"), CoalIsExhausted ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), @@ -7900,6 +8031,8 @@ public override string GetDebugStatus() AIFireOverride ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No") ); + } + if (SteamEngineType == SteamEngineTypes.Geared) { status.AppendFormat("\n\t\t === {0} === \n", Simulator.Catalog.GetString("Performance")); From c2a5d32d51e0ca1436dcec1f6d9021abbd1b04a6 Mon Sep 17 00:00:00 2001 From: peternewell Date: Mon, 3 Jun 2024 07:44:09 +1000 Subject: [PATCH 02/25] Fix conflict --- .../RollingStocks/MSTSLocomotive.cs | 19 ++++++++++--------- 1 file changed, 10 insertions(+), 9 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs index 706c14d80..1280c9d73 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs @@ -239,6 +239,16 @@ public float CurrentLocomotiveSteamHeatBoilerWaterCapacityL //float DebugSpeed = 5.0f; // Initialise at 5 mph //float DebugTimer = 0.0f; + public enum SteamLocomotiveFuelTypes + { + Unknown, + Oil, + Wood, // not used at the moment + Coal, // defaults to coal + } + + public SteamLocomotiveFuelTypes SteamLocomotiveFuelType; + // Adhesion parameters public enum SlipControlType { @@ -478,15 +488,6 @@ public float OdometerM public float ThrottleIntervention = -1; public float DynamicBrakeIntervention = -1; - public enum SteamLocomotiveFuelTypes - { - Unknown, - Oil, - Wood, // not used at the moment - Coal, // defaults to coal - } - - public SteamLocomotiveFuelTypes SteamLocomotiveFuelType; public enum TractionMotorTypes { DC, From 8ddd645a00636c0cbe068baeeb7f2f8348107ae7 Mon Sep 17 00:00:00 2001 From: peternewell Date: Thu, 6 Jun 2024 16:32:47 +1000 Subject: [PATCH 03/25] Allow use of oil input parameters --- .../RollingStocks/MSTSSteamLocomotive.cs | 28 +++++++++++++------ 1 file changed, 20 insertions(+), 8 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index c7dd80b16..ce0d36af4 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -297,7 +297,8 @@ public class MSTSSteamLocomotive : MSTSLocomotive SmoothedData OilBurnRateSmoothKGpS = new SmoothedData(3); // Changes in Oil BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. Oil faster then steam float FuelFeedRateSmoothedKGpS = 0.0f; // Smoothed Fuel feedd Rate public float FuelBurnRateSmoothedKGpS; // Smoothed fuel burning rate - float OilSpecificGravity = 0.9659f; // Assume a mid range of API for this value, say API = 15. + float OilSpecificGravity = 0.9659f; // Assume a mid range of API for this value, say API = 15 @ 20 Cdeg. + float WaterSpecificGravity = 1.0f; // Water @ 20 degC. int NumberofTractiveForceValues = 36; float[,] TractiveForceAverageN = new float[5, 37]; @@ -383,7 +384,7 @@ public class MSTSSteamLocomotive : MSTSLocomotive float IdealFireDepthIN = 7.0f; // Assume standard coal coverage of grate = 7 inches. float FuelDensityKGpM3 = 864.5f; // Anthracite Coal : 50 - 58 (lb/ft3), 800 - 929 (kg/m3) float DamperFactorManual = 1.0f; // factor to control draft through fire when locomotive is running in Manual mode - public float WaterLBpUKG = 10.0f; // lbs of water in 1 gal (uk) + public float WaterLBpUKG = 10.021f; // lbs of water in 1 gal (uk) public float MaxTenderCoalMassKG = 1; // Maximum read from Eng File - - this value must be non-zero, if not defined in ENG file, can cause NaN errors public float TenderCoalMassKG // Decreased by firing and increased by refilling { @@ -391,6 +392,7 @@ public float TenderCoalMassKG // Decreased by firing and increased set { FuelController.CurrentValue = value / MaxTenderCoalMassKG; } } + float MaxTenderOilMassL; float DamperBurnEffect; // Effect of the Damper control Used in manual firing) float Injector1Fraction = 0.0f; // Fraction (0-1) of injector 1 flow from Fireman controller or AI float Injector2Fraction = 0.0f; // Fraction (0-1) of injector of injector 2 flow from Fireman controller or AI @@ -914,10 +916,12 @@ public override void Parse(string lowercasetoken, STFReader stf) case "engine(ortssuperheatcutoffpressurefactor": SuperheatCutoffPressureFactor = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(shovelcoalmass": ShovelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "engine(maxtendercoalmass": MaxTenderCoalMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; + case "engine(ortsmaxtenderfueloilvolume": MaxTenderOilMassL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; case "engine(maxtenderwatermass": MaxLocoTenderWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "engine(steamfiremanmaxpossiblefiringrate": MaxFiringRateKGpS = stf.ReadFloatBlock(STFReader.UNITS.MassRateDefaultLBpH, null) / 2.2046f / 3600; break; case "engine(steamfiremanismechanicalstoker": Stoker = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(ortssteamfiremanmaxpossiblefiringrate": ORTSMaxFiringRateKGpS = stf.ReadFloatBlock(STFReader.UNITS.MassRateDefaultLBpH, null) / 2.2046f / 3600; break; + case "engine(ortsfueloilspecificgravity": OilSpecificGravity = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(enginecontrollers(cutoff": CutoffController.Parse(stf); break; case "engine(enginecontrollers(ortssmallejector": SmallEjectorController.Parse(stf); SmallEjectorControllerFitted = true; break; case "engine(enginecontrollers(ortslargeejector": LargeEjectorController.Parse(stf); LargeEjectorControllerFitted = true; break; @@ -1083,6 +1087,8 @@ public override void Copy(MSTSWagon copy) CylinderExhausttoCutoff = locoCopy.CylinderExhausttoCutoff; CylinderCompressiontoCutoff = locoCopy.CylinderCompressiontoCutoff; CylinderAdmissiontoCutoff = locoCopy.CylinderAdmissiontoCutoff; + OilSpecificGravity = locoCopy.OilSpecificGravity; + MaxTenderOilMassL = locoCopy.MaxTenderOilMassL; SteamEngines.Copy(locoCopy.SteamEngines); } @@ -1276,6 +1282,12 @@ public override void Initialize() CutoffInitialPressureDropRatioUpper = SteamTable.CutoffInitialPressureUpper(); CutoffInitialPressureDropRatioLower = SteamTable.CutoffInitialPressureLower(); + // Set Oil mass - if an oil locomotive + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil && MaxTenderOilMassL != 0) + { + MaxTenderCoalMassKG = MaxTenderOilMassL * OilSpecificGravity; + } + // Assign default steam table values if cylinder event is not in ENG file if (CylinderExhausttoCutoff == null) { @@ -1477,9 +1489,9 @@ public override void Initialize() // Set to compound operation intially // include check to see if it is a balanced compound locomotive, ie number of HP cylinders = number of LP cylinders - if(SteamEngines[i].NumberCylinders != SteamEngines[i].LPNumberCylinders && Simulator.Settings.VerboseConfigurationMessages) + if (SteamEngines[i].NumberCylinders != SteamEngines[i].LPNumberCylinders && Simulator.Settings.VerboseConfigurationMessages) { - + Trace.TraceInformation("This doesn't appear to be a balanced compound locomotive, ie LP Cylinders = HP Cylinders. Game performnce may not be realistic."); SteamEngines[i].NumberCylinders = 2; SteamEngines[i].LPNumberCylinders = 2; @@ -2029,13 +2041,13 @@ public override void Initialize() for (int i = 0; i < SteamEngines.Count; i++) { - if (SteamEngines[i].MaxIndicatedHorsePowerHP == 0 && SteamEngines.Count == 1 && MaxIndicatedHorsePowerHP != 0) - // if MaxIHP is not set in ENG file, then set a default + if (SteamEngines[i].MaxIndicatedHorsePowerHP == 0 && SteamEngines.Count == 1 && MaxIndicatedHorsePowerHP != 0) + // if MaxIHP is not set in ENG file, then set a default { SteamEngines[i].MaxIndicatedHorsePowerHP = MaxIndicatedHorsePowerHP; } else if (SteamEngines[i].MaxIndicatedHorsePowerHP == 0) - { + { // Max IHP = (Max TE x Speed) / 375.0, use a factor of 0.85 to calculate max TE SteamEngines[i].MaxIndicatedHorsePowerHP = MaxSpeedFactor * (SteamEngines[i].MaxTractiveEffortLbf * MaxLocoSpeedMpH) / 375.0f; // To be checked what MaxTractive Effort is for the purposes of this formula. MaxIndicatedHorsePowerHP += SteamEngines[i].MaxIndicatedHorsePowerHP; @@ -2065,7 +2077,7 @@ public override void Initialize() // Check to see if MaxIHP is in fact limited by the boiler if (MaxIndicatedHorsePowerHP > MaxBoilerOutputHP) { - // MaxIndicatedHorsePowerHP = MaxBoilerOutputHP; // Set maxIHp to limit set by boiler - No need to limit IHP, naturally limited by steam production????? + // MaxIndicatedHorsePowerHP = MaxBoilerOutputHP; // Set maxIHp to limit set by boiler - No need to limit IHP, naturally limited by steam production????? ISBoilerLimited = true; } else From df12eb4a643159992235bd51c1313162c331a1a1 Mon Sep 17 00:00:00 2001 From: peternewell Date: Fri, 7 Jun 2024 12:37:28 +1000 Subject: [PATCH 04/25] Set mechanical stoker always true --- .../Simulation/RollingStocks/MSTSSteamLocomotive.cs | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index ce0d36af4..01b603cbc 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -1288,6 +1288,12 @@ public override void Initialize() MaxTenderCoalMassKG = MaxTenderOilMassL * OilSpecificGravity; } + // Oil burning locomotives will always have mechanical stokers + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + StokerIsMechanical = true; + } + // Assign default steam table values if cylinder event is not in ENG file if (CylinderExhausttoCutoff == null) { From e1d8f1f991951cbbc5b97bde22d30c3cc73c5e63 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sat, 8 Jun 2024 11:41:29 +1000 Subject: [PATCH 05/25] Adjustments to display --- .../RollingStocks/MSTSSteamLocomotive.cs | 27 ++++++++++++++----- 1 file changed, 21 insertions(+), 6 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 01b603cbc..e0a8e951c 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -3880,10 +3880,14 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) { - // var oilburnrate = throttle * MaxFuelBurnGrateKGpS; - // OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); + var oilburnrate = BurnRateRawKGpS; - OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, BurnRateRawKGpS); + if (BurnRateRawKGpS > MaxFiringRateKGpS) + { + oilburnrate = MaxFiringRateKGpS; // burning rate can never be more then the max firing rate + } + + OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); FuelBurnRateSmoothedKGpS = OilBurnRateSmoothKGpS.SmoothedValue; } else @@ -7488,9 +7492,20 @@ public override string GetStatus() status.AppendFormat("{0} = {1:F0}%\n", Simulator.Catalog.GetString("Fire Heat Loss"), FireHeatLossPercent * 100); } - status.AppendFormat("{0}{5} = {3:F0}% {1}, {4:F0}% {2}{5}\n", Simulator.Catalog.GetString("Fuel levels"), Simulator.Catalog.GetString("coal"), Simulator.Catalog.GetString("water"), 100 * coalPercent, 100 * waterPercent, fuelSafety); + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + status.AppendFormat("{0}{5} = {3:F0}% {1}, {4:F0}% {2}{5}\n", Simulator.Catalog.GetString("Fuel levels"), Simulator.Catalog.GetString("oil"), Simulator.Catalog.GetString("water"), 100 * coalPercent, 100 * waterPercent, fuelSafety); + } + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + status.AppendFormat("{0}{5} = {3:F0}% {1}, {4:F0}% {2}{5}\n", Simulator.Catalog.GetString("Fuel levels"), Simulator.Catalog.GetString("wood"), Simulator.Catalog.GetString("water"), 100 * coalPercent, 100 * waterPercent, fuelSafety); + } + else + { + status.AppendFormat("{0}{5} = {3:F0}% {1}, {4:F0}% {2}{5}\n", Simulator.Catalog.GetString("Fuel levels"), Simulator.Catalog.GetString("coal"), Simulator.Catalog.GetString("water"), 100 * coalPercent, 100 * waterPercent, fuelSafety); + } - return status.ToString(); + return status.ToString(); } public override string GetDebugStatus() @@ -7967,7 +7982,7 @@ public override string GetDebugStatus() { if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) { - status.AppendFormat("{0}\t{1}\t{2:N0}\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", + status.AppendFormat("{0}\t{1}\t{2:N0}\t\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", Simulator.Catalog.GetString("Tender:"), Simulator.Catalog.GetString("Oil"), FormatStrings.FormatFuelVolume(L.FromGUK(OilSpecificGravity * (Kg.ToLb(TenderCoalMassKG) / WaterLBpUKG)), IsMetric, IsUK), From 956a8ec0b1d84d75a583eef3632bcfc391a6ae66 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sat, 8 Jun 2024 11:59:15 +1000 Subject: [PATCH 06/25] Turn off steam stoker steam usage values when not coal fired. --- .../Simulation/RollingStocks/MSTSSteamLocomotive.cs | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index e0a8e951c..235344c59 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -3887,6 +3887,8 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) oilburnrate = MaxFiringRateKGpS; // burning rate can never be more then the max firing rate } + // OilBurnRateSmoothKGpS.ForceSmoothValue(oilburnrate); + OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); FuelBurnRateSmoothedKGpS = OilBurnRateSmoothKGpS.SmoothedValue; } @@ -6801,7 +6803,7 @@ private void UpdateAuxiliaries(float elapsedClockSeconds, float absSpeedMpS) GeneratorSteamUsageLBpS = 0.0f; // No generator fitted to locomotive } - if (StokerIsMechanical) + if (StokerIsMechanical && SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Coal) { StokerSteamUsageLBpS = pS.FrompH(MaxBoilerOutputLBpH) * (StokerMinUsage + (StokerMaxUsage - StokerMinUsage) * FuelFeedRateKGpS / MaxFiringRateKGpS); // Caluculate current steam usage based on fuel feed rates BoilerMassLB -= elapsedClockSeconds * StokerSteamUsageLBpS; // Reduce boiler mass to reflect steam usage by mechanical stoker From 1985cbe4c3950c6549a87aa7cd364e78a3550a63 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sat, 8 Jun 2024 17:12:03 +1000 Subject: [PATCH 07/25] Adjust combustion times --- .../RollingStocks/MSTSSteamLocomotive.cs | 143 ++++++++++-------- 1 file changed, 76 insertions(+), 67 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 235344c59..bc5a394c8 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -294,7 +294,9 @@ public class MSTSSteamLocomotive : MSTSLocomotive SmoothedData FuelRateStoker = new SmoothedData(15); // Stoker is more responsive and only takes x seconds to fully react to changing needs. SmoothedData FuelRateCoal = new SmoothedData(45); // Automatic fireman takes x seconds to fully react to changing needs for coal firing. SmoothedData CoalBurnRateSmoothKGpS = new SmoothedData(150); // Changes in BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. - SmoothedData OilBurnRateSmoothKGpS = new SmoothedData(3); // Changes in Oil BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. Oil faster then steam + SmoothedData OilBurnRateSmoothKGpS = new SmoothedData(6); // Changes in Oil BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. Oil faster then steam + SmoothedData OilBurnRateReductionSmoothKGpS = new SmoothedData(0.25f); // When in reduction we would expect the heat to drop off rapidly for an oil fire + float FuelFeedRateSmoothedKGpS = 0.0f; // Smoothed Fuel feedd Rate public float FuelBurnRateSmoothedKGpS; // Smoothed fuel burning rate float OilSpecificGravity = 0.9659f; // Assume a mid range of API for this value, say API = 15 @ 20 Cdeg. @@ -488,6 +490,7 @@ public float TenderCoalMassKG // Decreased by firing and increased float TimeFuelBoostOnS = 300.0f; // Time to allow fuel boosting to go on for float TimeFuelBoostResetS = 1200.0f;// Time to wait before next fuel boost float throttle; + float previousThrottle; float SpeedEquivMpS = 27.0f; // Equvalent speed of 60mph in mps (27m/s) - used for damper control // Cylinder related parameters @@ -3742,84 +3745,77 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) else // *********** AI Fireman ***************** { -// if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) -// { -// BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG); -// } -// else + if (PreviousTotalSteamUsageLBpS > TheoreticalMaxSteamOutputLBpS) { - if (PreviousTotalSteamUsageLBpS > TheoreticalMaxSteamOutputLBpS) - { - FiringSteamUsageRateLBpS = TheoreticalMaxSteamOutputLBpS; // hold usage rate if steam usage rate exceeds boiler max output - } - else - { - FiringSteamUsageRateLBpS = PreviousTotalSteamUsageLBpS; // Current steam usage rate - } + FiringSteamUsageRateLBpS = TheoreticalMaxSteamOutputLBpS; // hold usage rate if steam usage rate exceeds boiler max output + } + else + { + FiringSteamUsageRateLBpS = PreviousTotalSteamUsageLBpS; // Current steam usage rate + } - AIFiremanBurnFactorExceed = HeatRatio * PressureRatio; // Firing rate for AI fireman if firemass drops, and fireman needs to exceed normal capacity - AIFiremanBurnFactor = HeatRatio * PressureRatio * FullBoilerHeatRatio * MaxBoilerHeatRatio; // Firing rate factor under normal conditions - float AIFiremanStartingBurnFactor = 4.0f; + AIFiremanBurnFactorExceed = HeatRatio * PressureRatio; // Firing rate for AI fireman if firemass drops, and fireman needs to exceed normal capacity + AIFiremanBurnFactor = HeatRatio * PressureRatio * FullBoilerHeatRatio * MaxBoilerHeatRatio; // Firing rate factor under normal conditions + float AIFiremanStartingBurnFactor = 4.0f; - if (ShovelAnyway && BoilerHeatBTU < MaxBoilerHeatBTU) // will force fire burn rate to increase even though boiler heat seems excessive + if (ShovelAnyway && BoilerHeatBTU < MaxBoilerHeatBTU) // will force fire burn rate to increase even though boiler heat seems excessive + { + // burnrate will be the radiation loss @ rest & then related to heat usage as a factor of the maximum boiler output + // ignores total bolier heat to allow burn rate to increase if boiler heat usage is exceeding input + BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactorExceed; + } + else + { + + if (BoilerHeatBTU > MaxBoilerHeatBTU && BoilerHeatBTU <= MaxBoilerSafetyPressHeatBTU && throttle > 0.1 && cutoff > 0.1 && BoilerHeatInBTUpS < PreviousBoilerHeatOutBTUpS && FullMaxPressBoilerHeat) + // This allows for situation where boiler heat has gone beyond safety valve heat, and is reducing, but steam will be required shortly so don't allow fire to go too low { - // burnrate will be the radiation loss @ rest & then related to heat usage as a factor of the maximum boiler output - // ignores total bolier heat to allow burn rate to increase if boiler heat usage is exceeding input - BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactorExceed; + // Burn Rate rate if Boiler Heat is too high + BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanStartingBurnFactor; // Calculate the amount of coal that should be burnt based upon heat used by boiler } else { - - if (BoilerHeatBTU > MaxBoilerHeatBTU && BoilerHeatBTU <= MaxBoilerSafetyPressHeatBTU && throttle > 0.1 && cutoff > 0.1 && BoilerHeatInBTUpS < PreviousBoilerHeatOutBTUpS && FullMaxPressBoilerHeat) - // This allows for situation where boiler heat has gone beyond safety valve heat, and is reducing, but steam will be required shortly so don't allow fire to go too low - { - // Burn Rate rate if Boiler Heat is too high - BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanStartingBurnFactor; // Calculate the amount of coal that should be burnt based upon heat used by boiler - } - else - { - // Burn Rate rate if Boiler Heat is "normal" - BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactor; - } + // Burn Rate rate if Boiler Heat is "normal" + BurnRateRawKGpS = (W.ToKW(W.FromBTUpS(PreviousBoilerHeatOutBTUpS - BoilerHeatExceptionsBtupS)) / FuelCalorificKJpKG) * AIFiremanBurnFactor; } + } - // AIFireOverride flag set to challenge driver if boiler AI fireman is overriden - ie steam safety valves will be set and blow if pressure is excessive - if (SetFireOn || SetFireOff) // indicate that AI fireman override is in use - { - AIFireOverride = true; // Set whenever SetFireOn or SetFireOff are selected - } - else if (BoilerPressurePSI < MaxBoilerPressurePSI && BoilerHeatSmoothedBTU < MaxBoilerHeatBTU && BoilerHeatBTU < MaxBoilerSafetyPressHeatBTU) - { - AIFireOverride = false; // Reset if pressure and heat back to "normal" - } + // AIFireOverride flag set to challenge driver if boiler AI fireman is overriden - ie steam safety valves will be set and blow if pressure is excessive + if (SetFireOn || SetFireOff) // indicate that AI fireman override is in use + { + AIFireOverride = true; // Set whenever SetFireOn or SetFireOff are selected + } + else if (BoilerPressurePSI < MaxBoilerPressurePSI && BoilerHeatSmoothedBTU < MaxBoilerHeatBTU && BoilerHeatBTU < MaxBoilerSafetyPressHeatBTU) + { + AIFireOverride = false; // Reset if pressure and heat back to "normal" + } - if (SetFireReset) // Check FireReset Override command - resets fireoff and fireon override - { - SetFireOff = false; - SetFireOn = false; - SetFireReset = false; - } + if (SetFireReset) // Check FireReset Override command - resets fireoff and fireon override + { + SetFireOff = false; + SetFireOn = false; + SetFireReset = false; + } - // Check FireOff Override command - allows player to force fire low in preparation for a station stop - if (SetFireOff) + // Check FireOff Override command - allows player to force fire low in preparation for a station stop + if (SetFireOff) + { + if (BoilerPressurePSI < MaxBoilerPressurePSI - 20.0f || BoilerHeatSmoothedBTU < 0.90f || (absSpeedMpS < 0.01f && throttle < 0.01f)) { - if (BoilerPressurePSI < MaxBoilerPressurePSI - 20.0f || BoilerHeatSmoothedBTU < 0.90f || (absSpeedMpS < 0.01f && throttle < 0.01f)) - { - SetFireOff = false; // Disable FireOff if bolierpressure too low - } - - BurnRateRawKGpS = 0.0035f; + SetFireOff = false; // Disable FireOff if bolierpressure too low } - // Check FireOn Override command - allows player to force the fire up in preparation for a station departure - if (SetFireOn) + BurnRateRawKGpS = 0.0035f; + } + + // Check FireOn Override command - allows player to force the fire up in preparation for a station departure + if (SetFireOn) + { + if ((BoilerHeatSmoothedBTU > 0.995f * MaxBoilerHeatBTU && absSpeedMpS > 10.0f) || BoilerPressurePSI > MaxBoilerPressurePSI || absSpeedMpS <= 10.0f && (BoilerHeatSmoothedBTU > MaxBoilerHeatBTU || BoilerHeatBTU > 1.1f * MaxBoilerSafetyPressHeatBTU)) { - if ((BoilerHeatSmoothedBTU > 0.995f * MaxBoilerHeatBTU && absSpeedMpS > 10.0f) || BoilerPressurePSI > MaxBoilerPressurePSI || absSpeedMpS <= 10.0f && (BoilerHeatSmoothedBTU > MaxBoilerHeatBTU || BoilerHeatBTU > 1.1f * MaxBoilerSafetyPressHeatBTU)) - { - SetFireOn = false; // Disable FireOn if bolierpressure and boilerheat back to "normal" - } - BurnRateRawKGpS = 0.9f * pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); // AI fire on goes to approx 100% of fire needed to maintain full boiler steam generation + SetFireOn = false; // Disable FireOn if bolierpressure and boilerheat back to "normal" } + BurnRateRawKGpS = 0.9f * pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); // AI fire on goes to approx 100% of fire needed to maintain full boiler steam generation } } @@ -3877,7 +3873,7 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) { BurnRateRawKGpS = 0.0f; // Drop fire due to melting of fusible plug and steam quenching fire, change later to allow graduate ramp down. } - + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) { var oilburnrate = BurnRateRawKGpS; @@ -3887,10 +3883,23 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) oilburnrate = MaxFiringRateKGpS; // burning rate can never be more then the max firing rate } - // OilBurnRateSmoothKGpS.ForceSmoothValue(oilburnrate); - + OilBurnRateReductionSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); - FuelBurnRateSmoothedKGpS = OilBurnRateSmoothKGpS.SmoothedValue; + + if (previousThrottle < throttle) + { + // Fire combustion drops rapidly if throttle closed (assume fuel feed is reduced rapidly) + FuelBurnRateSmoothedKGpS = OilBurnRateReductionSmoothKGpS.SmoothedValue; + OilBurnRateSmoothKGpS.ForceSmoothValue(oilburnrate); + } + else + { + // Fire combustion is normal + FuelBurnRateSmoothedKGpS = OilBurnRateSmoothKGpS.SmoothedValue; + } + + previousThrottle = throttle; + } else { From 33a3e1c0b8c748c5f41d12008035943cb95a82ea Mon Sep 17 00:00:00 2001 From: peternewell Date: Sat, 15 Jun 2024 16:58:32 +1000 Subject: [PATCH 08/25] Add model for calculating the amount of steam used to heat fuel oil in the tender --- .../RollingStocks/MSTSSteamLocomotive.cs | 136 ++++++++++++++++-- 1 file changed, 125 insertions(+), 11 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index bc5a394c8..c2776df21 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -230,6 +230,7 @@ public class MSTSSteamLocomotive : MSTSLocomotive public float CylinderSteamUsageLBpS; public float NewCylinderSteamUsageLBpS; public float BlowerSteamUsageLBpS; + public float FuelOilHeatingSteamUsageLbpS; public float EvaporationLBpS; // steam generation rate public float FireMassKG; // Mass of coal currently on grate area public float FireRatio; // Ratio of actual firemass to ideal firemass @@ -3582,6 +3583,52 @@ protected override void UpdateControllers(float elapsedClockSeconds) private void UpdateTender(float elapsedClockSeconds) { + // Calculate steam usage required to heat fuel oil in oil fired locomotive + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil ) + { + // The following calculations are based upon the interpretation of information in + // https://www.spiraxsarco.com/learn-about-steam/steam-engineering-principles-and-heat-transfer/heating-with-coils-and-jackets#article-top + // and movement of the locomotive is described in this article + // https://www.spiraxsarco.com/learn-about-steam/steam-engineering-principles-and-heat-transfer/energy-consumption-of-tanks-and-vats + // Currently due to the difficulty of finding input information some assumptions have be made, and will be described through the calculation process. + // + // The first step is to calculate the amount of heat lost through the sides of the oil tank on the tender (this loss will ultimately need to be + // compensated by steam heating + // This can be calculated by Q (heat Loss) = Txf Coeff for tank sides * Area of Sides * Temp Diff (external vs internal) + // The most accurate calculation could be made if the bottom, sides and top of oil tank area was known, and also if the aount of exposed surface + // (subject to wind effect) + float TankSurfaceAreatoMassRatio = 0.015f; // based upon the inspection of the oil masses in a couple of known locomotives + float AssumedSurfaceAreaFt2 = Kg.ToLb(MaxTenderOilMassL * OilSpecificGravity) * TankSurfaceAreatoMassRatio; + float AreaExposedtoWindMovementFraction = 0.42f; + float OilTempRequired = 150; // Oil to be heated to 150degF + float HeatDiff = OilTempRequired - C.ToF(CarOutsideTempC); + float HeatTransferCoefficientBtuphft2F = 0.00001263f * HeatDiff * HeatDiff + 0.001175f * HeatDiff + 1.776f; + + float HeatLossNoWindBTUph = (1- AreaExposedtoWindMovementFraction) * HeatTransferCoefficientBtuphft2F * AssumedSurfaceAreaFt2 * HeatDiff; + + // To compensate for the train movement we need to add a wind factor + float WindCoeff = -0.0074f * absSpeedMpS * absSpeedMpS + 0.3817f * absSpeedMpS + 1f; + WindCoeff = MathHelper.Clamp(WindCoeff, 1.0f, 5.78f); // Wind speed effect will not cause any more impact once over about 25 m/s + + float HeatLossWindBTUph = (1 - AreaExposedtoWindMovementFraction) * HeatTransferCoefficientBtuphft2F * AssumedSurfaceAreaFt2 * HeatDiff * WindCoeff; + + float HeatLossTotalBTUph = HeatLossWindBTUph + HeatLossNoWindBTUph; + + float HeatingCoilEfficiency = 0.12f; + float SteamEnthalpyBTUpLb = 980; + FuelOilHeatingSteamUsageLbpS = pS.FrompH(HeatLossTotalBTUph / (HeatingCoilEfficiency * SteamEnthalpyBTUpLb)); + + // adjust boiler heat and volume due to steam usage + BoilerMassLB -= elapsedClockSeconds * FuelOilHeatingSteamUsageLbpS; // Reduce boiler mass to reflect steam usage by fuel oil heating coils + BoilerHeatBTU -= elapsedClockSeconds * FuelOilHeatingSteamUsageLbpS * (BoilerSteamHeatBTUpLB - BoilerWaterHeatBTUpLB); // Reduce boiler Heat to reflect steam usage by fuel oil heating coils + BoilerHeatOutBTUpS += FuelOilHeatingSteamUsageLbpS * (BoilerSteamHeatBTUpLB - BoilerWaterHeatBTUpLB); // Reduce boiler Heat to reflect steam usage by mecahnical stoker + TotalSteamUsageLBpS += FuelOilHeatingSteamUsageLbpS; + + // Increase tender water mass as steam heating condensate feed back into tender + CombinedTenderWaterVolumeUKG += (elapsedClockSeconds * FuelOilHeatingSteamUsageLbpS) / WaterLBpUKG; + + } + TenderWaterLevelFraction = CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG; float TempCylinderSteamUsageLbpS = CylinderSteamUsageLBpS; // Limit Cylinder steam usage to the maximum boiler evaporation rate, lower limit is for when the locomotive is at rest and "no steam" is being used by cylinder, ensures some coal is used. @@ -7606,23 +7653,87 @@ public override string GetDebugStatus() if (!(BrakeSystem is Orts.Simulation.RollingStocks.SubSystems.Brakes.MSTS.VacuumSinglePipe)) { - // Display air compressor information - status.AppendFormat("{0}\t{1}\t{2}/{23}\t{3}\t{4}/{23}\t{5}\t{6}/{23}\t{7}\t{8}/{23}\t{9}\t{10}/{23}\t{11}\t{12}/{23}\t{13}\t{14}/{23}\t{15}\t{16}/{23}\t{17}\t{18}/{23}\t{19}\t{20}/{23}\t({21}x{22:N1}\")\n", + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + // Display air compressor information + status.AppendFormat("{0}\t{1}\t{2}/{23}\t{3}\t{4}/{23}\t{5}\t{6}/{23}\t{7}\t{8}/{23}\t{9}\t{10}/{23}\t{11}\t{12}/{23}\t{13}\t{14}/{23}\t{15}\t{16}/{23}\t{17}\t{18}/{23}\t{19}\t{20}/{23}\t({21}x{22:N1}\")\n", + Simulator.Catalog.GetString("Usage:"), + Simulator.Catalog.GetString("Cyl"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CylinderSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Blower"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(BlowerSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Comprsr"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CompSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("SafetyV"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(SafetyValveUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("CylCock"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CylCockSteamUsageDisplayLBpS)), IsMetric), + Simulator.Catalog.GetString("Genertr"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(GeneratorSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("OilHeat"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(FuelOilHeatingSteamUsageLbpS)), IsMetric), + Simulator.Catalog.GetString("BlowD"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(BlowdownSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Booster"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(HuDBoosterSteamConsumptionLbpS)), IsMetric), + Simulator.Catalog.GetString("MaxSafe"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(MaxSafetyValveDischargeLbspS)), IsMetric), + NumSafetyValves, + SafetyValveSizeIn, + FormatStrings.h); + } + else + { + // Display air compressor information with stoker fired locomotive + status.AppendFormat("{0}\t{1}\t{2}/{23}\t{3}\t{4}/{23}\t{5}\t{6}/{23}\t{7}\t{8}/{23}\t{9}\t{10}/{23}\t{11}\t{12}/{23}\t{13}\t{14}/{23}\t{15}\t{16}/{23}\t{17}\t{18}/{23}\t{19}\t{20}/{23}\t({21}x{22:N1}\")\n", + Simulator.Catalog.GetString("Usage:"), + Simulator.Catalog.GetString("Cyl"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CylinderSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Blower"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(BlowerSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Comprsr"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CompSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("SafetyV"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(SafetyValveUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("CylCock"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CylCockSteamUsageDisplayLBpS)), IsMetric), + Simulator.Catalog.GetString("Genertr"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(GeneratorSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Stoker"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(StokerSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("BlowD"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(BlowdownSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Booster"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(HuDBoosterSteamConsumptionLbpS)), IsMetric), + Simulator.Catalog.GetString("MaxSafe"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(MaxSafetyValveDischargeLbspS)), IsMetric), + NumSafetyValves, + SafetyValveSizeIn, + FormatStrings.h); + } + } + else + { + + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + // Display steam ejector information instead of air compressor with stoker + status.AppendFormat("{0}\t{1}\t{2}/{23}\t{3}\t{4}/{23}\t{5}\t{6}/{23}\t{7}\t{8}/{23}\t{9}\t{10}/{23}\t{11}\t{12}/{23}\t{13}\t{14}/{23}\t{15}\t{16}/{23}\t{17}\t{18}/{23}\t{19}\t{20}/{23}\t({21}x{22:N1}\")\n", Simulator.Catalog.GetString("Usage:"), Simulator.Catalog.GetString("Cyl"), FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CylinderSteamUsageLBpS)), IsMetric), Simulator.Catalog.GetString("Blower"), FormatStrings.FormatMass(pS.TopH(Kg.FromLb(BlowerSteamUsageLBpS)), IsMetric), - Simulator.Catalog.GetString("Comprsr"), - FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CompSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("Ejector"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(EjectorTotalSteamConsumptionLbpS)), IsMetric), Simulator.Catalog.GetString("SafetyV"), FormatStrings.FormatMass(pS.TopH(Kg.FromLb(SafetyValveUsageLBpS)), IsMetric), Simulator.Catalog.GetString("CylCock"), FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CylCockSteamUsageDisplayLBpS)), IsMetric), Simulator.Catalog.GetString("Genertr"), FormatStrings.FormatMass(pS.TopH(Kg.FromLb(GeneratorSteamUsageLBpS)), IsMetric), - Simulator.Catalog.GetString("Stoker"), - FormatStrings.FormatMass(pS.TopH(Kg.FromLb(StokerSteamUsageLBpS)), IsMetric), + Simulator.Catalog.GetString("OilHeat"), + FormatStrings.FormatMass(pS.TopH(Kg.FromLb(FuelOilHeatingSteamUsageLbpS)), IsMetric), Simulator.Catalog.GetString("BlowD"), FormatStrings.FormatMass(pS.TopH(Kg.FromLb(BlowdownSteamUsageLBpS)), IsMetric), Simulator.Catalog.GetString("Booster"), @@ -7632,11 +7743,13 @@ public override string GetDebugStatus() NumSafetyValves, SafetyValveSizeIn, FormatStrings.h); - } - else - { - // Display steam ejector information instead of air compressor - status.AppendFormat("{0}\t{1}\t{2}/{23}\t{3}\t{4}/{23}\t{5}\t{6}/{23}\t{7}\t{8}/{23}\t{9}\t{10}/{23}\t{11}\t{12}/{23}\t{13}\t{14}/{23}\t{15}\t{16}/{23}\t{17}\t{18}/{23}\t{19}\t{20}/{23}\t({21}x{22:N1}\")\n", + + } + else + { + + // Display steam ejector information instead of air compressor with stoker + status.AppendFormat("{0}\t{1}\t{2}/{23}\t{3}\t{4}/{23}\t{5}\t{6}/{23}\t{7}\t{8}/{23}\t{9}\t{10}/{23}\t{11}\t{12}/{23}\t{13}\t{14}/{23}\t{15}\t{16}/{23}\t{17}\t{18}/{23}\t{19}\t{20}/{23}\t({21}x{22:N1}\")\n", Simulator.Catalog.GetString("Usage:"), Simulator.Catalog.GetString("Cyl"), FormatStrings.FormatMass(pS.TopH(Kg.FromLb(CylinderSteamUsageLBpS)), IsMetric), @@ -7661,6 +7774,7 @@ public override string GetDebugStatus() NumSafetyValves, SafetyValveSizeIn, FormatStrings.h); + } } From 5026de6898741749e58d5c897a9c218a2cefcc46 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sun, 16 Jun 2024 16:17:05 +1000 Subject: [PATCH 09/25] Switch to turn oil heating on/off --- .../Simulation/RollingStocks/MSTSSteamLocomotive.cs | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index c2776df21..4ccd8f72e 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -303,6 +303,8 @@ public class MSTSSteamLocomotive : MSTSLocomotive float OilSpecificGravity = 0.9659f; // Assume a mid range of API for this value, say API = 15 @ 20 Cdeg. float WaterSpecificGravity = 1.0f; // Water @ 20 degC. + bool FuelOilSteamHeatingReqd = false; + int NumberofTractiveForceValues = 36; float[,] TractiveForceAverageN = new float[5, 37]; float AverageTractiveForceN; @@ -925,6 +927,11 @@ public override void Parse(string lowercasetoken, STFReader stf) case "engine(steamfiremanmaxpossiblefiringrate": MaxFiringRateKGpS = stf.ReadFloatBlock(STFReader.UNITS.MassRateDefaultLBpH, null) / 2.2046f / 3600; break; case "engine(steamfiremanismechanicalstoker": Stoker = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(ortssteamfiremanmaxpossiblefiringrate": ORTSMaxFiringRateKGpS = stf.ReadFloatBlock(STFReader.UNITS.MassRateDefaultLBpH, null) / 2.2046f / 3600; break; + case "engine(ortsfueloilheatingrequired": + var heating = stf.ReadIntBlock(null); + if (heating == 1) + FuelOilSteamHeatingReqd = true; + break; case "engine(ortsfueloilspecificgravity": OilSpecificGravity = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(enginecontrollers(cutoff": CutoffController.Parse(stf); break; case "engine(enginecontrollers(ortssmallejector": SmallEjectorController.Parse(stf); SmallEjectorControllerFitted = true; break; @@ -1053,6 +1060,7 @@ public override void Copy(MSTSWagon copy) MaxLocoTenderWaterMassKG = locoCopy.MaxLocoTenderWaterMassKG; MaxFiringRateKGpS = locoCopy.MaxFiringRateKGpS; Stoker = locoCopy.Stoker; + FuelOilSteamHeatingReqd = locoCopy.FuelOilSteamHeatingReqd; ORTSMaxFiringRateKGpS = locoCopy.ORTSMaxFiringRateKGpS; CutoffController = (MSTSNotchController)locoCopy.CutoffController.Clone(); Injector1Controller = (MSTSNotchController)locoCopy.Injector1Controller.Clone(); @@ -3584,7 +3592,7 @@ protected override void UpdateControllers(float elapsedClockSeconds) private void UpdateTender(float elapsedClockSeconds) { // Calculate steam usage required to heat fuel oil in oil fired locomotive - if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil ) + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil && FuelOilSteamHeatingReqd ) { // The following calculations are based upon the interpretation of information in // https://www.spiraxsarco.com/learn-about-steam/steam-engineering-principles-and-heat-transfer/heating-with-coils-and-jackets#article-top @@ -6859,7 +6867,7 @@ private void UpdateAuxiliaries(float elapsedClockSeconds, float absSpeedMpS) GeneratorSteamUsageLBpS = 0.0f; // No generator fitted to locomotive } - if (StokerIsMechanical && SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Coal) + if (StokerIsMechanical && SteamLocomotiveFuelType != SteamLocomotiveFuelTypes.Oil) { StokerSteamUsageLBpS = pS.FrompH(MaxBoilerOutputLBpH) * (StokerMinUsage + (StokerMaxUsage - StokerMinUsage) * FuelFeedRateKGpS / MaxFiringRateKGpS); // Caluculate current steam usage based on fuel feed rates BoilerMassLB -= elapsedClockSeconds * StokerSteamUsageLBpS; // Reduce boiler mass to reflect steam usage by mechanical stoker From ce2bbe117f22b54731640fc8ec8248795e0fb982 Mon Sep 17 00:00:00 2001 From: peternewell Date: Tue, 18 Jun 2024 21:16:06 +1000 Subject: [PATCH 10/25] Add some wood parameters --- .../RollingStocks/MSTSSteamLocomotive.cs | 44 +++++++++++++++++-- 1 file changed, 41 insertions(+), 3 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 4ccd8f72e..4b62c7881 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -8014,7 +8014,6 @@ public override string GetDebugStatus() } else { - status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t\t{5}\t{6:N0}/{13}\t\t{7}\t{8:N0}/{13}\t\t{9}\t{10:N0}/{13}\t\t{11}\t{12}/{14}{13}\t{15}\t{16}/{18}{17}\t\t{19}\t{20:N0}\n", Simulator.Catalog.GetString("Fire:"), Simulator.Catalog.GetString("Ideal"), @@ -8024,7 +8023,7 @@ public override string GetDebugStatus() Simulator.Catalog.GetString("MaxFireR"), FormatStrings.FormatMass(pS.TopH(DisplayMaxFiringRateKGpS), IsMetric), Simulator.Catalog.GetString("FeedRate"), - FormatStrings.FormatMass(pS.TopH(FuelFeedRateSmoothedKGpS), IsMetric), + FormatStrings.FormatMass(pS.TopH(FuelFeedRateKGpS), IsMetric), // Check why not a smoothed value? Simulator.Catalog.GetString("BurnRate"), FormatStrings.FormatMass(pS.TopH(FuelBurnRateSmoothedKGpS), IsMetric), Simulator.Catalog.GetString("Combust"), @@ -8129,6 +8128,22 @@ public override string GetDebugStatus() FormatStrings.FormatMass(Kg.FromLb(CummulativeTotalSteamConsumptionLbs), IsMetric) ); } + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + status.AppendFormat("{0}\t{1}\t{2:N0}\t\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", + Simulator.Catalog.GetString("Tender:"), + Simulator.Catalog.GetString("Wood"), + FormatStrings.FormatFuelVolume(L.FromGUK(OilSpecificGravity * (Kg.ToLb(TenderCoalMassKG) / WaterLBpUKG)), IsMetric, IsUK), + TenderCoalMassKG / MaxTenderCoalMassKG * 100, + Simulator.Catalog.GetString("Water"), + FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), + CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, + Simulator.Catalog.GetString("Steam"), + FormatStrings.FormatMass(Kg.FromLb(CumulativeCylinderSteamConsumptionLbs), IsMetric), + Simulator.Catalog.GetString("TotSteam"), + FormatStrings.FormatMass(Kg.FromLb(CummulativeTotalSteamConsumptionLbs), IsMetric) + ); + } else // default to coal { status.AppendFormat("{0}\t{1}\t{2}\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", @@ -8149,7 +8164,6 @@ public override string GetDebugStatus() if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) { - status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t{5}\t{6}\t{7}\t{8}\t{9}\t{10}\t{11}\t{12}\t{13}\t{14}\t{15}\t{16}\t{17}\t{18}\n", Simulator.Catalog.GetString("Status:"), Simulator.Catalog.GetString("OilOut"), @@ -8172,6 +8186,30 @@ public override string GetDebugStatus() AIFireOverride ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No") ); } + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + status.AppendFormat("{0}\t{1}\t{2}\t\t{3}\t{4}\t{5}\t{6}\t{7}\t{8}\t{9}\t{10}\t{11}\t{12}\t{13}\t{14}\t{15}\t{16}\t{17}\t{18}\n", + Simulator.Catalog.GetString("Status:"), + Simulator.Catalog.GetString("WoodOut"), + CoalIsExhausted ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("WaterOut"), + WaterIsExhausted ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("FireOut"), + FireIsExhausted ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("Stoker"), + StokerIsMechanical ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("Boost"), + FuelBoost ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("GrLimit"), + IsGrateLimit ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("FireOn"), + SetFireOn ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("FireOff"), + SetFireOff ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No"), + Simulator.Catalog.GetString("AIOR"), + AIFireOverride ? Simulator.Catalog.GetString("Yes") : Simulator.Catalog.GetString("No") + ); + } else { From aac0de8e861b1f722e79f98814ba7edbd249b933 Mon Sep 17 00:00:00 2001 From: peternewell Date: Wed, 19 Jun 2024 13:50:39 +1000 Subject: [PATCH 11/25] Add wood combustion rate --- .../RollingStocks/MSTSSteamLocomotive.cs | 50 +++++++++++++------ 1 file changed, 36 insertions(+), 14 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 4b62c7881..6fc314945 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -294,9 +294,11 @@ public class MSTSSteamLocomotive : MSTSLocomotive SmoothedData FuelRateOil = new SmoothedData(3); // Oil Stoker is more responsive and only takes x seconds to fully react to changing needs. SmoothedData FuelRateStoker = new SmoothedData(15); // Stoker is more responsive and only takes x seconds to fully react to changing needs. SmoothedData FuelRateCoal = new SmoothedData(45); // Automatic fireman takes x seconds to fully react to changing needs for coal firing. - SmoothedData CoalBurnRateSmoothKGpS = new SmoothedData(150); // Changes in BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. - SmoothedData OilBurnRateSmoothKGpS = new SmoothedData(6); // Changes in Oil BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. Oil faster then steam - SmoothedData OilBurnRateReductionSmoothKGpS = new SmoothedData(0.25f); // When in reduction we would expect the heat to drop off rapidly for an oil fire + SmoothedData BurnRateCoalSmoothKGpS = new SmoothedData(150); // Changes in BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. + SmoothedData BurnRateOilSmoothKGpS = new SmoothedData(6); // Changes in Oil BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. Oil faster then steam + SmoothedData BurnRateWoodSmoothKGpS = new SmoothedData(20); // Changes in Wood BurnRate take x seconds to fully react to changing needs - models increase and decrease in heat. + + SmoothedData BurnRateOilReductionSmoothKGpS = new SmoothedData(0.25f); // When in reduction we would expect the heat to drop off rapidly for an oil fire float FuelFeedRateSmoothedKGpS = 0.0f; // Smoothed Fuel feedd Rate public float FuelBurnRateSmoothedKGpS; // Smoothed fuel burning rate @@ -1228,8 +1230,9 @@ public override void Restore(BinaryReader inf) ControllerFactory.Restore(SmallEjectorController, inf); ControllerFactory.Restore(LargeEjectorController, inf); FuelBurnRateSmoothedKGpS = inf.ReadSingle(); - CoalBurnRateSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); - OilBurnRateSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + BurnRateCoalSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + BurnRateOilSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + BurnRateWoodSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); BoilerHeatSmoothedBTU = inf.ReadSingle(); BoilerHeatSmoothBTU.ForceSmoothValue(BoilerHeatSmoothedBTU); FuelFeedRateSmoothedKGpS = inf.ReadSingle(); @@ -3938,28 +3941,33 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) oilburnrate = MaxFiringRateKGpS; // burning rate can never be more then the max firing rate } - OilBurnRateReductionSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); - OilBurnRateSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); + BurnRateOilReductionSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); + BurnRateOilSmoothKGpS.Update(elapsedClockSeconds, oilburnrate); if (previousThrottle < throttle) { // Fire combustion drops rapidly if throttle closed (assume fuel feed is reduced rapidly) - FuelBurnRateSmoothedKGpS = OilBurnRateReductionSmoothKGpS.SmoothedValue; - OilBurnRateSmoothKGpS.ForceSmoothValue(oilburnrate); + FuelBurnRateSmoothedKGpS = BurnRateOilReductionSmoothKGpS.SmoothedValue; + BurnRateOilSmoothKGpS.ForceSmoothValue(oilburnrate); } else { // Fire combustion is normal - FuelBurnRateSmoothedKGpS = OilBurnRateSmoothKGpS.SmoothedValue; + FuelBurnRateSmoothedKGpS = BurnRateOilSmoothKGpS.SmoothedValue; } previousThrottle = throttle; } + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + BurnRateWoodSmoothKGpS.Update(elapsedClockSeconds, BurnRateRawKGpS); + FuelBurnRateSmoothedKGpS = BurnRateWoodSmoothKGpS.SmoothedValue; + } else { - CoalBurnRateSmoothKGpS.Update(elapsedClockSeconds, BurnRateRawKGpS); - FuelBurnRateSmoothedKGpS = CoalBurnRateSmoothKGpS.SmoothedValue; + BurnRateCoalSmoothKGpS.Update(elapsedClockSeconds, BurnRateRawKGpS); + FuelBurnRateSmoothedKGpS = BurnRateCoalSmoothKGpS.SmoothedValue; } FuelBurnRateSmoothedKGpS = MathHelper.Clamp(FuelBurnRateSmoothedKGpS, 0.0f, MaxFuelBurnGrateKGpS); // clamp burnrate to max fuel that can be burnt within grate limit @@ -4013,7 +4021,14 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) FuelBoost = true; // boost shoveling if (!StokerIsMechanical && IsPlayerTrain) // Don't display message if stoker in operation { - Simulator.Confirmer.Message(ConfirmLevel.Warning, Simulator.Catalog.GetString("FireMass is getting low. Your fireman will shovel faster, but don't wear him out.")); + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + Simulator.Confirmer.Message(ConfirmLevel.Warning, Simulator.Catalog.GetString("FireMass is getting low. Your fireman will add fuel faster, but don't wear him out.")); + } + else + { + Simulator.Confirmer.Message(ConfirmLevel.Warning, Simulator.Catalog.GetString("FireMass is getting low. Your fireman will shovel faster, but don't wear him out.")); + } } } } @@ -4025,7 +4040,14 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) FuelBoost = false; // disable boost shoveling if (!StokerIsMechanical && IsPlayerTrain) // Don't display message if stoker in operation { - Simulator.Confirmer.Message(ConfirmLevel.Warning, Simulator.Catalog.GetString("FireMass is back within limits. Your fireman will shovel as per normal.")); + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + Simulator.Confirmer.Message(ConfirmLevel.Warning, Simulator.Catalog.GetString("FireMass is back within limits. Your fireman will add fuel as normal.")); + } + else + { + Simulator.Confirmer.Message(ConfirmLevel.Warning, Simulator.Catalog.GetString("FireMass is back within limits. Your fireman will shovel as per normal.")); + } } } } From d29622b2dd9ac8be816eda94b1ec9d5e4f99e300 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sun, 23 Jun 2024 16:43:39 +1000 Subject: [PATCH 12/25] Add some defaults parameters --- .../RollingStocks/MSTSSteamLocomotive.cs | 267 +++++++++++------- 1 file changed, 167 insertions(+), 100 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 6fc314945..ba24d888c 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -365,7 +365,7 @@ public class MSTSSteamLocomotive : MSTSLocomotive Interpolator SaturatedSpeedFactorSpeedDropFtpMintoX; // Allowance for drop in TE for a saturated locomotive due to piston speed limitations Interpolator SuperheatedSpeedFactorSpeedDropFtpMintoX; // Allowance for drop in TE for a superheated locomotive due to piston speed limitations - Interpolator NewBurnRateSteamToCoalLbspH; // Combustion rate of steam generated per hour to Dry Coal per hour + Interpolator NewBurnRateSteamToFuelLbspH; // Combustion rate of steam generated per hour to Dry Coal per hour Interpolator2D CutoffInitialPressureDropRatioUpper; // Upper limit of the pressure drop from initial pressure to cut-off pressure Interpolator2D CutoffInitialPressureDropRatioLower; // Lower limit of the pressure drop from initial pressure to cut-off pressure @@ -958,7 +958,7 @@ public override void Parse(string lowercasetoken, STFReader stf) case "engine(ortscylinderefficiencyrate": CylinderEfficiencyRate = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(ortscylinderinitialpressuredrop": InitialPressureDropRatioRpMtoX = new Interpolator(stf); break; case "engine(ortscylinderbackpressure": BackPressureIHPtoPSI = new Interpolator(stf); break; - case "engine(ortsburnrate": NewBurnRateSteamToCoalLbspH = new Interpolator(stf); break; + case "engine(ortsburnrate": NewBurnRateSteamToFuelLbspH = new Interpolator(stf); break; case "engine(ortsboilerefficiency": BoilerEfficiencyGrateAreaLBpFT2toX = new Interpolator(stf); break; case "engine(ortscylindereventexhaust": CylinderExhausttoCutoff = new Interpolator(stf); break; case "engine(ortscylindereventcompression": CylinderCompressiontoCutoff = new Interpolator(stf); break; @@ -1085,7 +1085,7 @@ public override void Copy(MSTSWagon copy) CylinderEfficiencyRate = locoCopy.CylinderEfficiencyRate; InitialPressureDropRatioRpMtoX = new Interpolator(locoCopy.InitialPressureDropRatioRpMtoX); BackPressureIHPtoPSI = new Interpolator(locoCopy.BackPressureIHPtoPSI); - NewBurnRateSteamToCoalLbspH = new Interpolator(locoCopy.NewBurnRateSteamToCoalLbspH); + NewBurnRateSteamToFuelLbspH = new Interpolator(locoCopy.NewBurnRateSteamToFuelLbspH); BoilerEfficiency = locoCopy.BoilerEfficiency; SteamGearRatioLow = locoCopy.SteamGearRatioLow; SteamGearRatioHigh = locoCopy.SteamGearRatioHigh; @@ -1230,13 +1230,26 @@ public override void Restore(BinaryReader inf) ControllerFactory.Restore(SmallEjectorController, inf); ControllerFactory.Restore(LargeEjectorController, inf); FuelBurnRateSmoothedKGpS = inf.ReadSingle(); - BurnRateCoalSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); - BurnRateOilSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); - BurnRateWoodSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); BoilerHeatSmoothedBTU = inf.ReadSingle(); BoilerHeatSmoothBTU.ForceSmoothValue(BoilerHeatSmoothedBTU); FuelFeedRateSmoothedKGpS = inf.ReadSingle(); - FuelRateCoal.ForceSmoothValue(FuelFeedRateSmoothedKGpS); + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + BurnRateWoodSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + FuelRateCoal.ForceSmoothValue(FuelFeedRateSmoothedKGpS); // Wood and coalk rate currently the same + } + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + BurnRateOilSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + BurnRateOilReductionSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + FuelRateOil.ForceSmoothValue(FuelFeedRateSmoothedKGpS); + } + else + { + BurnRateCoalSmoothKGpS.ForceSmoothValue(FuelBurnRateSmoothedKGpS); + FuelRateCoal.ForceSmoothValue(FuelFeedRateSmoothedKGpS); + } + base.Restore(inf); } @@ -1297,6 +1310,16 @@ public override void Initialize() CutoffInitialPressureDropRatioUpper = SteamTable.CutoffInitialPressureUpper(); CutoffInitialPressureDropRatioLower = SteamTable.CutoffInitialPressureLower(); + // Type of fuel selected + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Unknown) + { + SteamLocomotiveFuelType = SteamLocomotiveFuelTypes.Coal; + + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceInformation("SteamLocomotiveFuelType set to Default value of {0}", SteamLocomotiveFuelType); + + } + // Set Oil mass - if an oil locomotive if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil && MaxTenderOilMassL != 0) { @@ -1315,7 +1338,6 @@ public override void Initialize() CylinderExhausttoCutoff = SteamTable.CylinderEventExhausttoCutoff(); } - if (CylinderCompressiontoCutoff == null) { CylinderCompressiontoCutoff = SteamTable.CylinderEventCompressiontoCutoff(); @@ -1326,33 +1348,132 @@ public override void Initialize() CylinderAdmissiontoCutoff = SteamTable.CylinderEventAdmissiontoCutoff(); } - // Assign default steam table values if table not in ENG file - if (BoilerEfficiencyGrateAreaLBpFT2toX == null) + // Computed Values + // Read alternative OR Value for calculation of Ideal Fire Mass + if (GrateAreaM2 == 0) // Calculate Grate Area if not present in ENG file { - if (HasSuperheater) + float MinGrateAreaSizeSqFt = 6.0f; + for (int i = 0; i < SteamEngines.Count; i++) { - BoilerEfficiencyGrateAreaLBpFT2toX = SteamTable.SuperBoilerEfficiencyGrateAreaInterpolatorLbstoX(); - Trace.TraceInformation("BoilerEfficiencyGrateAreaLBpFT2toX (Superheated) - default information read from SteamTables"); + GrateAreaM2 += Me2.FromFt2(((SteamEngines[i].NumberCylinders / 2.0f) * (Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersStrokeM)) * MEPFactor * MaxBoilerPressurePSI) / (Me.ToIn(SteamEngines[i].AttachedAxle.WheelRadiusM * 2.0f) * GrateAreaDesignFactor)); + } + + GrateAreaM2 = MathHelper.Clamp(GrateAreaM2, Me2.FromFt2(MinGrateAreaSizeSqFt), GrateAreaM2); // Clamp grate area to a minimum value of 6 sq ft + IdealFireMassKG = GrateAreaM2 * Me.FromIn(IdealFireDepthIN) * FuelDensityKGpM3; + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("Grate Area not found in ENG file and has been set to {0} m^2", GrateAreaM2); // Advise player that Grate Area is missing from ENG file + } + else + if (LocoIsOilBurner) + IdealFireMassKG = GrateAreaM2 * 720.0f * 0.08333f * 0.02382f * 1.293f; // Check this formula as conversion factors maybe incorrect, also grate area is now in SqM + else + IdealFireMassKG = GrateAreaM2 * Me.FromIn(IdealFireDepthIN) * FuelDensityKGpM3; + + // Calculate the maximum fuel burn rate based upon grate area and limit + float GrateLimitLBpFt2add = GrateLimitLBpFt2 * 1.10f; // Alow burn rate to slightly exceed grate limit (by 10%) + MaxFuelBurnGrateKGpS = pS.FrompH(Kg.FromLb(Me2.ToFt2(GrateAreaM2) * GrateLimitLBpFt2add)); + if (MaxFireMassKG == 0) // If not specified, assume twice as much as ideal. + // Scale FIREBOX control to show FireMassKG as fraction of MaxFireMassKG. + MaxFireMassKG = 2 * IdealFireMassKG; + + float baseTempK = C.ToK(C.FromF(PressureToTemperaturePSItoF[MaxBoilerPressurePSI])); + if (EvaporationAreaM2 == 0) // If evaporation Area is not in ENG file then synthesize a value + { + float MinEvaporationAreaM2 = 100.0f; + for (int i = 0; i < SteamEngines.Count; i++) + { + EvaporationAreaM2 = Me2.FromFt2(((SteamEngines[i].NumberCylinders / 2.0f) * (Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersStrokeM)) * MEPFactor * MaxBoilerPressurePSI) / (Me.ToIn(SteamEngines[i].AttachedAxle.WheelRadiusM * 2.0f) * EvapAreaDesignFactor)); + } + EvaporationAreaM2 = MathHelper.Clamp(EvaporationAreaM2, Me2.FromFt2(MinEvaporationAreaM2), EvaporationAreaM2); // Clamp evaporation area to a minimum value of 6 sq ft, so that NaN values don't occur. + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("Evaporation Area not found in ENG file and has been set to {0} m^2", EvaporationAreaM2); // Advise player that Evaporation Area is missing from ENG file + } + + CylinderSteamUsageLBpS = 1.0f; // Set to 1 to ensure that there are no divide by zero errors + WaterFraction = 0.9f; // Initialise boiler water level at 90% + + float MaxWaterFraction = 0.9f; // Initialise the max water fraction when the boiler starts + + + if (BoilerEvapRateLbspFt2 == 0) // If boiler evaporation rate is not in ENG file then set a default value + { + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + BoilerEvapRateLbspFt2 = 11.5f; // Default rate for evaporation rate. Assume a default rate of 12 lbs/sqft of evaporation area + } + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + BoilerEvapRateLbspFt2 = 18.0f; // Default rate for evaporation rate. Assume a default rate of 18 lbs/sqft of evaporation area } else { - BoilerEfficiencyGrateAreaLBpFT2toX = SteamTable.SatBoilerEfficiencyGrateAreaInterpolatorLbstoX(); - Trace.TraceInformation("BoilerEfficiencyGrateAreaLBpFT2toX (Saturated) - default information read from SteamTables"); + BoilerEvapRateLbspFt2 = 15.0f; // Default rate for evaporation rate. Assume a default rate of 15 lbs/sqft of evaporation area } + } + BoilerEvapRateLbspFt2 = MathHelper.Clamp(BoilerEvapRateLbspFt2, 7.5f, 30.0f); // Clamp BoilerEvap Rate to between 7.5 & 30 - some modern locomotives can go as high as 30, but majority are around 15. + TheoreticalMaxSteamOutputLBpS = pS.FrompH(Me2.ToFt2(EvaporationAreaM2) * BoilerEvapRateLbspFt2); // set max boiler theoretical steam output + float BoilerVolumeCheck = Me2.ToFt2(EvaporationAreaM2) / BoilerVolumeFT3; //Calculate the Boiler Volume Check value. + if (BoilerVolumeCheck > 15) // If boiler volume is not in ENG file or less then a viable figure (ie high ratio figure), then set to a default value + { + BoilerVolumeFT3 = Me2.ToFt2(EvaporationAreaM2) / 8.3f; // Default rate for evaporation rate. Assume a default ratio of evaporation area * 1/8.3 + // Advise player that Boiler Volume is missing from or incorrect in ENG file + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("Boiler Volume not found in ENG file, or doesn't appear to be a valid figure, and has been set to {0} Ft^3", BoilerVolumeFT3); } - // Type of fuel selected - if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Unknown) + + // Assign default steam table values if table not in ENG file + if (BoilerEfficiencyGrateAreaLBpFT2toX == null) { - SteamLocomotiveFuelType = SteamLocomotiveFuelTypes.Coal; + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { - if (Simulator.Settings.VerboseConfigurationMessages) - Trace.TraceInformation("SteamLocomotiveFuelType set to Default value of {0}", SteamLocomotiveFuelType); + // This will calculate a default burnrate curve based upon the fuel calorific, Max Steam Output, and Boiler Efficiency + // Firing Rate = (Max Evap / BE) x (Steam Btu/lb @ pressure / Fuel Calorific) + + float TempBoilerEfficiencyBurnRate = (0.94f / 2.0f); + float TempMaxFiringRateLbpH = (pS.TopH(TheoreticalMaxSteamOutputLBpS) / TempBoilerEfficiencyBurnRate) * (SteamHeatPSItoBTUpLB[MaxBoilerPressurePSI] / KJpKg.ToBTUpLb(FuelCalorificKJpKG)); + + // Create a new default boiler efficiency curve based upon default information + + float[] TempBoilerEfficiencyRate = new float[] { 0.94f, 0.47f }; + + float[] TempGrateFiringRate = new float[] { 0.0f, TempMaxFiringRateLbpH / Me2.ToFt2(GrateAreaM2) }; + + BoilerEfficiencyGrateAreaLBpFT2toX = new Interpolator(TempGrateFiringRate, TempBoilerEfficiencyRate); + + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceInformation("BoilerEfficiencyGrateAreaLBpFT2toX (Wood) - default information table created"); + + } + else + { + + if (HasSuperheater) + { + BoilerEfficiencyGrateAreaLBpFT2toX = SteamTable.SuperBoilerEfficiencyGrateAreaInterpolatorLbstoX(); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceInformation("BoilerEfficiencyGrateAreaLBpFT2toX (Superheated) - default information read from SteamTables"); + + } + else + { + BoilerEfficiencyGrateAreaLBpFT2toX = SteamTable.SatBoilerEfficiencyGrateAreaInterpolatorLbstoX(); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceInformation("BoilerEfficiencyGrateAreaLBpFT2toX (Saturated) - default information read from SteamTables"); + } + } } + // This will calculate a default burnrate curve based upon the fuel calorific, Max Steam Output, and Boiler Efficiency + // Firing Rate = (Max Evap / BE) x (Steam Btu/lb @ pressure / Fuel Calorific) + + float BoilerEfficiencyBurnRate = (BoilerEfficiencyGrateAreaLBpFT2toX[0.0f] / 2.0f); + MaxFiringRateLbpH = (pS.TopH(TheoreticalMaxSteamOutputLBpS) / BoilerEfficiencyBurnRate) * (SteamHeatPSItoBTUpLB[MaxBoilerPressurePSI] / KJpKg.ToBTUpLb(FuelCalorificKJpKG)); + // Calculate Grate Limit // Rule of thumb indicates that Grate limit occurs when the Boiler Efficiency is equal to 50% of the BE at zero firing rate // http://5at.co.uk/index.php/definitions/terrms-and-definitions/grate-limit.html @@ -1387,7 +1508,8 @@ public override void Initialize() if (SteamEngineType != SteamEngineTypes.Geared) { IsGearAssumed = true; - Trace.TraceWarning("Geared locomotive parameter not defined. Geared locomotive has been assumed"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("Geared locomotive parameter not defined. Geared locomotive has been assumed"); } } @@ -1548,12 +1670,14 @@ public override void Initialize() if (MaxSteamGearPistonRateFtpM == 0) { MaxSteamGearPistonRateFtpM = 700.0f; // Assume same value as standard steam locomotive - Trace.TraceWarning("MaxSteamGearPistonRateRpM not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("MaxSteamGearPistonRateRpM not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); } if (SteamGearRatioLow == 0) { SteamGearRatioLow = 5.0f; - Trace.TraceWarning("SteamGearRatioLow not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("SteamGearRatioLow not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); } MotiveForceGearRatio = SteamGearRatioLow; SteamGearRatio = SteamGearRatioLow; @@ -1583,17 +1707,20 @@ public override void Initialize() if (MaxSteamGearPistonRateFtpM == 0) { MaxSteamGearPistonRateFtpM = 500.0f; - Trace.TraceWarning("MaxSteamGearPistonRateRpM not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("MaxSteamGearPistonRateRpM not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); } if (SteamGearRatioLow == 0) { SteamGearRatioLow = 9.0f; - Trace.TraceWarning("SteamGearRatioLow not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("SteamGearRatioLow not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); } if (SteamGearRatioHigh == 0) { SteamGearRatioHigh = 4.5f; - Trace.TraceWarning("SteamGearRatioHigh not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("SteamGearRatioHigh not found in ENG file, or doesn't appear to be a valid figure, and has been set to default value"); } // Adjust resistance for neutral gearing GearedRetainedDavisAN = DavisAN; // remember davis a value for later @@ -1638,7 +1765,8 @@ public override void Initialize() } else // Default to Simple Locomotive (Assumed Simple) shows up as "Unknown" { - Trace.TraceWarning("Steam engine type parameter not formally defined. Simple locomotive has been assumed"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceWarning("Steam engine type parameter not formally defined. Simple locomotive has been assumed"); SteamLocoType = "Not formally defined (assumed simple) locomotive."; // SteamEngineType += "Simple"; MotiveForceGearRatio = 1.0f; // set gear ratio to default, as not a geared locomotive @@ -1683,74 +1811,18 @@ public override void Initialize() if (HasSuperheater) { InitialPressureDropRatioRpMtoX = SteamTable.SuperInitialPressureDropRatioInterpolatorRpMtoX(); - Trace.TraceInformation("InitialPressureDropRatioRpMtoX (Superheated) - default information read from SteamTables"); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceInformation("InitialPressureDropRatioRpMtoX (Superheated) - default information read from SteamTables"); } else { InitialPressureDropRatioRpMtoX = SteamTable.SatInitialPressureDropRatioInterpolatorRpMtoX(); - Trace.TraceInformation("InitialPressureDropRatioRpMtoX (Saturated) - default information read from SteamTables"); - } - - } - - // Computed Values - // Read alternative OR Value for calculation of Ideal Fire Mass - if (GrateAreaM2 == 0) // Calculate Grate Area if not present in ENG file - { - float MinGrateAreaSizeSqFt = 6.0f; - for (int i = 0; i < SteamEngines.Count; i++) - { - GrateAreaM2 += Me2.FromFt2(((SteamEngines[i].NumberCylinders / 2.0f) * (Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersStrokeM)) * MEPFactor * MaxBoilerPressurePSI) / (Me.ToIn(SteamEngines[i].AttachedAxle.WheelRadiusM * 2.0f) * GrateAreaDesignFactor)); - } - - GrateAreaM2 = MathHelper.Clamp(GrateAreaM2, Me2.FromFt2(MinGrateAreaSizeSqFt), GrateAreaM2); // Clamp grate area to a minimum value of 6 sq ft - IdealFireMassKG = GrateAreaM2 * Me.FromIn(IdealFireDepthIN) * FuelDensityKGpM3; - Trace.TraceWarning("Grate Area not found in ENG file and has been set to {0} m^2", GrateAreaM2); // Advise player that Grate Area is missing from ENG file - } - else - if (LocoIsOilBurner) - IdealFireMassKG = GrateAreaM2 * 720.0f * 0.08333f * 0.02382f * 1.293f; // Check this formula as conversion factors maybe incorrect, also grate area is now in SqM - else - IdealFireMassKG = GrateAreaM2 * Me.FromIn(IdealFireDepthIN) * FuelDensityKGpM3; - - // Calculate the maximum fuel burn rate based upon grate area and limit - float GrateLimitLBpFt2add = GrateLimitLBpFt2 * 1.10f; // Alow burn rate to slightly exceed grate limit (by 10%) - MaxFuelBurnGrateKGpS = pS.FrompH(Kg.FromLb(Me2.ToFt2(GrateAreaM2) * GrateLimitLBpFt2add)); - - if (MaxFireMassKG == 0) // If not specified, assume twice as much as ideal. - // Scale FIREBOX control to show FireMassKG as fraction of MaxFireMassKG. - MaxFireMassKG = 2 * IdealFireMassKG; - - float baseTempK = C.ToK(C.FromF(PressureToTemperaturePSItoF[MaxBoilerPressurePSI])); - if (EvaporationAreaM2 == 0) // If evaporation Area is not in ENG file then synthesize a value - { - float MinEvaporationAreaM2 = 100.0f; - for (int i = 0; i < SteamEngines.Count; i++) - { - EvaporationAreaM2 = Me2.FromFt2(((SteamEngines[i].NumberCylinders / 2.0f) * (Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersDiameterM) * Me.ToIn(SteamEngines[i].CylindersStrokeM)) * MEPFactor * MaxBoilerPressurePSI) / (Me.ToIn(SteamEngines[i].AttachedAxle.WheelRadiusM * 2.0f) * EvapAreaDesignFactor)); + if (Simulator.Settings.VerboseConfigurationMessages) + Trace.TraceInformation("InitialPressureDropRatioRpMtoX (Saturated) - default information read from SteamTables"); } - EvaporationAreaM2 = MathHelper.Clamp(EvaporationAreaM2, Me2.FromFt2(MinEvaporationAreaM2), EvaporationAreaM2); // Clamp evaporation area to a minimum value of 6 sq ft, so that NaN values don't occur. - Trace.TraceWarning("Evaporation Area not found in ENG file and has been set to {0} m^2", EvaporationAreaM2); // Advise player that Evaporation Area is missing from ENG file - } - CylinderSteamUsageLBpS = 1.0f; // Set to 1 to ensure that there are no divide by zero errors - WaterFraction = 0.9f; // Initialise boiler water level at 90% - - float MaxWaterFraction = 0.9f; // Initialise the max water fraction when the boiler starts - if (BoilerEvapRateLbspFt2 == 0) // If boiler evaporation rate is not in ENG file then set a default value - { - BoilerEvapRateLbspFt2 = 15.0f; // Default rate for evaporation rate. Assume a default rate of 15 lbs/sqft of evaporation area } - BoilerEvapRateLbspFt2 = MathHelper.Clamp(BoilerEvapRateLbspFt2, 7.5f, 30.0f); // Clamp BoilerEvap Rate to between 7.5 & 30 - some modern locomotives can go as high as 30, but majority are around 15. - TheoreticalMaxSteamOutputLBpS = pS.FrompH(Me2.ToFt2(EvaporationAreaM2) * BoilerEvapRateLbspFt2); // set max boiler theoretical steam output - float BoilerVolumeCheck = Me2.ToFt2(EvaporationAreaM2) / BoilerVolumeFT3; //Calculate the Boiler Volume Check value. - if (BoilerVolumeCheck > 15) // If boiler volume is not in ENG file or less then a viable figure (ie high ratio figure), then set to a default value - { - BoilerVolumeFT3 = Me2.ToFt2(EvaporationAreaM2) / 8.3f; // Default rate for evaporation rate. Assume a default ratio of evaporation area * 1/8.3 - // Advise player that Boiler Volume is missing from or incorrect in ENG file - Trace.TraceWarning("Boiler Volume not found in ENG file, or doesn't appear to be a valid figure, and has been set to {0} Ft^3", BoilerVolumeFT3); - } MaxBoilerHeatSafetyPressurePSI = MaxBoilerPressurePSI + SafetyValveStartPSI + 6.0f; // set locomotive maximum boiler pressure to calculate max heat, allow for safety valve + a bit MaxBoilerSafetyPressHeatBTU = MaxWaterFraction * BoilerVolumeFT3 * WaterDensityPSItoLBpFT3[MaxBoilerHeatSafetyPressurePSI] * WaterHeatPSItoBTUpLB[MaxBoilerHeatSafetyPressurePSI] + (1 - MaxWaterFraction) * BoilerVolumeFT3 * SteamDensityPSItoLBpFT3[MaxBoilerHeatSafetyPressurePSI] * SteamHeatPSItoBTUpLB[MaxBoilerHeatSafetyPressurePSI]; // calculate the maximum possible heat in the boiler, assuming safety valve and a small margin @@ -1855,13 +1927,8 @@ public override void Initialize() } // Assign default steam table values if table not in ENG file - if (NewBurnRateSteamToCoalLbspH == null) + if (NewBurnRateSteamToFuelLbspH == null) { - // This will calculate a default burnrate curve based upon the fuel calorific, Max Steam Output, and Boiler Efficiency - // Firing Rate = (Max Evap / BE) x (Steam Btu/lb @ pressure / Fuel Calorific) - - float BoilerEfficiencyBurnRate = (BoilerEfficiencyGrateAreaLBpFT2toX[0.0f] / 2.0f); - MaxFiringRateLbpH = (pS.TopH(TheoreticalMaxSteamOutputLBpS) / BoilerEfficiencyBurnRate) * (SteamHeatPSItoBTUpLB[MaxBoilerPressurePSI] / KJpKg.ToBTUpLb(FuelCalorificKJpKG)); // Create a new default burnrate curve locomotive based upon default information float[] TempSteamOutputRate = new float[] @@ -1874,11 +1941,11 @@ public override void Initialize() 0.0f, MaxFiringRateLbpH, (MaxFiringRateLbpH * 1.5f), (MaxFiringRateLbpH * 2.0f), (MaxFiringRateLbpH * 3.0f), (MaxFiringRateLbpH * 4.0f) }; - NewBurnRateSteamToCoalLbspH = new Interpolator(TempSteamOutputRate, TempCoalFiringRate); + NewBurnRateSteamToFuelLbspH = new Interpolator(TempSteamOutputRate, TempCoalFiringRate); } else // If user provided burn curve calculate the Max Firing Rate { - MaxFiringRateLbpH = NewBurnRateSteamToCoalLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)]; + MaxFiringRateLbpH = NewBurnRateSteamToFuelLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)]; } // Calculate Boiler output horsepower - based upon information in Johnsons book - The Steam Locomotive - pg 150 @@ -2142,7 +2209,7 @@ public override void Initialize() MaxCriticalSpeedTractiveEffortLbf = (MaxTractiveEffortLbf * CylinderEfficiencyRate) * MaxSpeedFactor; DisplayCriticalSpeedTractiveEffortLbf = MaxCriticalSpeedTractiveEffortLbf; - MaxCombustionRateKgpS = pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); + MaxCombustionRateKgpS = pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); // Calculate the maximum boiler heat input based on the steam generation rate @@ -3647,7 +3714,7 @@ private void UpdateTender(float elapsedClockSeconds) if (HasTenderCoupled) // If a tender is coupled then coal is available { - TenderCoalMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. + TenderCoalMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. TenderCoalMassKG = MathHelper.Clamp(TenderCoalMassKG, 0, MaxTenderCoalMassKG); // Clamp value so that it doesn't go out of bounds } else // if no tender coupled then check whether a tender is required @@ -3658,7 +3725,7 @@ private void UpdateTender(float elapsedClockSeconds) } else // Tender is not required (ie tank locomotive) - therefore coal will be carried on the locomotive { - TenderCoalMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. + TenderCoalMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. TenderCoalMassKG = MathHelper.Clamp(TenderCoalMassKG, 0, MaxTenderCoalMassKG); // Clamp value so that it doesn't go out of bounds } } @@ -3798,7 +3865,7 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) { // Manual Firing - a small burning effect is maintained by the Radiation Steam Loss. The Blower is designed to be used when stationary, or if required when regulator is closed // The exhaust steam from the cylinders drives the draught through the firebox, the damper is used to reduce (or increase) the draft as required. - BurnRateRawKGpS = pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH((RadiationSteamLossLBpS + CalculatedCarHeaterSteamUsageLBpS) + BlowerBurnEffect + DamperBurnEffect)])); + BurnRateRawKGpS = pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH((RadiationSteamLossLBpS + CalculatedCarHeaterSteamUsageLBpS) + BlowerBurnEffect + DamperBurnEffect)])); } else // *********** AI Fireman ***************** { @@ -3873,7 +3940,7 @@ private void UpdateFirebox(float elapsedClockSeconds, float absSpeedMpS) { SetFireOn = false; // Disable FireOn if bolierpressure and boilerheat back to "normal" } - BurnRateRawKGpS = 0.9f * pS.FrompH(Kg.FromLb(NewBurnRateSteamToCoalLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); // AI fire on goes to approx 100% of fire needed to maintain full boiler steam generation + BurnRateRawKGpS = 0.9f * pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TheoreticalMaxSteamOutputLBpS)])); // AI fire on goes to approx 100% of fire needed to maintain full boiler steam generation } } From e375e56171b3a020da25515061145232a39683f2 Mon Sep 17 00:00:00 2001 From: peternewell Date: Tue, 25 Jun 2024 17:08:04 +1000 Subject: [PATCH 13/25] Add water motion pump --- .../RollingStocks/MSTSSteamLocomotive.cs | 605 ++++++++++-------- 1 file changed, 346 insertions(+), 259 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index ba24d888c..3c0ae98ce 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -115,6 +115,13 @@ public class MSTSSteamLocomotive : MSTSLocomotive bool Injector1SoundIsOn = false; public bool Injector2IsOn; bool Injector2SoundIsOn = false; + bool WaterMotionPumpFitted = false; + float WaterMotionPump1FlowRateLBpS; + float WaterMotionPump2FlowRateLBpS; + float MaximumWaterMotionPumpFlowRateLBpS; + bool WaterMotionPump1IsOn = false; + bool WaterMotionPump2IsOn = false; + float WaterMotionPumpHeatLossBTU; public bool CylinderCocksAreOpen; public bool BlowdownValveOpen; public bool CylinderCompoundOn; // Flag to indicate whether compound locomotive is in compound or simple mode of operation - simple = true (ie bypass valve is open) @@ -934,6 +941,11 @@ public override void Parse(string lowercasetoken, STFReader stf) if (heating == 1) FuelOilSteamHeatingReqd = true; break; + case "engine(ortswatermotionpump": + var motionpump = stf.ReadIntBlock(null); + if (motionpump == 1) + WaterMotionPumpFitted = true; + break; case "engine(ortsfueloilspecificgravity": OilSpecificGravity = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(enginecontrollers(cutoff": CutoffController.Parse(stf); break; case "engine(enginecontrollers(ortssmallejector": SmallEjectorController.Parse(stf); SmallEjectorControllerFitted = true; break; @@ -1048,6 +1060,7 @@ public override void Copy(MSTSWagon copy) CylinderPortOpeningFactor = locoCopy.CylinderPortOpeningFactor; BoilerVolumeFT3 = locoCopy.BoilerVolumeFT3; MaxBoilerPressurePSI = locoCopy.MaxBoilerPressurePSI; + WaterMotionPumpFitted = locoCopy.WaterMotionPumpFitted; MaxSuperheatRefTempF = locoCopy.MaxSuperheatRefTempF; MaxIndicatedHorsePowerHP = locoCopy.MaxIndicatedHorsePowerHP; SuperheatCutoffPressureFactor = locoCopy.SuperheatCutoffPressureFactor; @@ -7002,144 +7015,180 @@ private void UpdateWaterGauge() private void UpdateInjectors(float elapsedClockSeconds) { - #region Calculate Injector size - // Calculate size of injectors to suit cylinder size. - for (int i = 0; i < SteamEngines.Count; i++) + if (WaterMotionPumpFitted) { - InjCylEquivSizeIN = (SteamEngines[i].NumberCylinders / 2.0f) * Me.ToIn(SteamEngines[i].CylindersDiameterM); - } + MaximumWaterMotionPumpFlowRateLBpS = (1.2f * EvaporationLBpS) / 2.0f; // Assume two pumps and that they can pump a fraction more water the the maximum steam production - // Based on equiv cyl size determine correct size injector - if (InjCylEquivSizeIN <= 19.0 && (2.0f * (pS.TopH(pS.FrompM(Injector09FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) - { - MaxInjectorFlowRateLBpS = pS.FrompM(Injector09FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 9mm Injector maximum flow rate @ maximm boiler pressure - InjectorFlowRateLBpS = pS.FrompM(Injector09FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 9mm Injector Flow rate - InjectorSize = 09.0f; // store size for display in HUD - } - else if (InjCylEquivSizeIN <= 24.0 && (2.0f * (pS.TopH(pS.FrompM(Injector10FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) - { - MaxInjectorFlowRateLBpS = pS.FrompM(Injector10FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 10mm Injector maximum flow rate @ maximm boiler pressure - InjectorFlowRateLBpS = pS.FrompM(Injector10FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 10 mm Injector Flow rate - InjectorSize = 10.0f; // store size for display in HUD - } - else if (InjCylEquivSizeIN <= 26.0 && (2.0f * (pS.TopH(pS.FrompM(Injector11FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) - { - MaxInjectorFlowRateLBpS = pS.FrompM(Injector11FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 11mm Injector maximum flow rate @ maximm boiler pressure - InjectorFlowRateLBpS = pS.FrompM(Injector11FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 11 mm Injector Flow rate - InjectorSize = 11.0f; // store size for display in HUD - } - else if (InjCylEquivSizeIN <= 28.0 && (2.0f * (pS.TopH(pS.FrompM(Injector13FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) - { - MaxInjectorFlowRateLBpS = pS.FrompM(Injector13FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 13mm Injector maximum flow rate @ maximm boiler pressure - InjectorFlowRateLBpS = pS.FrompM(Injector13FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 13 mm Injector Flow rate - InjectorSize = 13.0f; // store size for display in HUD - } - else if (InjCylEquivSizeIN <= 30.0 && (2.0f * (pS.TopH(pS.FrompM(Injector14FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) - { - MaxInjectorFlowRateLBpS = pS.FrompM(Injector14FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 14mm Injector maximum flow rate @ maximm boiler pressure - InjectorFlowRateLBpS = pS.FrompM(Injector14FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 14 mm Injector Flow rate - InjectorSize = 14.0f; // store size for display in HUD - } - else - { - MaxInjectorFlowRateLBpS = pS.FrompM(Injector15FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 15mm Injector maximum flow rate @ maximm boiler pressure - InjectorFlowRateLBpS = pS.FrompM(Injector15FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 15 mm Injector Flow rate - InjectorSize = 15.0f; // store size for display in HUD - } - #endregion + if (WaterMotionPump1IsOn && absSpeedMpS > 0) + { + WaterMotionPump1FlowRateLBpS = MaximumWaterMotionPumpFlowRateLBpS * absSpeedMpS / MpS.FromMpH(MaxLocoSpeedMpH); + } - if (WaterIsExhausted) - { - InjectorFlowRateLBpS = 0.0f; // If the tender water is empty, stop flow into boiler - } + if (WaterMotionPump2IsOn && absSpeedMpS > 0) + { + WaterMotionPump2FlowRateLBpS = MaximumWaterMotionPumpFlowRateLBpS * absSpeedMpS / MpS.FromMpH(MaxLocoSpeedMpH); + } - InjectorBoilerInputLB = 0; // Used by UpdateTender() later in the cycle - if (WaterIsExhausted) - { - // don't fill boiler with injectors + if (WaterIsExhausted) + { + WaterMotionPump1FlowRateLBpS = 0.0f; // If the tender water is empty, stop flow into boiler + WaterMotionPump2FlowRateLBpS = 0.0f; // If the tender water is empty, stop flow into boiler + } + + float TotalPumpFlowRateLbpS = WaterMotionPump1FlowRateLBpS + WaterMotionPump2FlowRateLBpS; + + // Calculate heat loss for water injected + // Loss of boiler heat due to water injection - loss is the diff between steam and water Heat + WaterMotionPumpHeatLossBTU = TotalPumpFlowRateLbpS * (WaterHeatPSItoBTUpLB[BoilerPressurePSI] - WaterHeatPSItoBTUpLB[0]); + + // calculate Water steam heat based on injector water delivery temp + BoilerMassLB += elapsedClockSeconds * TotalPumpFlowRateLbpS; // Boiler Mass increase by Injector both pumps + BoilerHeatBTU -= elapsedClockSeconds * WaterMotionPumpHeatLossBTU; // Total loss of boiler heat due to water injection - inject steam and water Heat +// InjectorBoilerInputLB += (elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 + BoilerHeatOutBTUpS += WaterMotionPumpHeatLossBTU; // Total loss of boiler heat due to water injection - inject steam and water Heat } else { - // Injectors to fill boiler - if (Injector1IsOn) - { - // Calculate Injector 1 delivery water temp - if (Injector1Fraction < InjCapMinFactorX[BoilerPressurePSI]) - { - Injector1WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI]; // set water delivery temp to minimum value - } - else - { - Injector1TempFraction = (Injector1Fraction - InjCapMinFactorX[BoilerPressurePSI]) / (1 - InjCapMinFactorX[MaxBoilerPressurePSI]); // Find the fraction above minimum value - Injector1WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - ((InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - InjDelWaterTempMaxPressureFtoPSI[BoilerPressurePSI]) * Injector1TempFraction); - Injector1WaterDelTempF = MathHelper.Clamp(Injector1WaterDelTempF, 65.0f, 500.0f); - } + #region Calculate Injector size - Injector1WaterTempPressurePSI = WaterTempFtoPSI[Injector1WaterDelTempF]; // calculate the pressure of the delivery water - - // Calculate amount of steam used to inject water - MaxInject1SteamUsedLbpS = InjWaterFedSteamPressureFtoPSI[BoilerPressurePSI]; // Maximum amount of steam used at actual boiler pressure - ActInject1SteamUsedLbpS = (Injector1Fraction * InjectorFlowRateLBpS) / MaxInject1SteamUsedLbpS; // Lbs of steam injected into boiler to inject water. + // Calculate size of injectors to suit cylinder size. + for (int i = 0; i < SteamEngines.Count; i++) + { + InjCylEquivSizeIN = (SteamEngines[i].NumberCylinders / 2.0f) * Me.ToIn(SteamEngines[i].CylindersDiameterM); + } - // Calculate heat loss for steam injection - Inject1SteamHeatLossBTU = ActInject1SteamUsedLbpS * (BoilerSteamHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector1WaterTempPressurePSI]); // Calculate heat loss for injection steam, ie steam heat to water delivery temperature + // Based on equiv cyl size determine correct size injector + if (InjCylEquivSizeIN <= 19.0 && (2.0f * (pS.TopH(pS.FrompM(Injector09FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) + { + MaxInjectorFlowRateLBpS = pS.FrompM(Injector09FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 9mm Injector maximum flow rate @ maximm boiler pressure + InjectorFlowRateLBpS = pS.FrompM(Injector09FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 9mm Injector Flow rate + InjectorSize = 09.0f; // store size for display in HUD + } + else if (InjCylEquivSizeIN <= 24.0 && (2.0f * (pS.TopH(pS.FrompM(Injector10FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) + { + MaxInjectorFlowRateLBpS = pS.FrompM(Injector10FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 10mm Injector maximum flow rate @ maximm boiler pressure + InjectorFlowRateLBpS = pS.FrompM(Injector10FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 10 mm Injector Flow rate + InjectorSize = 10.0f; // store size for display in HUD + } + else if (InjCylEquivSizeIN <= 26.0 && (2.0f * (pS.TopH(pS.FrompM(Injector11FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) + { + MaxInjectorFlowRateLBpS = pS.FrompM(Injector11FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 11mm Injector maximum flow rate @ maximm boiler pressure + InjectorFlowRateLBpS = pS.FrompM(Injector11FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 11 mm Injector Flow rate + InjectorSize = 11.0f; // store size for display in HUD + } + else if (InjCylEquivSizeIN <= 28.0 && (2.0f * (pS.TopH(pS.FrompM(Injector13FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) + { + MaxInjectorFlowRateLBpS = pS.FrompM(Injector13FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 13mm Injector maximum flow rate @ maximm boiler pressure + InjectorFlowRateLBpS = pS.FrompM(Injector13FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 13 mm Injector Flow rate + InjectorSize = 13.0f; // store size for display in HUD + } + else if (InjCylEquivSizeIN <= 30.0 && (2.0f * (pS.TopH(pS.FrompM(Injector14FlowratePSItoUKGpM[MaxBoilerPressurePSI])) * WaterLBpUKG)) > MaxBoilerOutputLBpH) + { + MaxInjectorFlowRateLBpS = pS.FrompM(Injector14FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 14mm Injector maximum flow rate @ maximm boiler pressure + InjectorFlowRateLBpS = pS.FrompM(Injector14FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 14 mm Injector Flow rate + InjectorSize = 14.0f; // store size for display in HUD + } + else + { + MaxInjectorFlowRateLBpS = pS.FrompM(Injector15FlowratePSItoUKGpM[MaxBoilerPressurePSI]) * WaterLBpUKG; // 15mm Injector maximum flow rate @ maximm boiler pressure + InjectorFlowRateLBpS = pS.FrompM(Injector15FlowratePSItoUKGpM[BoilerPressurePSI]) * WaterLBpUKG; // 15 mm Injector Flow rate + InjectorSize = 15.0f; // store size for display in HUD + } + #endregion - // Calculate heat loss for water injected - // Loss of boiler heat due to water injection - loss is the diff between steam and water Heat - Inject1WaterHeatLossBTU = Injector1Fraction * InjectorFlowRateLBpS * (BoilerWaterHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector1WaterTempPressurePSI]); + if (WaterIsExhausted) + { + InjectorFlowRateLBpS = 0.0f; // If the tender water is empty, stop flow into boiler + } - // calculate Water steam heat based on injector water delivery temp - BoilerMassLB += elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS; // Boiler Mass increase by Injector 1 - BoilerHeatBTU -= elapsedClockSeconds * (Inject1WaterHeatLossBTU + Inject1SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat - InjectorBoilerInputLB += (elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 - BoilerHeatOutBTUpS += (Inject1WaterHeatLossBTU + Inject1SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat + InjectorBoilerInputLB = 0; // Used by UpdateTender() later in the cycle + if (WaterIsExhausted) + { + // don't fill boiler with injectors } - if (Injector2IsOn) + else { - // Calculate Injector 2 delivery water temp - if (Injector2Fraction < InjCapMinFactorX[BoilerPressurePSI]) + // Injectors to fill boiler + if (Injector1IsOn) { - Injector2WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI]; // set water delivery temp to minimum value + // Calculate Injector 1 delivery water temp + if (Injector1Fraction < InjCapMinFactorX[BoilerPressurePSI]) + { + Injector1WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI]; // set water delivery temp to minimum value + } + else + { + Injector1TempFraction = (Injector1Fraction - InjCapMinFactorX[BoilerPressurePSI]) / (1 - InjCapMinFactorX[MaxBoilerPressurePSI]); // Find the fraction above minimum value + Injector1WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - ((InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - InjDelWaterTempMaxPressureFtoPSI[BoilerPressurePSI]) * Injector1TempFraction); + Injector1WaterDelTempF = MathHelper.Clamp(Injector1WaterDelTempF, 65.0f, 500.0f); + } + + Injector1WaterTempPressurePSI = WaterTempFtoPSI[Injector1WaterDelTempF]; // calculate the pressure of the delivery water + + // Calculate amount of steam used to inject water + MaxInject1SteamUsedLbpS = InjWaterFedSteamPressureFtoPSI[BoilerPressurePSI]; // Maximum amount of steam used at actual boiler pressure + ActInject1SteamUsedLbpS = (Injector1Fraction * InjectorFlowRateLBpS) / MaxInject1SteamUsedLbpS; // Lbs of steam injected into boiler to inject water. + + // Calculate heat loss for steam injection + Inject1SteamHeatLossBTU = ActInject1SteamUsedLbpS * (BoilerSteamHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector1WaterTempPressurePSI]); // Calculate heat loss for injection steam, ie steam heat to water delivery temperature + + // Calculate heat loss for water injected + // Loss of boiler heat due to water injection - loss is the diff between steam and water Heat + Inject1WaterHeatLossBTU = Injector1Fraction * InjectorFlowRateLBpS * (BoilerWaterHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector1WaterTempPressurePSI]); + + // calculate Water steam heat based on injector water delivery temp + BoilerMassLB += elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS; // Boiler Mass increase by Injector 1 + BoilerHeatBTU -= elapsedClockSeconds * (Inject1WaterHeatLossBTU + Inject1SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat + InjectorBoilerInputLB += (elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 + BoilerHeatOutBTUpS += (Inject1WaterHeatLossBTU + Inject1SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat } - else + if (Injector2IsOn) { - Injector2TempFraction = (Injector2Fraction - InjCapMinFactorX[BoilerPressurePSI]) / (1 - InjCapMinFactorX[MaxBoilerPressurePSI]); // Find the fraction above minimum value - Injector2WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - ((InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - InjDelWaterTempMaxPressureFtoPSI[BoilerPressurePSI]) * Injector2TempFraction); - Injector2WaterDelTempF = MathHelper.Clamp(Injector2WaterDelTempF, 65.0f, 500.0f); - } - Injector2WaterTempPressurePSI = WaterTempFtoPSI[Injector2WaterDelTempF]; // calculate the pressure of the delivery water + // Calculate Injector 2 delivery water temp + if (Injector2Fraction < InjCapMinFactorX[BoilerPressurePSI]) + { + Injector2WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI]; // set water delivery temp to minimum value + } + else + { + Injector2TempFraction = (Injector2Fraction - InjCapMinFactorX[BoilerPressurePSI]) / (1 - InjCapMinFactorX[MaxBoilerPressurePSI]); // Find the fraction above minimum value + Injector2WaterDelTempF = InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - ((InjDelWaterTempMinPressureFtoPSI[BoilerPressurePSI] - InjDelWaterTempMaxPressureFtoPSI[BoilerPressurePSI]) * Injector2TempFraction); + Injector2WaterDelTempF = MathHelper.Clamp(Injector2WaterDelTempF, 65.0f, 500.0f); + } + Injector2WaterTempPressurePSI = WaterTempFtoPSI[Injector2WaterDelTempF]; // calculate the pressure of the delivery water - // Calculate amount of steam used to inject water - MaxInject2SteamUsedLbpS = InjWaterFedSteamPressureFtoPSI[BoilerPressurePSI]; // Maximum amount of steam used at boiler pressure - ActInject2SteamUsedLbpS = (Injector2Fraction * InjectorFlowRateLBpS) / MaxInject2SteamUsedLbpS; // Lbs of steam injected into boiler to inject water. + // Calculate amount of steam used to inject water + MaxInject2SteamUsedLbpS = InjWaterFedSteamPressureFtoPSI[BoilerPressurePSI]; // Maximum amount of steam used at boiler pressure + ActInject2SteamUsedLbpS = (Injector2Fraction * InjectorFlowRateLBpS) / MaxInject2SteamUsedLbpS; // Lbs of steam injected into boiler to inject water. - // Calculate heat loss for steam injection - Inject2SteamHeatLossBTU = ActInject2SteamUsedLbpS * (BoilerSteamHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector2WaterTempPressurePSI]); // Calculate heat loss for injection steam, ie steam heat to water delivery temperature + // Calculate heat loss for steam injection + Inject2SteamHeatLossBTU = ActInject2SteamUsedLbpS * (BoilerSteamHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector2WaterTempPressurePSI]); // Calculate heat loss for injection steam, ie steam heat to water delivery temperature - // Calculate heat loss for water injected - Inject2WaterHeatLossBTU = Injector2Fraction * InjectorFlowRateLBpS * (BoilerWaterHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector2WaterTempPressurePSI]); // Loss of boiler heat due to water injection - loss is the diff between steam and water Heat + // Calculate heat loss for water injected + Inject2WaterHeatLossBTU = Injector2Fraction * InjectorFlowRateLBpS * (BoilerWaterHeatBTUpLB - WaterHeatPSItoBTUpLB[Injector2WaterTempPressurePSI]); // Loss of boiler heat due to water injection - loss is the diff between steam and water Heat - // calculate Water steam heat based on injector water delivery temp - BoilerMassLB += elapsedClockSeconds * Injector2Fraction * InjectorFlowRateLBpS; // Boiler Mass increase by Injector 1 - BoilerHeatBTU -= elapsedClockSeconds * (Inject2WaterHeatLossBTU + Inject2SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat - InjectorBoilerInputLB += (elapsedClockSeconds * Injector2Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 - BoilerHeatOutBTUpS += (Inject2WaterHeatLossBTU + Inject2SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat + // calculate Water steam heat based on injector water delivery temp + BoilerMassLB += elapsedClockSeconds * Injector2Fraction * InjectorFlowRateLBpS; // Boiler Mass increase by Injector 1 + BoilerHeatBTU -= elapsedClockSeconds * (Inject2WaterHeatLossBTU + Inject2SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat + InjectorBoilerInputLB += (elapsedClockSeconds * Injector2Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 + BoilerHeatOutBTUpS += (Inject2WaterHeatLossBTU + Inject2SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat + } } - } - // Update injector lockout timer - if (Injector1IsOn || Injector2IsOn) - { - if (InjectorLockedOut) - { - InjectorLockOutTimeS += elapsedClockSeconds; - } - if (InjectorLockOutTimeS > InjectorLockOutResetTimeS) + // Update injector lockout timer + if (Injector1IsOn || Injector2IsOn) { - InjectorLockedOut = false; - InjectorLockOutTimeS = 0.0f; + if (InjectorLockedOut) + { + InjectorLockOutTimeS += elapsedClockSeconds; + } + if (InjectorLockOutTimeS > InjectorLockOutResetTimeS) + { + InjectorLockedOut = false; + InjectorLockOutTimeS = 0.0f; + } } } } @@ -7193,179 +7242,201 @@ private void UpdateFiring(float absSpeedMpS) #region AI Fireman { - // Injectors - // Injectors normally not on when stationary? - // Injector water delivery heat decreases with the capacity of the injectors, ideally one injector would be used as appropriate to match steam consumption. @nd one only used if required. - if (WaterGlassLevelIN > 7.99) // turn injectors off if water level in boiler greater then 8.0, to stop cycling - { - Injector1IsOn = false; - Injector1Fraction = 0.0f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - StopInjector1Sound(); - StopInjector2Sound(); - } - else if (WaterGlassLevelIN <= 7.0 && WaterGlassLevelIN > 6.875 && !InjectorLockedOut) // turn injector 1 on 20% if water level in boiler drops below 7.0 - { - Injector1IsOn = true; - Injector1Fraction = 0.1f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.875 && WaterGlassLevelIN > 6.75 && !InjectorLockedOut) // turn injector 1 on 20% if water level in boiler drops below 7.0 - { - Injector1IsOn = true; - Injector1Fraction = 0.2f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.75 && WaterGlassLevelIN > 6.675 && !InjectorLockedOut) // turn injector 1 on 20% if water level in boiler drops below 7.0 - { - Injector1IsOn = true; - Injector1Fraction = 0.3f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.675 && WaterGlassLevelIN > 6.5 && !InjectorLockedOut) - { - Injector1IsOn = true; - Injector1Fraction = 0.4f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.5 && WaterGlassLevelIN > 6.375 && !InjectorLockedOut) - { - Injector1IsOn = true; - Injector1Fraction = 0.5f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.375 && WaterGlassLevelIN > 6.25 && !InjectorLockedOut) - { - Injector1IsOn = true; - Injector1Fraction = 0.6f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.25 && WaterGlassLevelIN > 6.125 && !InjectorLockedOut) - { - Injector1IsOn = true; - Injector1Fraction = 0.7f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.125 && WaterGlassLevelIN > 6.0 && !InjectorLockedOut) - { - Injector1IsOn = true; - Injector1Fraction = 0.8f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 6.0 && WaterGlassLevelIN > 5.875 && !InjectorLockedOut) - { - Injector1IsOn = true; - Injector1Fraction = 0.9f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (WaterGlassLevelIN <= 5.875 && WaterGlassLevelIN > 5.75 && !InjectorLockedOut) - { - Injector1IsOn = true; - Injector1Fraction = 1.0f; - Injector2IsOn = false; - Injector2Fraction = 0.0f; - InjectorLockedOut = true; - PlayInjector1SoundIfStarting(); - } - else if (BoilerPressurePSI > (MaxBoilerPressurePSI - 100.0)) // If boiler pressure is not too low then turn on injector 2 - { - if (WaterGlassLevelIN <= 5.75 && WaterGlassLevelIN > 5.675 && !InjectorLockedOut) - { - Injector2IsOn = true; - Injector2Fraction = 0.1f; + + if (WaterMotionPumpFitted && !WaterIsExhausted) + { + if (WaterGlassLevelIN > 7.99) // turn pumps off if water level in boiler greater then 8.0, to stop cycling + { + WaterMotionPump1IsOn = false; + WaterMotionPump2IsOn = false; + } + else if (WaterGlassLevelIN <= 7.0 && WaterGlassLevelIN > 5.75) // turn water pump #1 on if water level in boiler drops below 7.0 and is above + { + WaterMotionPump1IsOn = true; + WaterMotionPump2IsOn = false; + } + else if (WaterGlassLevelIN <= 5.75 && WaterGlassLevelIN > 4.5) // turn water pump #1 on if water level in boiler drops below 7.0 and is above + { + WaterMotionPump1IsOn = true; + WaterMotionPump2IsOn = true; + } + } + else + { + // Injectors + // Injectors normally not on when stationary? + // Injector water delivery heat decreases with the capacity of the injectors, ideally one injector would be used as appropriate to match steam consumption. @nd one only used if required. + if (WaterGlassLevelIN > 7.99) // turn injectors off if water level in boiler greater then 8.0, to stop cycling + { + Injector1IsOn = false; + Injector1Fraction = 0.0f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; + StopInjector1Sound(); + StopInjector2Sound(); + } + else if (WaterGlassLevelIN <= 7.0 && WaterGlassLevelIN > 6.875 && !InjectorLockedOut) // turn injector 1 on 20% if water level in boiler drops below 7.0 + { + Injector1IsOn = true; + Injector1Fraction = 0.1f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 5.675 && WaterGlassLevelIN > 5.5 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.875 && WaterGlassLevelIN > 6.75 && !InjectorLockedOut) // turn injector 1 on 20% if water level in boiler drops below 7.0 { - Injector2IsOn = true; - Injector2Fraction = 0.2f; + Injector1IsOn = true; + Injector1Fraction = 0.2f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 5.5 && WaterGlassLevelIN > 5.325 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.75 && WaterGlassLevelIN > 6.675 && !InjectorLockedOut) // turn injector 1 on 20% if water level in boiler drops below 7.0 { - Injector2IsOn = true; - Injector2Fraction = 0.3f; + Injector1IsOn = true; + Injector1Fraction = 0.3f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 5.325 && WaterGlassLevelIN > 5.25 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.675 && WaterGlassLevelIN > 6.5 && !InjectorLockedOut) { - Injector2IsOn = true; - Injector2Fraction = 0.4f; + Injector1IsOn = true; + Injector1Fraction = 0.4f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 5.25 && WaterGlassLevelIN > 5.125 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.5 && WaterGlassLevelIN > 6.375 && !InjectorLockedOut) { - Injector2IsOn = true; - Injector2Fraction = 0.5f; + Injector1IsOn = true; + Injector1Fraction = 0.5f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 5.125 && WaterGlassLevelIN > 5.0 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.375 && WaterGlassLevelIN > 6.25 && !InjectorLockedOut) { - Injector2IsOn = true; - Injector2Fraction = 0.6f; + Injector1IsOn = true; + Injector1Fraction = 0.6f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 5.0 && WaterGlassLevelIN > 4.875 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.25 && WaterGlassLevelIN > 6.125 && !InjectorLockedOut) { - Injector2IsOn = true; - Injector2Fraction = 0.7f; + Injector1IsOn = true; + Injector1Fraction = 0.7f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 4.875 && WaterGlassLevelIN > 4.75 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.125 && WaterGlassLevelIN > 6.0 && !InjectorLockedOut) { - Injector2IsOn = true; - Injector2Fraction = 0.8f; + Injector1IsOn = true; + Injector1Fraction = 0.8f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 4.75 && WaterGlassLevelIN > 4.625 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 6.0 && WaterGlassLevelIN > 5.875 && !InjectorLockedOut) { - Injector2IsOn = true; - Injector2Fraction = 0.9f; + Injector1IsOn = true; + Injector1Fraction = 0.9f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); } - else if (WaterGlassLevelIN <= 4.625 && WaterGlassLevelIN > 4.5 && !InjectorLockedOut) + else if (WaterGlassLevelIN <= 5.875 && WaterGlassLevelIN > 5.75 && !InjectorLockedOut) { - Injector2IsOn = true; - Injector2Fraction = 1.0f; + Injector1IsOn = true; + Injector1Fraction = 1.0f; + Injector2IsOn = false; + Injector2Fraction = 0.0f; InjectorLockedOut = true; - PlayInjector2SoundIfStarting(); + PlayInjector1SoundIfStarting(); + } + else if (BoilerPressurePSI > (MaxBoilerPressurePSI - 100.0)) // If boiler pressure is not too low then turn on injector 2 + { + if (WaterGlassLevelIN <= 5.75 && WaterGlassLevelIN > 5.675 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.1f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 5.675 && WaterGlassLevelIN > 5.5 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.2f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 5.5 && WaterGlassLevelIN > 5.325 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.3f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 5.325 && WaterGlassLevelIN > 5.25 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.4f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 5.25 && WaterGlassLevelIN > 5.125 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.5f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 5.125 && WaterGlassLevelIN > 5.0 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.6f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 5.0 && WaterGlassLevelIN > 4.875 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.7f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 4.875 && WaterGlassLevelIN > 4.75 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.8f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 4.75 && WaterGlassLevelIN > 4.625 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 0.9f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } + else if (WaterGlassLevelIN <= 4.625 && WaterGlassLevelIN > 4.5 && !InjectorLockedOut) + { + Injector2IsOn = true; + Injector2Fraction = 1.0f; + InjectorLockedOut = true; + PlayInjector2SoundIfStarting(); + } } } @@ -8162,7 +8233,21 @@ public override string GetDebugStatus() "FHLoss", FireHeatLossPercent); #endif - + if (WaterMotionPumpFitted) + { + status.AppendFormat("{0}\t{1}\t{2}/{7}\t\t{3}\t{4}/{7}\t\t{5}\t{6}/{7}\n", + Simulator.Catalog.GetString("Pump:"), + Simulator.Catalog.GetString("Max"), + FormatStrings.FormatFuelVolume(pS.TopH(L.FromGUK(MaximumWaterMotionPumpFlowRateLBpS / WaterLBpUKG)), IsMetric, IsUK), + Simulator.Catalog.GetString("Pump1"), + FormatStrings.FormatFuelVolume(pS.TopH(L.FromGUK(WaterMotionPump1FlowRateLBpS / WaterLBpUKG)), IsMetric, IsUK), + Simulator.Catalog.GetString("Pump2"), + FormatStrings.FormatFuelVolume(pS.TopH(L.FromGUK(WaterMotionPump2FlowRateLBpS / WaterLBpUKG)), IsMetric, IsUK), + FormatStrings.h + ); + } + else + { status.AppendFormat("{0}\t{1}\t{6}/{12}\t\t({7:N0} {13})\t{2}\t{8}/{12}\t\t{3}\t{9}\t\t{4}\t{10}/{12}\t\t{5}\t{11}\n", Simulator.Catalog.GetString("Injector:"), Simulator.Catalog.GetString("Max"), @@ -8178,6 +8263,8 @@ public override string GetDebugStatus() FormatStrings.FormatTemperature(C.FromF(Injector2WaterDelTempF), IsMetric, false), FormatStrings.h, FormatStrings.mm); + } + if (SteamIsAuxTenderCoupled) { From 0306d754848ef106aedde98867fea61b1b4565ed Mon Sep 17 00:00:00 2001 From: peternewell Date: Tue, 25 Jun 2024 20:54:24 +1000 Subject: [PATCH 14/25] Correct bug with water motion pump --- .../RollingStocks/MSTSSteamLocomotive.cs | 45 ++++++++++++++++--- 1 file changed, 38 insertions(+), 7 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 3c0ae98ce..fa3e6113a 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -122,6 +122,9 @@ public class MSTSSteamLocomotive : MSTSLocomotive bool WaterMotionPump1IsOn = false; bool WaterMotionPump2IsOn = false; float WaterMotionPumpHeatLossBTU; + bool WaterMotionPumpLockedOut = false; + float WaterMotionPumpLockOutResetTimeS = 15.0f; // Time to reset the pump lock out time - time to prevent change of pumps + float WaterMotionPumpLockOutTimeS; // Current lock out time - reset after Reset Time exceeded public bool CylinderCocksAreOpen; public bool BlowdownValveOpen; public bool CylinderCompoundOn; // Flag to indicate whether compound locomotive is in compound or simple mode of operation - simple = true (ie bypass valve is open) @@ -1155,8 +1158,9 @@ public override void Save(BinaryWriter outf) outf.Write(Injector2IsOn); outf.Write(Injector2Fraction); outf.Write(InjectorLockedOut); + outf.Write(WaterMotionPumpLockedOut); outf.Write(InjectorLockOutTimeS); - outf.Write(InjectorLockOutResetTimeS); + outf.Write(WaterMotionPumpLockOutTimeS); outf.Write(WaterTempNewK); outf.Write(BkW_Diff); outf.Write(WaterFraction); @@ -1219,8 +1223,9 @@ public override void Restore(BinaryReader inf) Injector2IsOn = inf.ReadBoolean(); Injector2Fraction = inf.ReadSingle(); InjectorLockedOut = inf.ReadBoolean(); + WaterMotionPumpLockedOut = inf.ReadBoolean(); InjectorLockOutTimeS = inf.ReadSingle(); - InjectorLockOutResetTimeS = inf.ReadSingle(); + WaterMotionPumpLockOutTimeS = inf.ReadSingle(); WaterTempNewK = inf.ReadSingle(); BkW_Diff = inf.ReadSingle(); WaterFraction = inf.ReadSingle(); @@ -7024,11 +7029,19 @@ private void UpdateInjectors(float elapsedClockSeconds) { WaterMotionPump1FlowRateLBpS = MaximumWaterMotionPumpFlowRateLBpS * absSpeedMpS / MpS.FromMpH(MaxLocoSpeedMpH); } + else + { + WaterMotionPump1FlowRateLBpS = 0; + } if (WaterMotionPump2IsOn && absSpeedMpS > 0) { WaterMotionPump2FlowRateLBpS = MaximumWaterMotionPumpFlowRateLBpS * absSpeedMpS / MpS.FromMpH(MaxLocoSpeedMpH); } + else + { + WaterMotionPump2FlowRateLBpS = 0; + } if (WaterIsExhausted) { @@ -7039,14 +7052,29 @@ private void UpdateInjectors(float elapsedClockSeconds) float TotalPumpFlowRateLbpS = WaterMotionPump1FlowRateLBpS + WaterMotionPump2FlowRateLBpS; // Calculate heat loss for water injected - // Loss of boiler heat due to water injection - loss is the diff between steam and water Heat + // Loss of boiler heat due to water injection - loss is the diff between steam and ambient temperature (or pressure) WaterMotionPumpHeatLossBTU = TotalPumpFlowRateLbpS * (WaterHeatPSItoBTUpLB[BoilerPressurePSI] - WaterHeatPSItoBTUpLB[0]); // calculate Water steam heat based on injector water delivery temp - BoilerMassLB += elapsedClockSeconds * TotalPumpFlowRateLbpS; // Boiler Mass increase by Injector both pumps - BoilerHeatBTU -= elapsedClockSeconds * WaterMotionPumpHeatLossBTU; // Total loss of boiler heat due to water injection - inject steam and water Heat + BoilerMassLB += elapsedClockSeconds * TotalPumpFlowRateLbpS; // Boiler Mass increase by both pumps + BoilerHeatBTU -= elapsedClockSeconds * WaterMotionPumpHeatLossBTU; // Total loss of boiler heat due to water pump - inject cold water straight from tender // InjectorBoilerInputLB += (elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 BoilerHeatOutBTUpS += WaterMotionPumpHeatLossBTU; // Total loss of boiler heat due to water injection - inject steam and water Heat + + // Update pump lockout timer + if (WaterMotionPump1IsOn || WaterMotionPump2IsOn) + { + if (WaterMotionPumpLockedOut) + { + WaterMotionPumpLockOutTimeS += elapsedClockSeconds; + } + if (WaterMotionPumpLockOutTimeS > WaterMotionPumpLockOutResetTimeS) + { + WaterMotionPumpLockedOut = false; + WaterMotionPumpLockOutTimeS = 0.0f; + + } + } } else { @@ -7245,20 +7273,23 @@ private void UpdateFiring(float absSpeedMpS) if (WaterMotionPumpFitted && !WaterIsExhausted) { + if (WaterGlassLevelIN > 7.99) // turn pumps off if water level in boiler greater then 8.0, to stop cycling { WaterMotionPump1IsOn = false; WaterMotionPump2IsOn = false; } - else if (WaterGlassLevelIN <= 7.0 && WaterGlassLevelIN > 5.75) // turn water pump #1 on if water level in boiler drops below 7.0 and is above + else if (WaterGlassLevelIN <= 7.0 && WaterGlassLevelIN > 5.75 && !WaterMotionPumpLockedOut) // turn water pump #1 on if water level in boiler drops below 7.0 and is above { WaterMotionPump1IsOn = true; WaterMotionPump2IsOn = false; + WaterMotionPumpLockedOut = true; } - else if (WaterGlassLevelIN <= 5.75 && WaterGlassLevelIN > 4.5) // turn water pump #1 on if water level in boiler drops below 7.0 and is above + else if (WaterGlassLevelIN <= 5.75 && WaterGlassLevelIN > 4.5 && !WaterMotionPumpLockedOut) // turn water pump #2 on as well if water level in boiler drops below 5.75 and is above { WaterMotionPump1IsOn = true; WaterMotionPump2IsOn = true; + WaterMotionPumpLockedOut = true; } } else From 3af1460b231f2e4c70bbfc8e570dc39adbdaad8c Mon Sep 17 00:00:00 2001 From: peternewell Date: Wed, 26 Jun 2024 14:33:53 +1000 Subject: [PATCH 15/25] Add sound triggers for water pumps --- Source/Orts.Simulation/Common/Events.cs | 10 +++ .../RollingStocks/MSTSLocomotive.cs | 9 ++ .../RollingStocks/MSTSSteamLocomotive.cs | 83 ++++++++++++++++--- 3 files changed, 90 insertions(+), 12 deletions(-) diff --git a/Source/Orts.Simulation/Common/Events.cs b/Source/Orts.Simulation/Common/Events.cs index 7c704d814..19f60868a 100644 --- a/Source/Orts.Simulation/Common/Events.cs +++ b/Source/Orts.Simulation/Common/Events.cs @@ -141,6 +141,10 @@ public enum Event WaterInjector1On, WaterInjector2Off, WaterInjector2On, + WaterMotionPump1Off, + WaterMotionPump1On, + WaterMotionPump2Off, + WaterMotionPump2On, BlowdownValveToggle, SteamHeatChange, SteamPulse1, @@ -551,6 +555,12 @@ public static Event From(Source source, int eventID) case 321: return Event.BoosterCylinderCocksOpen; case 322: return Event.BoosterCylinderCocksClose; + // Miscellaneous + case 350: return Event.WaterMotionPump1On; + case 351: return Event.WaterMotionPump1Off; + case 352: return Event.WaterMotionPump2On; + case 353: return Event.WaterMotionPump2Off; + default: return 0; } case Source.MSTSCrossing: diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs index 1280c9d73..10b839584 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs @@ -249,6 +249,15 @@ public enum SteamLocomotiveFuelTypes public SteamLocomotiveFuelTypes SteamLocomotiveFuelType; + public enum SteamLocomotiveFeedWaterSystemTypes + { + Unknown, + MotionPump, + SteamInjector, // not used at the moment + } + + public SteamLocomotiveFeedWaterSystemTypes SteamLocomotiveFeedWaterType; + // Adhesion parameters public enum SlipControlType { diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index fa3e6113a..96c799e73 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -115,12 +115,13 @@ public class MSTSSteamLocomotive : MSTSLocomotive bool Injector1SoundIsOn = false; public bool Injector2IsOn; bool Injector2SoundIsOn = false; - bool WaterMotionPumpFitted = false; float WaterMotionPump1FlowRateLBpS; float WaterMotionPump2FlowRateLBpS; float MaximumWaterMotionPumpFlowRateLBpS; bool WaterMotionPump1IsOn = false; bool WaterMotionPump2IsOn = false; + bool WaterMotionPumpSound1IsOn = false; + bool WaterMotionPumpSound2IsOn = false; float WaterMotionPumpHeatLossBTU; bool WaterMotionPumpLockedOut = false; float WaterMotionPumpLockOutResetTimeS = 15.0f; // Time to reset the pump lock out time - time to prevent change of pumps @@ -944,11 +945,6 @@ public override void Parse(string lowercasetoken, STFReader stf) if (heating == 1) FuelOilSteamHeatingReqd = true; break; - case "engine(ortswatermotionpump": - var motionpump = stf.ReadIntBlock(null); - if (motionpump == 1) - WaterMotionPumpFitted = true; - break; case "engine(ortsfueloilspecificgravity": OilSpecificGravity = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(enginecontrollers(cutoff": CutoffController.Parse(stf); break; case "engine(enginecontrollers(ortssmallejector": SmallEjectorController.Parse(stf); SmallEjectorControllerFitted = true; break; @@ -999,7 +995,7 @@ public override void Parse(string lowercasetoken, STFReader stf) STFException.TraceWarning(stf, "Assumed unknown engine type " + steamengineType); } break; - case "engine(steamlocomotivefueltype": + case "engine(ortssteamlocomotivefueltype": stf.MustMatch("("); var steamLocomotiveFuelType = stf.ReadString(); try @@ -1009,7 +1005,20 @@ public override void Parse(string lowercasetoken, STFReader stf) catch { if (Simulator.Settings.VerboseConfigurationMessages) - STFException.TraceWarning(stf, "Assumed unknown engine type " + steamLocomotiveFuelType); + STFException.TraceWarning(stf, "Assumed unknown fuel type " + steamLocomotiveFuelType); + } + break; + case "engine(ortssteamlocomotivefeedwatersystemtype": + stf.MustMatch("("); + var feedwaterType = stf.ReadString(); + try + { + SteamLocomotiveFeedWaterType = (SteamLocomotiveFeedWaterSystemTypes)Enum.Parse(typeof(SteamLocomotiveFeedWaterSystemTypes), feedwaterType); + } + catch + { + if (Simulator.Settings.VerboseConfigurationMessages) + STFException.TraceWarning(stf, "Assumed unknown feedwater type " + feedwaterType); } break; case "engine(ortssteamboilertype": @@ -1063,7 +1072,7 @@ public override void Copy(MSTSWagon copy) CylinderPortOpeningFactor = locoCopy.CylinderPortOpeningFactor; BoilerVolumeFT3 = locoCopy.BoilerVolumeFT3; MaxBoilerPressurePSI = locoCopy.MaxBoilerPressurePSI; - WaterMotionPumpFitted = locoCopy.WaterMotionPumpFitted; + SteamLocomotiveFeedWaterType = locoCopy.SteamLocomotiveFeedWaterType; MaxSuperheatRefTempF = locoCopy.MaxSuperheatRefTempF; MaxIndicatedHorsePowerHP = locoCopy.MaxIndicatedHorsePowerHP; SuperheatCutoffPressureFactor = locoCopy.SuperheatCutoffPressureFactor; @@ -7021,7 +7030,7 @@ private void UpdateWaterGauge() private void UpdateInjectors(float elapsedClockSeconds) { - if (WaterMotionPumpFitted) + if (SteamLocomotiveFeedWaterType == SteamLocomotiveFeedWaterSystemTypes.MotionPump) { MaximumWaterMotionPumpFlowRateLBpS = (1.2f * EvaporationLBpS) / 2.0f; // Assume two pumps and that they can pump a fraction more water the the maximum steam production @@ -7271,25 +7280,29 @@ private void UpdateFiring(float absSpeedMpS) #region AI Fireman { - if (WaterMotionPumpFitted && !WaterIsExhausted) + if (SteamLocomotiveFeedWaterType == SteamLocomotiveFeedWaterSystemTypes.MotionPump && !WaterIsExhausted) { if (WaterGlassLevelIN > 7.99) // turn pumps off if water level in boiler greater then 8.0, to stop cycling { WaterMotionPump1IsOn = false; WaterMotionPump2IsOn = false; + StopMotionPump1Sound(); + StopMotionPump2Sound(); } else if (WaterGlassLevelIN <= 7.0 && WaterGlassLevelIN > 5.75 && !WaterMotionPumpLockedOut) // turn water pump #1 on if water level in boiler drops below 7.0 and is above { WaterMotionPump1IsOn = true; WaterMotionPump2IsOn = false; WaterMotionPumpLockedOut = true; + PlayMotionPump1SoundIfStarting(); } else if (WaterGlassLevelIN <= 5.75 && WaterGlassLevelIN > 4.5 && !WaterMotionPumpLockedOut) // turn water pump #2 on as well if water level in boiler drops below 5.75 and is above { WaterMotionPump1IsOn = true; WaterMotionPump2IsOn = true; WaterMotionPumpLockedOut = true; + PlayMotionPump2SoundIfStarting(); } } else @@ -7575,7 +7588,53 @@ private void StopInjector2Sound() } } + /// + /// Turn on the MotionPump 1 sound only when the pump starts. + /// + private void PlayMotionPump1SoundIfStarting() + { + if (!WaterMotionPumpSound1IsOn) + { + WaterMotionPumpSound1IsOn = true; + SignalEvent(Event.WaterMotionPump1On); + } + } + + /// + /// Turn on the MotionPump 2 sound only when the pump starts. + /// + private void PlayMotionPump2SoundIfStarting() + { + if (!WaterMotionPumpSound2IsOn) + { + WaterMotionPumpSound2IsOn = true; + SignalEvent(Event.WaterMotionPump2On); + } + } + + /// + /// Turn off the MotionPump 1 sound only when the pump stops. + /// + private void StopMotionPump1Sound() + { + if (WaterMotionPumpSound1IsOn) + { + WaterMotionPumpSound1IsOn = false; + SignalEvent(Event.WaterMotionPump1Off); + } + } + /// + /// Turn off the MotionPump 2 sound only when the pump stops. + /// + private void StopMotionPump2Sound() + { + if (WaterMotionPumpSound2IsOn) + { + WaterMotionPumpSound2IsOn = false; + SignalEvent(Event.WaterMotionPump2Off); + } + } protected override void UpdateCarSteamHeat(float elapsedClockSeconds) { // Update Steam Heating System @@ -8264,7 +8323,7 @@ public override string GetDebugStatus() "FHLoss", FireHeatLossPercent); #endif - if (WaterMotionPumpFitted) + if (SteamLocomotiveFeedWaterType == SteamLocomotiveFeedWaterSystemTypes.MotionPump) { status.AppendFormat("{0}\t{1}\t{2}/{7}\t\t{3}\t{4}/{7}\t\t{5}\t{6}/{7}\n", Simulator.Catalog.GetString("Pump:"), From 2a3bfc559fc9d1761110f5568e0e63b5bb4a5d99 Mon Sep 17 00:00:00 2001 From: peternewell Date: Wed, 26 Jun 2024 14:58:01 +1000 Subject: [PATCH 16/25] Change to resolve conflict --- Source/Orts.Simulation/Common/Events.cs | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Source/Orts.Simulation/Common/Events.cs b/Source/Orts.Simulation/Common/Events.cs index 19f60868a..a74f6e164 100644 --- a/Source/Orts.Simulation/Common/Events.cs +++ b/Source/Orts.Simulation/Common/Events.cs @@ -376,7 +376,13 @@ public static Event From(Source source, int eventID) case 66: return Event.Pantograph2Up; case 67: return Event.Pantograph2Down; + // ORTS only Events + case 90: return Event.WaterMotionPump1On; + case 91: return Event.WaterMotionPump1Off; + case 92: return Event.WaterMotionPump2On; + case 93: return Event.WaterMotionPump2Off; + case 101: return Event.GearUp; // for gearbox based engines case 102: return Event.GearDown; // for gearbox based engines case 103: return Event.ReverserToForwardBackward; // reverser moved to forward or backward position @@ -555,12 +561,6 @@ public static Event From(Source source, int eventID) case 321: return Event.BoosterCylinderCocksOpen; case 322: return Event.BoosterCylinderCocksClose; - // Miscellaneous - case 350: return Event.WaterMotionPump1On; - case 351: return Event.WaterMotionPump1Off; - case 352: return Event.WaterMotionPump2On; - case 353: return Event.WaterMotionPump2Off; - default: return 0; } case Source.MSTSCrossing: From 8e14b45451d3b611225a5510b52c6d43d798fe5b Mon Sep 17 00:00:00 2001 From: peternewell Date: Thu, 27 Jun 2024 14:51:20 +1000 Subject: [PATCH 17/25] Correct water consumption rate for wood fired --- .../RollingStocks/MSTSSteamLocomotive.cs | 24 +++++++++---------- 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 96c799e73..18ce284c1 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -415,7 +415,7 @@ public float TenderCoalMassKG // Decreased by firing and increased float Injector1Fraction = 0.0f; // Fraction (0-1) of injector 1 flow from Fireman controller or AI float Injector2Fraction = 0.0f; // Fraction (0-1) of injector of injector 2 flow from Fireman controller or AI float SafetyValveStartPSI = 0.1f; // Set safety valve to just over max pressure - allows for safety valve not to operate in AI firing - float InjectorBoilerInputLB = 0.0f; // Input into boiler from injectors + float BoilerWaterInputLB = 0.0f; // Water input into boiler from injectors and pumps const float WaterDensityAt100DegC1BarKGpM3 = 954.8f; @@ -2750,7 +2750,7 @@ public override void Update(float elapsedClockSeconds) #region adjust state UpdateWaterGauge(); - UpdateInjectors(elapsedClockSeconds); + UpdateWaterInjection(elapsedClockSeconds); UpdateFiring(absSpeedMpS); #endregion @@ -3812,7 +3812,7 @@ private void UpdateTender(float elapsedClockSeconds) if (HasTenderCoupled) // If a tender is coupled then water is available { - CombinedTenderWaterVolumeUKG -= InjectorBoilerInputLB / WaterLBpUKG; // Adjust water usage in tender + CombinedTenderWaterVolumeUKG -= BoilerWaterInputLB / WaterLBpUKG; // Adjust water usage in tender } else // if no tender coupled then check whether a tender is required { @@ -3822,7 +3822,7 @@ private void UpdateTender(float elapsedClockSeconds) } else // Tender is not required (ie tank locomotive) - therefore water will be carried on the locomotive (and possibly on aux tender) { - CombinedTenderWaterVolumeUKG -= InjectorBoilerInputLB / WaterLBpUKG; // Adjust water usage in tender + CombinedTenderWaterVolumeUKG -= BoilerWaterInputLB / WaterLBpUKG; // Adjust water usage in tender } } @@ -3831,9 +3831,9 @@ private void UpdateTender(float elapsedClockSeconds) CurrentLocoTenderWaterVolumeUKG = (Kg.ToLb(MaxLocoTenderWaterMassKG) / WaterLBpUKG) * TenderWaterPercent; // Adjust water level in locomotive tender PrevCombinedTenderWaterVolumeUKG = CombinedTenderWaterVolumeUKG; // Store value for next iteration PreviousTenderWaterVolumeUKG = CombinedTenderWaterVolumeUKG; // Store value for next iteration - WaterConsumptionLbpS = InjectorBoilerInputLB / elapsedClockSeconds; // water consumption + WaterConsumptionLbpS = BoilerWaterInputLB / elapsedClockSeconds; // water consumption WaterConsumptionLbpS = MathHelper.Clamp(WaterConsumptionLbpS, 0, WaterConsumptionLbpS); - CumulativeWaterConsumptionLbs += InjectorBoilerInputLB; + CumulativeWaterConsumptionLbs += BoilerWaterInputLB; if (CumulativeWaterConsumptionLbs > 0) DbfEvalCumulativeWaterConsumptionLbs = CumulativeWaterConsumptionLbs;//DebriefEval #if DEBUG_AUXTENDER @@ -7027,8 +7027,9 @@ private void UpdateWaterGauge() } } - private void UpdateInjectors(float elapsedClockSeconds) + private void UpdateWaterInjection(float elapsedClockSeconds) { + BoilerWaterInputLB = 0; // Used by UpdateTender() later in the cycle if (SteamLocomotiveFeedWaterType == SteamLocomotiveFeedWaterSystemTypes.MotionPump) { @@ -7064,10 +7065,10 @@ private void UpdateInjectors(float elapsedClockSeconds) // Loss of boiler heat due to water injection - loss is the diff between steam and ambient temperature (or pressure) WaterMotionPumpHeatLossBTU = TotalPumpFlowRateLbpS * (WaterHeatPSItoBTUpLB[BoilerPressurePSI] - WaterHeatPSItoBTUpLB[0]); - // calculate Water steam heat based on injector water delivery temp + // calculate Water steam heat based on pump flow rate BoilerMassLB += elapsedClockSeconds * TotalPumpFlowRateLbpS; // Boiler Mass increase by both pumps BoilerHeatBTU -= elapsedClockSeconds * WaterMotionPumpHeatLossBTU; // Total loss of boiler heat due to water pump - inject cold water straight from tender -// InjectorBoilerInputLB += (elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 + BoilerWaterInputLB += (elapsedClockSeconds * TotalPumpFlowRateLbpS); // Keep track of water flow into boilers from Pump BoilerHeatOutBTUpS += WaterMotionPumpHeatLossBTU; // Total loss of boiler heat due to water injection - inject steam and water Heat // Update pump lockout timer @@ -7139,7 +7140,6 @@ private void UpdateInjectors(float elapsedClockSeconds) InjectorFlowRateLBpS = 0.0f; // If the tender water is empty, stop flow into boiler } - InjectorBoilerInputLB = 0; // Used by UpdateTender() later in the cycle if (WaterIsExhausted) { // don't fill boiler with injectors @@ -7177,7 +7177,7 @@ private void UpdateInjectors(float elapsedClockSeconds) // calculate Water steam heat based on injector water delivery temp BoilerMassLB += elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS; // Boiler Mass increase by Injector 1 BoilerHeatBTU -= elapsedClockSeconds * (Inject1WaterHeatLossBTU + Inject1SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat - InjectorBoilerInputLB += (elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 + BoilerWaterInputLB += (elapsedClockSeconds * Injector1Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 BoilerHeatOutBTUpS += (Inject1WaterHeatLossBTU + Inject1SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat } if (Injector2IsOn) @@ -7208,7 +7208,7 @@ private void UpdateInjectors(float elapsedClockSeconds) // calculate Water steam heat based on injector water delivery temp BoilerMassLB += elapsedClockSeconds * Injector2Fraction * InjectorFlowRateLBpS; // Boiler Mass increase by Injector 1 BoilerHeatBTU -= elapsedClockSeconds * (Inject2WaterHeatLossBTU + Inject2SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat - InjectorBoilerInputLB += (elapsedClockSeconds * Injector2Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 + BoilerWaterInputLB += (elapsedClockSeconds * Injector2Fraction * InjectorFlowRateLBpS); // Keep track of water flow into boilers from Injector 1 BoilerHeatOutBTUpS += (Inject2WaterHeatLossBTU + Inject2SteamHeatLossBTU); // Total loss of boiler heat due to water injection - inject steam and water Heat } } From 624f7954d84ca58d663c5f4ac8a234f7ff4ad781 Mon Sep 17 00:00:00 2001 From: peternewell Date: Fri, 28 Jun 2024 19:51:02 +1000 Subject: [PATCH 18/25] Adjust fuel masses in steam locomotive --- Source/Orts.Simulation/Simulation/AIs/AI.cs | 2 +- .../RollingStocks/MSTSSteamLocomotive.cs | 102 ++++++++++++------ .../Simulation/RollingStocks/MSTSWagon.cs | 8 +- .../Orts.Simulation/Simulation/Simulator.cs | 2 +- .../RunActivity/Viewer3D/Popups/HelpWindow.cs | 6 +- 5 files changed, 76 insertions(+), 44 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/AIs/AI.cs b/Source/Orts.Simulation/Simulation/AIs/AI.cs index dd285eeee..ee65843cd 100644 --- a/Source/Orts.Simulation/Simulation/AIs/AI.cs +++ b/Source/Orts.Simulation/Simulation/AIs/AI.cs @@ -923,7 +923,7 @@ public AITrain CreateAITrainDetail(Service_Definition sd, Traffic_Service_Defini if (Simulator.Activity != null && car is MSTSSteamLocomotive mstsSteamLocomotive) { mstsSteamLocomotive.CombinedTenderWaterVolumeUKG = ORTS.Common.Kg.ToLb(mstsSteamLocomotive.MaxLocoTenderWaterMassKG) / 10.0f * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelWater / 100.0f; - mstsSteamLocomotive.TenderCoalMassKG = mstsSteamLocomotive.MaxTenderCoalMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; } if (train.InitialSpeed != 0 && car is MSTSLocomotive loco) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 18ce284c1..d5761abba 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -403,11 +403,11 @@ public class MSTSSteamLocomotive : MSTSLocomotive float FuelDensityKGpM3 = 864.5f; // Anthracite Coal : 50 - 58 (lb/ft3), 800 - 929 (kg/m3) float DamperFactorManual = 1.0f; // factor to control draft through fire when locomotive is running in Manual mode public float WaterLBpUKG = 10.021f; // lbs of water in 1 gal (uk) - public float MaxTenderCoalMassKG = 1; // Maximum read from Eng File - - this value must be non-zero, if not defined in ENG file, can cause NaN errors - public float TenderCoalMassKG // Decreased by firing and increased by refilling + public float MaxTenderFuelMassKG = 1; // Maximum read from Eng File - - this value must be non-zero, if not defined in ENG file, can cause NaN errors + public float TenderFuelMassKG // Decreased by firing and increased by refilling { - get { return FuelController.CurrentValue * MaxTenderCoalMassKG; } - set { FuelController.CurrentValue = value / MaxTenderCoalMassKG; } + get { return FuelController.CurrentValue * MaxTenderFuelMassKG; } + set { FuelController.CurrentValue = value / MaxTenderFuelMassKG; } } float MaxTenderOilMassL; @@ -856,7 +856,7 @@ public void RefillTenderWithWater() /// public void InitializeTenderWithCoal() { - FuelController.CurrentValue = TenderCoalMassKG / MaxTenderCoalMassKG; + FuelController.CurrentValue = TenderFuelMassKG / MaxTenderFuelMassKG; } /// @@ -934,7 +934,7 @@ public override void Parse(string lowercasetoken, STFReader stf) case "engine(vacuumbrakessmallejectorusagerate": EjectorSmallSteamConsumptionLbpS = pS.FrompH(stf.ReadFloatBlock(STFReader.UNITS.MassRateDefaultLBpH, null)); break; case "engine(ortssuperheatcutoffpressurefactor": SuperheatCutoffPressureFactor = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(shovelcoalmass": ShovelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; - case "engine(maxtendercoalmass": MaxTenderCoalMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; + case "engine(maxtendercoalmass": MaxTenderFuelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "engine(ortsmaxtenderfueloilvolume": MaxTenderOilMassL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; case "engine(maxtenderwatermass": MaxLocoTenderWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "engine(steamfiremanmaxpossiblefiringrate": MaxFiringRateKGpS = stf.ReadFloatBlock(STFReader.UNITS.MassRateDefaultLBpH, null) / 2.2046f / 3600; break; @@ -1083,7 +1083,7 @@ public override void Copy(MSTSWagon copy) ShovelMassKG = locoCopy.ShovelMassKG; GearedTractiveEffortFactor = locoCopy.GearedTractiveEffortFactor; TractiveEffortFactor = locoCopy.TractiveEffortFactor; - MaxTenderCoalMassKG = locoCopy.MaxTenderCoalMassKG; + MaxTenderFuelMassKG = locoCopy.MaxTenderFuelMassKG; MaxLocoTenderWaterMassKG = locoCopy.MaxLocoTenderWaterMassKG; MaxFiringRateKGpS = locoCopy.MaxFiringRateKGpS; Stoker = locoCopy.Stoker; @@ -1143,7 +1143,7 @@ public override void Save(BinaryWriter outf) outf.Write(PreviousBoilerHeatOutBTUpS); outf.Write(PreviousBoilerHeatSmoothedBTU); outf.Write(BurnRateRawKGpS); - outf.Write(TenderCoalMassKG); + outf.Write(TenderFuelMassKG); outf.Write(MaxTotalCombinedWaterVolumeUKG); outf.Write(CombinedTenderWaterVolumeUKG); outf.Write(CumulativeWaterConsumptionLbs); @@ -1208,7 +1208,7 @@ public override void Restore(BinaryReader inf) PreviousBoilerHeatOutBTUpS = inf.ReadSingle(); PreviousBoilerHeatSmoothedBTU = inf.ReadSingle(); BurnRateRawKGpS = inf.ReadSingle(); - TenderCoalMassKG = inf.ReadSingle(); + TenderFuelMassKG = inf.ReadSingle(); MaxTotalCombinedWaterVolumeUKG = inf.ReadSingle(); CombinedTenderWaterVolumeUKG = inf.ReadSingle(); CumulativeWaterConsumptionLbs = inf.ReadSingle(); @@ -1350,7 +1350,7 @@ public override void Initialize() // Set Oil mass - if an oil locomotive if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil && MaxTenderOilMassL != 0) { - MaxTenderCoalMassKG = MaxTenderOilMassL * OilSpecificGravity; + MaxTenderFuelMassKG = MaxTenderOilMassL * OilSpecificGravity; } // Oil burning locomotives will always have mechanical stokers @@ -2327,7 +2327,7 @@ public override void Initialize() Trace.TraceInformation("**************** Fire ****************"); Trace.TraceInformation("Grate - Area {0:N1} sq ft, Limit {1:N1} lb/sq ft", Me2.ToFt2(GrateAreaM2), GrateLimitLBpFt2); - Trace.TraceInformation("Fuel - Calorific {0} btu/lb, Max Firing Rate {1} lbs/h Max Coal Load {2} lbs", KJpKg.ToBTUpLb(FuelCalorificKJpKG), Kg.ToLb(pS.TopH(MaxFiringRateKGpS)), Kg.ToLb(MaxTenderCoalMassKG)); + Trace.TraceInformation("Fuel - Calorific {0} btu/lb, Max Firing Rate {1} lbs/h Max Coal Load {2} lbs", KJpKg.ToBTUpLb(FuelCalorificKJpKG), Kg.ToLb(pS.TopH(MaxFiringRateKGpS)), Kg.ToLb(MaxTenderFuelMassKG)); Trace.TraceInformation("========================================================================================================================================================"); @@ -3741,23 +3741,23 @@ private void UpdateTender(float elapsedClockSeconds) if (HasTenderCoupled) // If a tender is coupled then coal is available { - TenderCoalMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. - TenderCoalMassKG = MathHelper.Clamp(TenderCoalMassKG, 0, MaxTenderCoalMassKG); // Clamp value so that it doesn't go out of bounds + TenderFuelMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. + TenderFuelMassKG = MathHelper.Clamp(TenderFuelMassKG, 0, MaxTenderFuelMassKG); // Clamp value so that it doesn't go out of bounds } else // if no tender coupled then check whether a tender is required { if (IsTenderRequired == 1.0) // Tender is required { - TenderCoalMassKG = 0.0f; // Set tender coal to zero (none available) + TenderFuelMassKG = 0.0f; // Set tender coal to zero (none available) } else // Tender is not required (ie tank locomotive) - therefore coal will be carried on the locomotive { - TenderCoalMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. - TenderCoalMassKG = MathHelper.Clamp(TenderCoalMassKG, 0, MaxTenderCoalMassKG); // Clamp value so that it doesn't go out of bounds + TenderFuelMassKG -= elapsedClockSeconds * pS.FrompH(Kg.FromLb(NewBurnRateSteamToFuelLbspH[pS.TopH(TempCylinderSteamUsageLbpS)])); // Current Tender coal mass determined by burn rate. + TenderFuelMassKG = MathHelper.Clamp(TenderFuelMassKG, 0, MaxTenderFuelMassKG); // Clamp value so that it doesn't go out of bounds } } - if (TenderCoalMassKG < 1.0) + if (TenderFuelMassKG < 1.0) { if (!CoalIsExhausted) { @@ -7758,9 +7758,9 @@ public override float GetDataOf(CabViewControl cvc) } case CABViewControlTypes.FUEL_GAUGE: if (cvc.Units == CABViewControlUnits.LBS) - data = Kg.ToLb(TenderCoalMassKG); + data = Kg.ToLb(TenderFuelMassKG); else - data = TenderCoalMassKG; + data = TenderFuelMassKG; break; default: data = base.GetDataOf(cvc); @@ -7783,7 +7783,7 @@ public override string GetStatus() boilerPressureSafety = boilerPressurePercent <= 0.25 ? "!!!" : boilerPressurePercent <= 0.5 ? "???" : ""; } var boilerWaterSafety = WaterFraction < WaterMinLevel || WaterFraction > WaterMaxLevel ? "!!!" : WaterFraction < WaterMinLevelSafe || WaterFraction > WaterMaxLevelSafe ? "???" : ""; - var coalPercent = TenderCoalMassKG / MaxTenderCoalMassKG; + var coalPercent = TenderFuelMassKG / MaxTenderFuelMassKG; var waterPercent = CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG; var fuelSafety = CoalIsExhausted || WaterIsExhausted ? "!!!" : coalPercent <= 0.105 || waterPercent <= 0.105 ? "???" : ""; var steamusagesafety = PreviousTotalSteamUsageLBpS > EvaporationLBpS ? "!!!" : PreviousTotalSteamUsageLBpS > EvaporationLBpS * 0.95f ? "???" : ""; @@ -8361,8 +8361,8 @@ public override string GetDebugStatus() status.AppendFormat("{0}\t{1}\t{2}\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8}\t\t{9}\t{10}\t\t{11}\t{12:N0}\t{13}\t{14:N0}\n", Simulator.Catalog.GetString("Tender:"), Simulator.Catalog.GetString("Coal"), - FormatStrings.FormatMass(TenderCoalMassKG, IsMetric), - TenderCoalMassKG / MaxTenderCoalMassKG * 100, + FormatStrings.FormatMass(TenderFuelMassKG, IsMetric), + TenderFuelMassKG / MaxTenderFuelMassKG * 100, Simulator.Catalog.GetString("Water(C)"), FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, @@ -8383,8 +8383,8 @@ public override string GetDebugStatus() status.AppendFormat("{0}\t{1}\t{2:N0}\t\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", Simulator.Catalog.GetString("Tender:"), Simulator.Catalog.GetString("Oil"), - FormatStrings.FormatFuelVolume(L.FromGUK(OilSpecificGravity * (Kg.ToLb(TenderCoalMassKG) / WaterLBpUKG)), IsMetric, IsUK), - TenderCoalMassKG / MaxTenderCoalMassKG * 100, + FormatStrings.FormatFuelVolume(L.FromGUK(OilSpecificGravity * (Kg.ToLb(TenderFuelMassKG) / WaterLBpUKG)), IsMetric, IsUK), + TenderFuelMassKG / MaxTenderFuelMassKG * 100, Simulator.Catalog.GetString("Water"), FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, @@ -8399,8 +8399,8 @@ public override string GetDebugStatus() status.AppendFormat("{0}\t{1}\t{2:N0}\t\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", Simulator.Catalog.GetString("Tender:"), Simulator.Catalog.GetString("Wood"), - FormatStrings.FormatFuelVolume(L.FromGUK(OilSpecificGravity * (Kg.ToLb(TenderCoalMassKG) / WaterLBpUKG)), IsMetric, IsUK), - TenderCoalMassKG / MaxTenderCoalMassKG * 100, + FormatStrings.FormatMass(TenderFuelMassKG, IsMetric), + TenderFuelMassKG / MaxTenderFuelMassKG * 100, Simulator.Catalog.GetString("Water"), FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, @@ -8415,8 +8415,8 @@ public override string GetDebugStatus() status.AppendFormat("{0}\t{1}\t{2}\t{3:N0}%\t{4}\t{5}\t\t{6:N0}%\t{7}\t{8:N0}\t{9}\t\t{10:N0}\n", Simulator.Catalog.GetString("Tender:"), Simulator.Catalog.GetString("Coal"), - FormatStrings.FormatMass(TenderCoalMassKG, IsMetric), - TenderCoalMassKG / MaxTenderCoalMassKG * 100, + FormatStrings.FormatMass(TenderFuelMassKG, IsMetric), + TenderFuelMassKG / MaxTenderFuelMassKG * 100, Simulator.Catalog.GetString("Water"), FormatStrings.FormatFuelVolume(L.FromGUK(CombinedTenderWaterVolumeUKG), IsMetric, IsUK), CombinedTenderWaterVolumeUKG / MaxTotalCombinedWaterVolumeUKG * 100, @@ -9802,7 +9802,19 @@ public void AIFireReset() /// Matching controller or null public override MSTSNotchController GetRefillController(uint type) { - if (type == (uint)PickupType.FuelCoal) return FuelController; + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) + { + if (type == (uint)PickupType.FuelWood) return FuelController; + } + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + if (type == (uint)PickupType.FuelDiesel) return FuelController; + } + else + { + if (type == (uint)PickupType.FuelCoal) return FuelController; + } + if (type == (uint)PickupType.FuelWater) return WaterController; return null; } @@ -9814,8 +9826,8 @@ public override MSTSNotchController GetRefillController(uint type) public override void SetStepSize(PickupObj matchPickup) { uint type = matchPickup.PickupType; - if (type == (uint)PickupType.FuelCoal && MaxTenderCoalMassKG != 0) - FuelController.SetStepSize(matchPickup.PickupCapacity.FeedRateKGpS / MSTSNotchController.StandardBoost / MaxTenderCoalMassKG); + if (type == (uint)PickupType.FuelCoal && MaxTenderFuelMassKG != 0) + FuelController.SetStepSize(matchPickup.PickupCapacity.FeedRateKGpS / MSTSNotchController.StandardBoost / MaxTenderFuelMassKG); else if (type == (uint)PickupType.FuelWater && MaxLocoTenderWaterMassKG != 0) WaterController.SetStepSize(matchPickup.PickupCapacity.FeedRateKGpS / MSTSNotchController.StandardBoost / MaxLocoTenderWaterMassKG); } @@ -9837,14 +9849,34 @@ public override void RefillImmediately() /// 0.0 to 1.0. If type is unknown, returns 0.0 public override float GetFilledFraction(uint pickupType) { - if (pickupType == (uint)PickupType.FuelWater) + if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Wood) { - return WaterController.CurrentValue; + if (pickupType == (uint)PickupType.FuelWood) + { + return FuelController.CurrentValue; + } } - if (pickupType == (uint)PickupType.FuelCoal) + else if (SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + if (pickupType == (uint)PickupType.FuelDiesel) + { + return FuelController.CurrentValue; + } + } + else { - return FuelController.CurrentValue; + if (pickupType == (uint)PickupType.FuelCoal) + { + return FuelController.CurrentValue; + } } + + + if (pickupType == (uint)PickupType.FuelWater) + { + return WaterController.CurrentValue; + } + return 0f; } diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs index e008eae68..936cf92d5 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs @@ -1986,12 +1986,12 @@ public override void Update(float elapsedClockSeconds) float TempMaxCombinedWater = TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG; TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG = (TempMaxCombinedWater - (Kg.ToLb(TendersSteamLocomotive.MaxLocoTenderWaterMassKG) / WaterLBpUKG)) + (Kg.ToLb(TenderWagonMaxWaterMassKG) / WaterLBpUKG); - TendersSteamLocomotive.MaxTenderCoalMassKG = TenderWagonMaxCoalMassKG; + TendersSteamLocomotive.MaxTenderFuelMassKG = TenderWagonMaxCoalMassKG; TendersSteamLocomotive.MaxLocoTenderWaterMassKG = TenderWagonMaxWaterMassKG; if (Simulator.Settings.VerboseConfigurationMessages) { - Trace.TraceInformation("Fuel and Water Masses adjusted to Tender Values Specified in WAG File - Coal mass {0} kg, Water Mass {1}", FormatStrings.FormatMass(TendersSteamLocomotive.MaxTenderCoalMassKG, IsMetric), FormatStrings.FormatFuelVolume(L.FromGUK(TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG), IsMetric, IsUK)); + Trace.TraceInformation("Fuel and Water Masses adjusted to Tender Values Specified in WAG File - Coal mass {0} kg, Water Mass {1}", FormatStrings.FormatMass(TendersSteamLocomotive.MaxTenderFuelMassKG, IsMetric), FormatStrings.FormatFuelVolume(L.FromGUK(TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG), IsMetric, IsUK)); } } } @@ -2182,7 +2182,7 @@ private void UpdateLocomotiveLoadPhysics() // If = 0, then locomotive must be a tank type locomotive. A tank locomotive has the fuel (coal and water) onboard. // Thus the loco weight changes as boiler level goes up and down, and coal mass varies with the fire mass. Also onboard fuel (coal and water ) will vary as used. { - MassKG = LoadEmptyMassKg + Kg.FromLb(SteamLocomotiveIdentification.BoilerMassLB) + SteamLocomotiveIdentification.FireMassKG + SteamLocomotiveIdentification.TenderCoalMassKG + Kg.FromLb(SteamLocomotiveIdentification.CombinedTenderWaterVolumeUKG * WaterLBpUKG); + MassKG = LoadEmptyMassKg + Kg.FromLb(SteamLocomotiveIdentification.BoilerMassLB) + SteamLocomotiveIdentification.FireMassKG + SteamLocomotiveIdentification.TenderFuelMassKG + Kg.FromLb(SteamLocomotiveIdentification.CombinedTenderWaterVolumeUKG * WaterLBpUKG); MassKG = MathHelper.Clamp(MassKG, LoadEmptyMassKg, LoadFullMassKg); // Clamp Mass to between the empty and full wagon values // Adjust drive wheel weight SteamLocomotiveIdentification.DrvWheelWeightKg = (MassKG / InitialMassKG) * SteamLocomotiveIdentification.InitialDrvWheelWeightKg; @@ -3191,7 +3191,7 @@ private void UpdateTenderLoad() Trace.TraceInformation("Tender @ position {0} does not have a locomotive associated with. Check that it is preceeded by a steam locomotive.", CarID); } - MassKG = FreightAnimations.WagonEmptyWeight + TendersSteamLocomotive.TenderCoalMassKG + Kg.FromLb( (TendersSteamLocomotive.CurrentLocoTenderWaterVolumeUKG * WaterLBpUKG)); + MassKG = FreightAnimations.WagonEmptyWeight + TendersSteamLocomotive.TenderFuelMassKG + Kg.FromLb( (TendersSteamLocomotive.CurrentLocoTenderWaterVolumeUKG * WaterLBpUKG)); MassKG = MathHelper.Clamp(MassKG, LoadEmptyMassKg, LoadFullMassKg); // Clamp Mass to between the empty and full wagon values // Update wagon parameters sensitive to wagon mass change diff --git a/Source/Orts.Simulation/Simulation/Simulator.cs b/Source/Orts.Simulation/Simulation/Simulator.cs index 141bb8820..47751b16c 100644 --- a/Source/Orts.Simulation/Simulation/Simulator.cs +++ b/Source/Orts.Simulation/Simulation/Simulator.cs @@ -1313,7 +1313,7 @@ private Train InitializePlayerTrain() if (Activity != null && mstsSteamLocomotive != null) { mstsSteamLocomotive.CombinedTenderWaterVolumeUKG = (Kg.ToLb(mstsSteamLocomotive.MaxLocoTenderWaterMassKG) / 10.0f) * Activity.Tr_Activity.Tr_Activity_Header.FuelWater / 100.0f; - mstsSteamLocomotive.TenderCoalMassKG = mstsSteamLocomotive.MaxTenderCoalMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; } } catch (Exception error) diff --git a/Source/RunActivity/Viewer3D/Popups/HelpWindow.cs b/Source/RunActivity/Viewer3D/Popups/HelpWindow.cs index 781912e84..749188f53 100644 --- a/Source/RunActivity/Viewer3D/Popups/HelpWindow.cs +++ b/Source/RunActivity/Viewer3D/Popups/HelpWindow.cs @@ -696,10 +696,10 @@ private void ReportEvaluation(WindowManager owner, ControlLayout cl, TrainCar lo if (item.EngineType == TrainCar.EngineTypes.Steam && item.AuxWagonType == "Engine") {//Fuel Steam - nCoalvolume = nCoalvolume + (item as MSTSSteamLocomotive).MaxTenderCoalMassKG; - nCoallevel = nCoallevel + (item as MSTSSteamLocomotive).TenderCoalMassKG; + nCoalvolume = nCoalvolume + (item as MSTSSteamLocomotive).MaxTenderFuelMassKG; + nCoallevel = nCoallevel + (item as MSTSSteamLocomotive).TenderFuelMassKG; nCoalburned = nCoalvolume - nCoallevel; - nCoalBurnedPerc = 1 - ((item as MSTSSteamLocomotive).TenderCoalMassKG / (item as MSTSSteamLocomotive).MaxTenderCoalMassKG); + nCoalBurnedPerc = 1 - ((item as MSTSSteamLocomotive).TenderFuelMassKG / (item as MSTSSteamLocomotive).MaxTenderFuelMassKG); cEnginetype.Add("Steam"); nWaterBurnedPerc = 1 - ((item as MSTSSteamLocomotive).CombinedTenderWaterVolumeUKG / (item as MSTSSteamLocomotive).MaxTotalCombinedWaterVolumeUKG); From a4a44b58ad32d2536d5c878d6da50d1cd6a703ab Mon Sep 17 00:00:00 2001 From: peternewell Date: Sat, 29 Jun 2024 18:21:56 +1000 Subject: [PATCH 19/25] Further adjustments for multiple fuels --- Source/Orts.Simulation/Simulation/Confirmer.cs | 2 +- .../Simulation/RollingStocks/MSTSSteamLocomotive.cs | 11 ++++++----- .../Simulation/RollingStocks/MSTSWagon.cs | 11 ++++++----- 3 files changed, 13 insertions(+), 11 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/Confirmer.cs b/Source/Orts.Simulation/Simulation/Confirmer.cs index 97c2a0ff7..9b41d30a8 100644 --- a/Source/Orts.Simulation/Simulation/Confirmer.cs +++ b/Source/Orts.Simulation/Simulation/Confirmer.cs @@ -226,7 +226,7 @@ public Confirmer(Simulator simulator, double defaultDurationS) , new string [] { GetString("LargeEjector"), null, null, null, GetString("decrease"), GetString("increase") } , new string [] { GetString("SmallEjector"), null, null, null, GetString("decrease"), GetString("increase") } , new string [] { GetString("VacuumExhauster"), GetString("normal"), null, GetString("fast") } - , new string [] { GetString("Tender"), null, null, GetString("Coal re-filled"), null, GetString("Coal level") } + , new string [] { GetString("Tender"), null, null, GetString("Fuel re-filled"), null, GetString("Fuel level") } , new string [] { GetString("Tender"), null, null, GetString("Water re-filled"), null, GetString("Water level") } // General , new string [] { GetString("Water Scoop"), GetString("up"), null, GetString("down") } diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index d5761abba..5f159e1b2 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -831,14 +831,14 @@ public MSTSSteamLocomotive(Simulator simulator, string wagFile) SteamEngines = new SteamEngines(this); PowerSupply = new SteamPowerSupply(this); - RefillTenderWithCoal(); + RefillTenderWithFuel(); RefillTenderWithWater(); } /// - /// Sets the coal level to maximum. + /// Sets the fuel level to maximum. /// - public void RefillTenderWithCoal() + public void RefillTenderWithFuel() { FuelController.CurrentValue = 1.0f; } @@ -934,6 +934,7 @@ public override void Parse(string lowercasetoken, STFReader stf) case "engine(vacuumbrakessmallejectorusagerate": EjectorSmallSteamConsumptionLbpS = pS.FrompH(stf.ReadFloatBlock(STFReader.UNITS.MassRateDefaultLBpH, null)); break; case "engine(ortssuperheatcutoffpressurefactor": SuperheatCutoffPressureFactor = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "engine(shovelcoalmass": ShovelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; + case "engine(ortsmaxtenderwoodmass": case "engine(maxtendercoalmass": MaxTenderFuelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "engine(ortsmaxtenderfueloilvolume": MaxTenderOilMassL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; case "engine(maxtenderwatermass": MaxLocoTenderWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; @@ -9826,7 +9827,7 @@ public override MSTSNotchController GetRefillController(uint type) public override void SetStepSize(PickupObj matchPickup) { uint type = matchPickup.PickupType; - if (type == (uint)PickupType.FuelCoal && MaxTenderFuelMassKG != 0) + if (type == (uint)PickupType.FuelCoal && type == (uint)PickupType.FuelWood && type == (uint)PickupType.FuelDiesel && MaxTenderFuelMassKG != 0) FuelController.SetStepSize(matchPickup.PickupCapacity.FeedRateKGpS / MSTSNotchController.StandardBoost / MaxTenderFuelMassKG); else if (type == (uint)PickupType.FuelWater && MaxLocoTenderWaterMassKG != 0) WaterController.SetStepSize(matchPickup.PickupCapacity.FeedRateKGpS / MSTSNotchController.StandardBoost / MaxLocoTenderWaterMassKG); @@ -9838,7 +9839,7 @@ public override void SetStepSize(PickupObj matchPickup) /// public override void RefillImmediately() { - RefillTenderWithCoal(); + RefillTenderWithFuel(); RefillTenderWithWater(); } diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs index 936cf92d5..baea23d4e 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs @@ -148,7 +148,7 @@ public enum WindowState public float TrailLocoResistanceFactor; // Factor to reduce base and wind resistance if locomotive is not leading - based upon original Davis drag coefficients bool TenderWeightInitialize = true; - float TenderWagonMaxCoalMassKG; + float TenderWagonMaxFuelMassKG; float TenderWagonMaxWaterMassKG; // Wind Impacts @@ -1201,7 +1201,8 @@ public virtual void Parse(string lowercasetoken, STFReader stf) } break; case "wagon(ortsauxtenderwatermass": AuxTenderWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; - case "wagon(ortstenderwagoncoalmass": TenderWagonMaxCoalMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; + case "wagon(ortstenderwagonwoodmass": + case "wagon(ortstenderwagoncoalmass": TenderWagonMaxFuelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortstenderwagonwatermass": TenderWagonMaxWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortsheatingwindowderatingfactor": WindowDeratingFactor = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "wagon(ortsheatingcompartmenttemperatureset": DesiredCompartmentTempSetpointC = stf.ReadFloatBlock(STFReader.UNITS.Temperature, null); break; @@ -1598,7 +1599,7 @@ public virtual void Copy(MSTSWagon copy) MaximumWheelFlangeAngleRad = copy.MaximumWheelFlangeAngleRad; WheelFlangeLengthM = copy.WheelFlangeLengthM; AuxTenderWaterMassKG = copy.AuxTenderWaterMassKG; - TenderWagonMaxCoalMassKG = copy.TenderWagonMaxCoalMassKG; + TenderWagonMaxFuelMassKG = copy.TenderWagonMaxFuelMassKG; TenderWagonMaxWaterMassKG = copy.TenderWagonMaxWaterMassKG; InitWagonNumAxles = copy.InitWagonNumAxles; WagonNumAxles = copy.WagonNumAxles; @@ -1964,7 +1965,7 @@ public override void Update(float elapsedClockSeconds) ConfirmSteamLocomotiveTender(); // Confirms that a tender is connected to the steam locomotive // Adjusts water and coal mass based upon values assigned to the tender found in the WAG file rather then those defined in ENG file. - if (WagonType == WagonTypes.Tender && TenderWeightInitialize && TenderWagonMaxCoalMassKG != 0 && TenderWagonMaxWaterMassKG != 0) + if (WagonType == WagonTypes.Tender && TenderWeightInitialize && TenderWagonMaxFuelMassKG != 0 && TenderWagonMaxWaterMassKG != 0) { // Find the associated steam locomotive for this tender @@ -1986,7 +1987,7 @@ public override void Update(float elapsedClockSeconds) float TempMaxCombinedWater = TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG; TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG = (TempMaxCombinedWater - (Kg.ToLb(TendersSteamLocomotive.MaxLocoTenderWaterMassKG) / WaterLBpUKG)) + (Kg.ToLb(TenderWagonMaxWaterMassKG) / WaterLBpUKG); - TendersSteamLocomotive.MaxTenderFuelMassKG = TenderWagonMaxCoalMassKG; + TendersSteamLocomotive.MaxTenderFuelMassKG = TenderWagonMaxFuelMassKG; TendersSteamLocomotive.MaxLocoTenderWaterMassKG = TenderWagonMaxWaterMassKG; if (Simulator.Settings.VerboseConfigurationMessages) From d73de81fcfe87de04b00c0b39881da118fb82689 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sun, 30 Jun 2024 16:51:51 +1000 Subject: [PATCH 20/25] Add correction for oil burning parameters --- .../RollingStocks/MSTSSteamLocomotive.cs | 23 +++++++++++-------- .../Simulation/RollingStocks/MSTSWagon.cs | 21 ++++++++++++++--- 2 files changed, 31 insertions(+), 13 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 5f159e1b2..205cc57f8 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -313,7 +313,7 @@ public class MSTSSteamLocomotive : MSTSLocomotive float FuelFeedRateSmoothedKGpS = 0.0f; // Smoothed Fuel feedd Rate public float FuelBurnRateSmoothedKGpS; // Smoothed fuel burning rate - float OilSpecificGravity = 0.9659f; // Assume a mid range of API for this value, say API = 15 @ 20 Cdeg. + public float OilSpecificGravity = 0.9659f; // Assume a mid range of API for this value, say API = 15 @ 20 Cdeg. float WaterSpecificGravity = 1.0f; // Water @ 20 degC. bool FuelOilSteamHeatingReqd = false; @@ -410,7 +410,7 @@ public float TenderFuelMassKG // Decreased by firing and increased set { FuelController.CurrentValue = value / MaxTenderFuelMassKG; } } - float MaxTenderOilMassL; + public float MaxTenderOilMassL; float DamperBurnEffect; // Effect of the Damper control Used in manual firing) float Injector1Fraction = 0.0f; // Fraction (0-1) of injector 1 flow from Fireman controller or AI float Injector2Fraction = 0.0f; // Fraction (0-1) of injector of injector 2 flow from Fireman controller or AI @@ -7834,18 +7834,21 @@ public override string GetDebugStatus() status.AppendFormat("{0}\t\t{1}\n", Simulator.Catalog.GetString("Locomotive Type:"), SteamLocoType); - status.AppendFormat("{0}\t{1}\t{6}\t{2}\t{7}\t{3}\t{8}\t{4}\t{9}\t{5}\t{10}\n", + status.AppendFormat("{0}\t{1}\t{2}\t{3}\t{4}\t{5}\t{6}\t{7}\t{8}\t{9}\t{10}\t{11}\t{12}\n", Simulator.Catalog.GetString("Input:"), Simulator.Catalog.GetString("Evap"), - Simulator.Catalog.GetString("Grate"), - Simulator.Catalog.GetString("Boiler"), - Simulator.Catalog.GetString("SuperHr"), - Simulator.Catalog.GetString("FuelCal"), FormatStrings.FormatArea(EvaporationAreaM2, IsMetric), - FormatStrings.FormatArea(GrateAreaM2, IsMetric), - FormatStrings.FormatVolume(Me3.FromFt3(BoilerVolumeFT3), IsMetric), + Simulator.Catalog.GetString("SuperHr"), FormatStrings.FormatArea(SuperheatAreaM2, IsMetric), - FormatStrings.FormatEnergyDensityByMass(FuelCalorificKJpKG, IsMetric)); + Simulator.Catalog.GetString("Boiler"), + FormatStrings.FormatVolume(Me3.FromFt3(BoilerVolumeFT3), IsMetric), + Simulator.Catalog.GetString("Grate"), + FormatStrings.FormatArea(GrateAreaM2, IsMetric), + Simulator.Catalog.GetString("FType"), + SteamLocomotiveFuelType.ToString(), + Simulator.Catalog.GetString("FuelCal"), + FormatStrings.FormatEnergyDensityByMass(FuelCalorificKJpKG, IsMetric) + ); status.AppendFormat("{0}\t{1}\t{2:N2}\t{3}\t{4:N3}\n", Simulator.Catalog.GetString("Adj:"), diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs index baea23d4e..e9b95e9c1 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs @@ -52,6 +52,7 @@ using System.Diagnostics; using System.IO; using System.Linq; +using static Orts.Simulation.RollingStocks.MSTSLocomotive; using Event = Orts.Common.Event; namespace Orts.Simulation.RollingStocks @@ -149,6 +150,7 @@ public enum WindowState bool TenderWeightInitialize = true; float TenderWagonMaxFuelMassKG; + float TenderWagonMaxOilMassL; float TenderWagonMaxWaterMassKG; // Wind Impacts @@ -1203,6 +1205,7 @@ public virtual void Parse(string lowercasetoken, STFReader stf) case "wagon(ortsauxtenderwatermass": AuxTenderWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortstenderwagonwoodmass": case "wagon(ortstenderwagoncoalmass": TenderWagonMaxFuelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; + case "wagon(ortsmaxtenderfueloilvolume": TenderWagonMaxOilVolumeL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; case "wagon(ortstenderwagonwatermass": TenderWagonMaxWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortsheatingwindowderatingfactor": WindowDeratingFactor = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "wagon(ortsheatingcompartmenttemperatureset": DesiredCompartmentTempSetpointC = stf.ReadFloatBlock(STFReader.UNITS.Temperature, null); break; @@ -1600,6 +1603,7 @@ public virtual void Copy(MSTSWagon copy) WheelFlangeLengthM = copy.WheelFlangeLengthM; AuxTenderWaterMassKG = copy.AuxTenderWaterMassKG; TenderWagonMaxFuelMassKG = copy.TenderWagonMaxFuelMassKG; + TenderWagonMaxOilVolumeL = copy.TenderWagonMaxOilMassL; TenderWagonMaxWaterMassKG = copy.TenderWagonMaxWaterMassKG; InitWagonNumAxles = copy.InitWagonNumAxles; WagonNumAxles = copy.WagonNumAxles; @@ -1965,7 +1969,7 @@ public override void Update(float elapsedClockSeconds) ConfirmSteamLocomotiveTender(); // Confirms that a tender is connected to the steam locomotive // Adjusts water and coal mass based upon values assigned to the tender found in the WAG file rather then those defined in ENG file. - if (WagonType == WagonTypes.Tender && TenderWeightInitialize && TenderWagonMaxFuelMassKG != 0 && TenderWagonMaxWaterMassKG != 0) + if (WagonType == WagonTypes.Tender && TenderWeightInitialize && (TenderWagonMaxFuelMassKG != 0 || TenderWagonMaxOilMassL != 0) && TenderWagonMaxWaterMassKG != 0) { // Find the associated steam locomotive for this tender @@ -1986,10 +1990,18 @@ public override void Update(float elapsedClockSeconds) // amount of water defined in the ENG file, and adding the water defined in the WAG file. float TempMaxCombinedWater = TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG; TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG = (TempMaxCombinedWater - (Kg.ToLb(TendersSteamLocomotive.MaxLocoTenderWaterMassKG) / WaterLBpUKG)) + (Kg.ToLb(TenderWagonMaxWaterMassKG) / WaterLBpUKG); - - TendersSteamLocomotive.MaxTenderFuelMassKG = TenderWagonMaxFuelMassKG; + TendersSteamLocomotive.MaxLocoTenderWaterMassKG = TenderWagonMaxWaterMassKG; + if (TendersSteamLocomotive.SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + { + TendersSteamLocomotive.MaxTenderFuelMassKG = TendersSteamLocomotive.MaxTenderOilMassL * TendersSteamLocomotive.OilSpecificGravity; + } + else + { + TendersSteamLocomotive.MaxTenderFuelMassKG = TenderWagonMaxFuelMassKG; + } + if (Simulator.Settings.VerboseConfigurationMessages) { Trace.TraceInformation("Fuel and Water Masses adjusted to Tender Values Specified in WAG File - Coal mass {0} kg, Water Mass {1}", FormatStrings.FormatMass(TendersSteamLocomotive.MaxTenderFuelMassKG, IsMetric), FormatStrings.FormatFuelVolume(L.FromGUK(TendersSteamLocomotive.MaxTotalCombinedWaterVolumeUKG), IsMetric, IsUK)); @@ -3784,6 +3796,9 @@ public MSTSCoupling Coupler return Couplers[0]; // defaults to the rear coupler (typically the first read) } } + + public float TenderWagonMaxOilVolumeL { get; private set; } + public override float GetCouplerZeroLengthM() { if (IsPlayerTrain && Simulator.UseAdvancedAdhesion && !Simulator.Settings.SimpleControlPhysics && IsAdvancedCoupler) From f2fff29652fbaa919f7f20f6e6e9648f166324e5 Mon Sep 17 00:00:00 2001 From: peternewell Date: Mon, 1 Jul 2024 07:55:15 +1000 Subject: [PATCH 21/25] Add extra fuel types to activity initialisation --- Source/Orts.Formats.Msts/ActivityFile.cs | 4 ++++ Source/Orts.Simulation/Simulation/AIs/AI.cs | 16 +++++++++++++++- .../Simulation/RollingStocks/MSTSLocomotive.cs | 2 +- Source/Orts.Simulation/Simulation/Simulator.cs | 17 ++++++++++++++++- 4 files changed, 36 insertions(+), 3 deletions(-) diff --git a/Source/Orts.Formats.Msts/ActivityFile.cs b/Source/Orts.Formats.Msts/ActivityFile.cs index c1fd16024..728615531 100644 --- a/Source/Orts.Formats.Msts/ActivityFile.cs +++ b/Source/Orts.Formats.Msts/ActivityFile.cs @@ -400,6 +400,8 @@ public class Tr_Activity_Header { public int FuelWater = 100; // percent public int FuelCoal = 100; // percent public int FuelDiesel = 100; // percent + public int FuelWood = 100; // percent + public int FuelSand = 100; // percent public string LoadStationsPopulationFile; public Tr_Activity_Header(STFReader stf) { @@ -423,7 +425,9 @@ public Tr_Activity_Header(STFReader stf) { new STFReader.TokenProcessor("workers", ()=>{ Workers = stf.ReadIntBlock(Workers); }), new STFReader.TokenProcessor("fuelwater", ()=>{ FuelWater = stf.ReadIntBlock(FuelWater); }), new STFReader.TokenProcessor("fuelcoal", ()=>{ FuelCoal = stf.ReadIntBlock(FuelCoal); }), + new STFReader.TokenProcessor("fuelwood", ()=>{ FuelWood = stf.ReadIntBlock(FuelWood); }), new STFReader.TokenProcessor("fueldiesel", ()=>{ FuelDiesel = stf.ReadIntBlock(FuelDiesel); }), + new STFReader.TokenProcessor("fuelsand", ()=>{ FuelSand = stf.ReadIntBlock(FuelSand); }), new STFReader.TokenProcessor("ortsloadstationspopulation", ()=>{ LoadStationsPopulationFile = stf.ReadStringBlock(null); }), }); } diff --git a/Source/Orts.Simulation/Simulation/AIs/AI.cs b/Source/Orts.Simulation/Simulation/AIs/AI.cs index ee65843cd..61b605390 100644 --- a/Source/Orts.Simulation/Simulation/AIs/AI.cs +++ b/Source/Orts.Simulation/Simulation/AIs/AI.cs @@ -922,8 +922,22 @@ public AITrain CreateAITrainDetail(Service_Definition sd, Traffic_Service_Defini if (Simulator.Activity != null && car is MSTSSteamLocomotive mstsSteamLocomotive) { + mstsSteamLocomotive.CombinedTenderWaterVolumeUKG = ORTS.Common.Kg.ToLb(mstsSteamLocomotive.MaxLocoTenderWaterMassKG) / 10.0f * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelWater / 100.0f; - mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; + + // Adjust fuel stocks depending upon fuel used - in Activity mode + if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Wood) + { + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelWood / 100.0f; + } + else if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Oil) + { + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelDiesel / 100.0f; + } + else // defaults to coal fired + { + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; + } } if (train.InitialSpeed != 0 && car is MSTSLocomotive loco) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs index 10b839584..341f33633 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs @@ -186,7 +186,7 @@ public enum SoundState bool WaterScoopSoundOn = false; public float MaxTotalCombinedWaterVolumeUKG; public MSTSNotchController WaterController = new MSTSNotchController(0, 1, 0.01f); - public float CombinedTenderWaterVolumeUKG // Decreased by running injectors and increased by refilling + public float CombinedTenderWaterVolumeUKG // Decreased by running injectors or pumps and increased by refilling { get { return WaterController.CurrentValue * MaxTotalCombinedWaterVolumeUKG; } set { WaterController.CurrentValue = value / MaxTotalCombinedWaterVolumeUKG; } diff --git a/Source/Orts.Simulation/Simulation/Simulator.cs b/Source/Orts.Simulation/Simulation/Simulator.cs index 47751b16c..548e55c10 100644 --- a/Source/Orts.Simulation/Simulation/Simulator.cs +++ b/Source/Orts.Simulation/Simulation/Simulator.cs @@ -1309,11 +1309,26 @@ private Train InitializePlayerTrain() if (Activity != null && mstsDieselLocomotive != null) mstsDieselLocomotive.DieselLevelL = mstsDieselLocomotive.MaxDieselLevelL * Activity.Tr_Activity.Tr_Activity_Header.FuelDiesel / 100.0f; + + var mstsSteamLocomotive = car as MSTSSteamLocomotive; if (Activity != null && mstsSteamLocomotive != null) { mstsSteamLocomotive.CombinedTenderWaterVolumeUKG = (Kg.ToLb(mstsSteamLocomotive.MaxLocoTenderWaterMassKG) / 10.0f) * Activity.Tr_Activity.Tr_Activity_Header.FuelWater / 100.0f; - mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; + + // Adjust fuel stocks depending upon fuel used - in Explore mode + if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Wood) + { + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelWood / 100.0f; + } + else if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Oil) + { + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelDiesel / 100.0f; + } + else // defaults to coal fired + { + mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelCoal / 100.0f; + } } } catch (Exception error) From 0037ba2095c4ae9a420c853c5083f7bda4018887 Mon Sep 17 00:00:00 2001 From: peternewell Date: Thu, 4 Jul 2024 16:33:18 +1000 Subject: [PATCH 22/25] Correct a parameter --- Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs index e9b95e9c1..ea8c316fa 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs @@ -1205,7 +1205,7 @@ public virtual void Parse(string lowercasetoken, STFReader stf) case "wagon(ortsauxtenderwatermass": AuxTenderWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortstenderwagonwoodmass": case "wagon(ortstenderwagoncoalmass": TenderWagonMaxFuelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; - case "wagon(ortsmaxtenderfueloilvolume": TenderWagonMaxOilVolumeL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; + case "wagon(ortstenderfueloilvolume": TenderWagonMaxOilVolumeL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; case "wagon(ortstenderwagonwatermass": TenderWagonMaxWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortsheatingwindowderatingfactor": WindowDeratingFactor = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "wagon(ortsheatingcompartmenttemperatureset": DesiredCompartmentTempSetpointC = stf.ReadFloatBlock(STFReader.UNITS.Temperature, null); break; From 72ddad85d1a94b292457cf7df21e63a31f7f97c9 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sun, 7 Jul 2024 13:26:26 +1000 Subject: [PATCH 23/25] Add manual information --- Source/Documentation/Manual/physics.rst | 49 ++++++++++++++++++- .../Simulation/RollingStocks/MSTSWagon.cs | 2 +- 2 files changed, 49 insertions(+), 2 deletions(-) diff --git a/Source/Documentation/Manual/physics.rst b/Source/Documentation/Manual/physics.rst index 0b5df8401..36ee037aa 100644 --- a/Source/Documentation/Manual/physics.rst +++ b/Source/Documentation/Manual/physics.rst @@ -1224,6 +1224,7 @@ locomotive can be thought of in terms of the following broadly defined components: - Boiler and Fire (Heat conversion) +- Fuel Type - Cylinder (Work done) Boiler and Fire (Heat conversion) @@ -1246,7 +1247,7 @@ the Boiler Evaporation Area. contact with the boiler and the heat tubes running through the boiler. This area determined the amount of heat that could be transferred to the water in the boiler. As a rule of thumb a boiler could produce - approximately 12-15 lbs/h of steam per ft\ :sup:`2` of evaporation area. + approximately 12-15 lbs/h of steam per ft\ :sup:`2` of evaporation area (coal fired). - *Boiler Superheater Area* -- Typically modern steam locomotives are superheated, whereas older locomotives used only saturated steam. Superheating is the process of putting more heat into the steam @@ -1255,6 +1256,52 @@ the Boiler Evaporation Area. in steam and fuel usage. In other words a superheated locomotive tended to be more efficient then a saturated locomotive. +Fuel Type +......... + +Different fuel types will produce different levels of heat. For example, +Coal has a fuel calorific value of around 13,800 BTU/lb, whereas Wood may +have values of between 3,000 and 7,000 BTU/lb (depending upon the condition +of the wood fuel), and Oil (Diesel) may have a value up around 17,000 BTU/lb. + +Hence the variations in fuel calorific value can dramatically impact the +amount of steam that it is able to produce and ultimately the performance of +the steam locomotive. + +Hence Open Rails supports the use of different fuel types for steam locomotives, +and these different fuel types can be configured with the following parameters. + +``ORTSSteamLocomotiveFuelType`` - indicates the type of fuel used by the locomotive - +currently Wood, Coal or Oil are available. Defaults to Coal. + +``ORTSMaxTenderWoodMass`` - amount of wood mass in tender (in ENG file) + +``ORTSTenderWagonWoodMass`` - amount of wood mass in tender (in tender WAG file - overwrites +the above value - non mandatory). + +``ORTSSteamLocomotiveFeedwaterSystemType`` - Older steam locomotives were fitted with +motion pumps to transfer water from the tender into the boiler. Steam injectors were +fitted to new locomotives from the 1860s. Currently MotionPump or Injector available. +Defaults to Injector. + +*FuelWood* is used in any IntakePoint statements used, and route fuel points need to +be set as wood type. + +``ORTSFuelOilHeatingRequired`` - some locomotives required the oil in the tender to be +steam heated, set to 1 (true) if this is the case. Defaults to false. + +``ORTSFuelOilSpecificGravity`` - specific gravity of the oil used as fuel on the locomotive. + +``ORTSMaxTenderFuelOilVolume`` - volume of oil carried in the tender, typically in gallons +or litres. (in ENG file) + +``ORTSTenderWagonFuelOilVolume`` - volume of oil carried in the tender, typically in gallons +or litres. (in tender WAG file - overwrites the above value - non mandatory). + +*FuelDiesel* - is used in any IntakePoint statements, and route fuel points need to be set +as diesel type. + + Cylinder (Work done) .................... diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs index ea8c316fa..c10787c0a 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs @@ -1205,7 +1205,7 @@ public virtual void Parse(string lowercasetoken, STFReader stf) case "wagon(ortsauxtenderwatermass": AuxTenderWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortstenderwagonwoodmass": case "wagon(ortstenderwagoncoalmass": TenderWagonMaxFuelMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; - case "wagon(ortstenderfueloilvolume": TenderWagonMaxOilVolumeL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; + case "wagon(ortstenderwagonfueloilvolume": TenderWagonMaxOilVolumeL = stf.ReadFloatBlock(STFReader.UNITS.Volume, null); break; case "wagon(ortstenderwagonwatermass": TenderWagonMaxWaterMassKG = stf.ReadFloatBlock(STFReader.UNITS.Mass, null); break; case "wagon(ortsheatingwindowderatingfactor": WindowDeratingFactor = stf.ReadFloatBlock(STFReader.UNITS.None, null); break; case "wagon(ortsheatingcompartmenttemperatureset": DesiredCompartmentTempSetpointC = stf.ReadFloatBlock(STFReader.UNITS.Temperature, null); break; From da7663bd91c0fbb70299b6174be13d02a9b149e9 Mon Sep 17 00:00:00 2001 From: peternewell Date: Sun, 7 Jul 2024 14:13:02 +1000 Subject: [PATCH 24/25] Add sound triggers to manual --- Source/Documentation/Manual/sound.rst | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/Source/Documentation/Manual/sound.rst b/Source/Documentation/Manual/sound.rst index e1b7a465a..da7b313d1 100644 --- a/Source/Documentation/Manual/sound.rst +++ b/Source/Documentation/Manual/sound.rst @@ -147,6 +147,10 @@ trigger the above triggers. ========= ============================================================================================================================================================== Trigger Function ========= ============================================================================================================================================================== + 90 WaterPump1ON - triggered whenever motion pump 1 turns on + 91 WaterPump1OFF - triggered whenever motion pump 1 turns off + 92 WaterPump2ON - triggered whenever motion pump 2 turns on + 93 WaterPump2OFF - triggered whenever motion pump 2 turns off 101 GearUp : for gear-based engines, triggered by the ```` key, propagated to all gear-based diesel engines of a train and run also for AI trains 102 GearDown : for gear-based engines, triggered by the ```` key, propagated to all gear-based diesel engines of a train and run also for AI trains 103 ReverserToForwardBackward : reverser moved towards the forward or backward position From 8fdc42b6fff8b9133e565f2bfd4764a42a7bc83c Mon Sep 17 00:00:00 2001 From: peternewell Date: Sun, 14 Jul 2024 12:49:16 +1000 Subject: [PATCH 25/25] Suggested reviewer changes --- Source/Orts.Simulation/Simulation/AIs/AI.cs | 4 +-- .../RollingStocks/MSTSLocomotive.cs | 19 ------------- .../RollingStocks/MSTSSteamLocomotive.cs | 27 ++++++++++++++++--- .../Simulation/RollingStocks/MSTSWagon.cs | 9 +++---- .../Orts.Simulation/Simulation/Simulator.cs | 4 +-- 5 files changed, 32 insertions(+), 31 deletions(-) diff --git a/Source/Orts.Simulation/Simulation/AIs/AI.cs b/Source/Orts.Simulation/Simulation/AIs/AI.cs index 61b605390..1b8245af5 100644 --- a/Source/Orts.Simulation/Simulation/AIs/AI.cs +++ b/Source/Orts.Simulation/Simulation/AIs/AI.cs @@ -926,11 +926,11 @@ public AITrain CreateAITrainDetail(Service_Definition sd, Traffic_Service_Defini mstsSteamLocomotive.CombinedTenderWaterVolumeUKG = ORTS.Common.Kg.ToLb(mstsSteamLocomotive.MaxLocoTenderWaterMassKG) / 10.0f * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelWater / 100.0f; // Adjust fuel stocks depending upon fuel used - in Activity mode - if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Wood) + if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSSteamLocomotive.SteamLocomotiveFuelTypes.Wood) { mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelWood / 100.0f; } - else if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Oil) + else if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSSteamLocomotive.SteamLocomotiveFuelTypes.Oil) { mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Simulator.Activity.Tr_Activity.Tr_Activity_Header.FuelDiesel / 100.0f; } diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs index 341f33633..afd83b9c8 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSLocomotive.cs @@ -239,25 +239,6 @@ public float CurrentLocomotiveSteamHeatBoilerWaterCapacityL //float DebugSpeed = 5.0f; // Initialise at 5 mph //float DebugTimer = 0.0f; - public enum SteamLocomotiveFuelTypes - { - Unknown, - Oil, - Wood, // not used at the moment - Coal, // defaults to coal - } - - public SteamLocomotiveFuelTypes SteamLocomotiveFuelType; - - public enum SteamLocomotiveFeedWaterSystemTypes - { - Unknown, - MotionPump, - SteamInjector, // not used at the moment - } - - public SteamLocomotiveFeedWaterSystemTypes SteamLocomotiveFeedWaterType; - // Adhesion parameters public enum SlipControlType { diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs index 205cc57f8..2895cea91 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSSteamLocomotive.cs @@ -84,6 +84,8 @@ using Orts.Simulation.Simulation.RollingStocks.SubSystems.PowerSupplies; using Orts.Simulation.RollingStocks.SubSystems.PowerTransmissions; using SharpDX.Direct3D9; +using Orts.Simulation.RollingStocks; +using static Orts.Simulation.RollingStocks.MSTSSteamLocomotive; namespace Orts.Simulation.RollingStocks { @@ -684,11 +686,30 @@ public float TenderFuelMassKG // Decreased by firing and increased float ConnectRodLengthFt = 10.8f; float RodCoGFt = 4.32f; // 0.4 from crank end of rod - #endregion + public enum SteamLocomotiveFuelTypes + { + Unknown, + Oil, + Wood, + Coal, // defaults to coal + } + + public SteamLocomotiveFuelTypes SteamLocomotiveFuelType; + + public enum SteamLocomotiveFeedWaterSystemTypes + { + Unknown, + MotionPump, + SteamInjector, // not used at the moment + } + + public SteamLocomotiveFeedWaterSystemTypes SteamLocomotiveFeedWaterType; + +#endregion - #region Variables for visual effects (steam, smoke) +#region Variables for visual effects (steam, smoke) - public readonly SmoothedData StackSteamVelocityMpS = new SmoothedData(2); +public readonly SmoothedData StackSteamVelocityMpS = new SmoothedData(2); public float StackSteamVolumeM3pS; public float StackParticleDurationS; public float StackCount; diff --git a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs index c10787c0a..2a155c4ea 100644 --- a/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs +++ b/Source/Orts.Simulation/Simulation/RollingStocks/MSTSWagon.cs @@ -52,7 +52,6 @@ using System.Diagnostics; using System.IO; using System.Linq; -using static Orts.Simulation.RollingStocks.MSTSLocomotive; using Event = Orts.Common.Event; namespace Orts.Simulation.RollingStocks @@ -148,7 +147,7 @@ public enum WindowState public float WagonFrontalAreaM2; // Frontal area of wagon public float TrailLocoResistanceFactor; // Factor to reduce base and wind resistance if locomotive is not leading - based upon original Davis drag coefficients - bool TenderWeightInitialize = true; + bool TenderWeightInitialized = false; float TenderWagonMaxFuelMassKG; float TenderWagonMaxOilMassL; float TenderWagonMaxWaterMassKG; @@ -1969,7 +1968,7 @@ public override void Update(float elapsedClockSeconds) ConfirmSteamLocomotiveTender(); // Confirms that a tender is connected to the steam locomotive // Adjusts water and coal mass based upon values assigned to the tender found in the WAG file rather then those defined in ENG file. - if (WagonType == WagonTypes.Tender && TenderWeightInitialize && (TenderWagonMaxFuelMassKG != 0 || TenderWagonMaxOilMassL != 0) && TenderWagonMaxWaterMassKG != 0) + if (WagonType == WagonTypes.Tender && !TenderWeightInitialized && (TenderWagonMaxFuelMassKG != 0 || TenderWagonMaxOilMassL != 0) && TenderWagonMaxWaterMassKG != 0) { // Find the associated steam locomotive for this tender @@ -1993,7 +1992,7 @@ public override void Update(float elapsedClockSeconds) TendersSteamLocomotive.MaxLocoTenderWaterMassKG = TenderWagonMaxWaterMassKG; - if (TendersSteamLocomotive.SteamLocomotiveFuelType == SteamLocomotiveFuelTypes.Oil) + if (TendersSteamLocomotive.SteamLocomotiveFuelType == MSTSSteamLocomotive.SteamLocomotiveFuelTypes.Oil) { TendersSteamLocomotive.MaxTenderFuelMassKG = TendersSteamLocomotive.MaxTenderOilMassL * TendersSteamLocomotive.OilSpecificGravity; } @@ -2010,7 +2009,7 @@ public override void Update(float elapsedClockSeconds) } // Rest flag so that this loop is not executed again - TenderWeightInitialize = false; + TenderWeightInitialized = true; } UpdateTenderLoad(); // Updates the load physics characteristics of tender and aux tender diff --git a/Source/Orts.Simulation/Simulation/Simulator.cs b/Source/Orts.Simulation/Simulation/Simulator.cs index 548e55c10..1d5c67ada 100644 --- a/Source/Orts.Simulation/Simulation/Simulator.cs +++ b/Source/Orts.Simulation/Simulation/Simulator.cs @@ -1317,11 +1317,11 @@ private Train InitializePlayerTrain() mstsSteamLocomotive.CombinedTenderWaterVolumeUKG = (Kg.ToLb(mstsSteamLocomotive.MaxLocoTenderWaterMassKG) / 10.0f) * Activity.Tr_Activity.Tr_Activity_Header.FuelWater / 100.0f; // Adjust fuel stocks depending upon fuel used - in Explore mode - if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Wood) + if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSSteamLocomotive.SteamLocomotiveFuelTypes.Wood) { mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelWood / 100.0f; } - else if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSLocomotive.SteamLocomotiveFuelTypes.Oil) + else if (mstsSteamLocomotive.SteamLocomotiveFuelType == MSTSSteamLocomotive.SteamLocomotiveFuelTypes.Oil) { mstsSteamLocomotive.TenderFuelMassKG = mstsSteamLocomotive.MaxTenderFuelMassKG * Activity.Tr_Activity.Tr_Activity_Header.FuelDiesel / 100.0f; }