Jump to content

The curious case of the missing jet engine thrust.


CastleBravoTest

Recommended Posts

 

This is my first post here. I was convinced by some friends to post the results of one of my investigations on this forum. It will be partly a question about things I have yet to fully discover, a bit of an explanation of how jet engines work. I will tell this investigation in chronological order explaining each step so if I have an error anywhere in my understanding feel free dear reader to point it out.

I started this little investigation because of an unrelated efficiency calculation I wanted to do however it quickly spiraled into a bit of a rabit hole. I wanted to mathematically determine the most efficient jet engines for each flight regime but for this I needed a way to create a function for the thrust multiplier given the flight conditions(pressure and mach). To my understanding at the time the thrust was simply a function of two values those being the mach multiplier and the pressure multiplier. In other words:

Thrust = mach_curve x pressure_curve

Nice and simple right? The wiki shows these two curves as looking as follows for the whiplash:

 600px-J-X4_Whiplash_Turbo_Ramjet_Engine_

600px-J-X4_Whiplash_Turbo_Ramjet_Engine_

The data for these two curves are easy enough to find in the game files and look as follows (for the whiplash at least):

velCurve
        {
            key = 0 1 0 0
            key = 0.2 0.98 0 0
            key = 0.72 1.716 2.433527 2.433527
            key = 1.36 3.2 1.986082 1.986082
            key = 2.15 4.9 1.452677 1.452677
            key = 3 5.8 0.0005786046 0.0005786046
            key = 4.5 3 -4.279616 -4.279616
            key = 5.5 0 -0.02420209 0
        }
        atmCurve
        {
            // definite 'kink' to the curve at high altitude, compared to flatter BJE curve
            key = 0 0 0 0
            key = 0.045 0.166 4.304647 4.304647
            key = 0.16 0.5 0.5779132 0.5779132
            key = 0.5 0.6 0.4809403 0.4809403
            key = 1 1 1.013946 0
        }

The way in which this is stored is not entirely obvious but through some googling i figured out it uses a technique called Cubic Hermite spline also known as Cubic Hermite interpolation [source for the curious reader: https://en.wikipedia.org/wiki/Cubic_Hermite_spline#:~:text=In numerical analysis%2C a cubic,of the corresponding domain interval.]. Each point (represented by a key) is defined by 4 properties which are the x, y coordinates and incoming and outgoing tangents in that order. 

To do this numerically in the program I was using(aka MATLAB) I had the option to either implement this interpolation from scratch or use one of the preexisting interpolation methods available to me. The functions that MATLAB had available were:

spline()
pchip()
makima()

I first decided on spline()  and through that I aquired functions that produced the following curves:

lIqjL8U.png

and:
cesbN1M.png

These curves look good but they differ slightly from the ones found in the wiki.  This error could be through a number of reasons, it could be that i used the wrong interpolation function, it could be that the wiki is wrong or it could even be that both I and the wiki are wrong. I tried googling for which interpolation the devs used or which interpolation unity would use by default however very predictably this gave no results (Like many things in ksp odds are they implemented a custom interpolation method). Unable to find the answers I needed online i then proceeded to look at the functions themselves available to me. Each giving ever so slightly different behavior. If i am to show the difference in the interpolation method i am going to borrow an example from the online documentation for these functions:[image source: https://se.mathworks.com/help/matlab/ref/spline.html?searchHighlight=spline&s_tid=srchtitle_support_results_1_spline]  

DataInterpolationComparisonExample_02.pn

As you can see from this image each interpolation method does it ever so slightly differently (though granted makima and spline are almost identical). The pchip()  method seems most similar to the one used in the wiki however at this point I was through taking peoples word for it so i decided on a different course of action.  To determine which of these methods I should use for my function I decided to conduct a little experiment. I slapped together a tiny test rig which  would help me (with the aid of the autopilot mod) determine the properties of such a curve. The vehicle is seen below but there is not much to say about it:

2xnIeeM.jpg

This vehicle flew on 100% throttle with infinite fuel turned on to limit the variables. The autopilot mod used was atmospheric autopilot [link: https://spacedock.info/mod/683/AtmosphereAutopilot]. This autopilot held 200m above sea level and flew only in the ocean east of the KSC. It held the speed asigned to it in mach and i confirmed the mach number both with the ingame aero data in the gui setting and through atmospheric autopilot. Then i sat down to record the data I needed. Here is a table (oh and since this whole journey started with efficiency I decided it could not hurt to record the fuel consumption as well). I did a little tweaking to the PID parameters mid flight to ensure zero steady state error and also a fast enough response.  Here is the raw data:

Raw experimental data
Mach Thrust [kN] fuel drain [u/s]
1 292.2 1.372
1.2 297.6 1.487
2 350.2 1.785
2.2 357.2 1.821
2.4 363 1.852
2.6 368.4 1.878
2.8 371.8 1.89
3 375.6 1.915
3.1 375.4 1.914
3.2 374.8 1.910
3.3 374.2 1.908
3.4 372.8 1.901
3.5 371.1 1.89
3.6 368.8 1.88
3.7 365.8 1.865
3.8 362.1 1.846
3.9 357.6 1.823
4 352.0 1.795

I would have sampled higher but this is where my poor slapped together craft reached its maximum velocity and at any rate I had enough data to work with. I took extra readings in the top of that "hump" to check if it is flat or curved. Something else stood out about this though which became all the more apparent when this data was subsiquently graphed:

 0rfqMcE.png

There seems to be quite a bit of thrust missing from this. In other words we have a mystery multiplier at play somewhere in the code. It is less simple than the equation for thrust I stated near the beginning.  That said from the smoothness of this curve especially around the mach 3 region i can at the very least conclude that the interpolation method in the wiki is wrong. To Investigate the thrust discrepency further I decided to plot the theoretical data divided by the experimental data and the result looks as follows:

GuetKUw.png

The results of this show that our mystery multiplier is nonlinear making life difficult. That said I have tracked down the identity of this multiplier. It can be found in a seemingly unrelated spot in the data for the engine. The parameters section for this engine is as follows:

// Jet params
atmChangeFlow = True
useVelCurve = True
useAtmCurve = True
flowMultCap = 2.0
machLimit = 2.5
machHeatMult = 6.0

Can you spot it? well it may not necessarily be obvious at first glance but looking back on the raw data it is curious that the fuel flow never goes above the flowMultCap  set by the game. That is because this is our culprit. The specific impulse is tied to both thrust and fuel flow rate. The equation for which looks like this:

Isp = T * (dm/dt)-1 *g-1 

where:
T  is "thrust" expressed in newtons,
dm/dt is the mass flwo rate of the fuel into the engine expressed in kg/s
g  is the gravitational acceleration at the surface (9.81 m/s2)
 

The equation for Isp Ties the two values together and since the thrust can't rise without an increase in fuel flow rate and the fuel flow rate can't rise due to the limiter the end result is a situatuation where the thrust is effectivly capped at low altitudes.  Just to show this equation is accurate i am going to take a random set of values (say, @mach 3 ) and confirm the fuel flow rate from the thrust value and Isp:
For the whiplash at this point
Isp = 4000
T = 375.6 [kN] = 375600  [N]
Rarranging the equation for the specific impulse gives:
(dm/dt) = T * Isp-1 *g-1 
substituting in the values gives:
dm/dt = 375600 / (4000*9.81) => (dm/dt)theoretical = 9.572 [kg/s]
To convert this to u/s we need to know the density which can be found on the wiki [https://wiki.kerbalspaceprogram.com/wiki/Liquid_fuel] to be 5kg/unit so:
  (dm/dt)theoretical = 9.572 [kg/s] / 5 [kg/u] =>  (dm/dt)theoretical  = 1.9144 [u/s]
this value is similar to the experimental value:
(dm/dt)experimental = 1.9148 [u/s]
The discrepency in the last digit can be ignored due to me doing this calculation with values with up to 4 significant figures and the limiting value (g) has 3 significant figures. Thus the experimental and theoretical values can be seen as equal and thus the equation works.

When I went in and changed the flowMultCap to a much higher number then the whiplash behaved as expected reaching its full thrust multiplier so it iscertainly the culprit here. When looking at the KSP API [source: https://www.kerbalspaceprogram.com/ksp/api/class_module_engines.html] the description for this is:

  Quote

float ModuleEngines.flowMultCap = float.MaxValue
cap beyond which increases to flow multiplier aren't fully felt (start to taper off)

Expand  

This quote does not provide any actual details of how this tapering off multiplier is implemented or how it works in detail. I did a little more digging around this but found no actual answers so this is where this investigation of mine ends for now.  
 

For anybody still reading this i thank you for your time. I also shall ask the question I mentioned at the beginning but never actually asked throughout the entire text. Does anybody how this flowMultCap curve is implemented? maybe you have decompiled the code somewhere and you have it handy and can share? maybe you instantly know from the shape of the functions i have presented? maybe you are a dev and have the actual code infront of you? no matter how any answers to this are appriciated and would also appriciate the feedback to any errors you have seen me make during this investigation.

Thank you for reading.

Edited by CastleBravoTest
undesired strike through i could not get rid of¨. Further formatting errors
Link to comment
Share on other sites

The actual trueThrustMult you get once above the flowMultCap is:

trueThrustMult = flowMultCap + (thrustMult - flowMultCap) / (flowMultCapSharpness + (thrustMult - flowMultCap)/flowMultCap)

By default, flowMultCapSharpness is 2 and no stock engine has an alternate value

Working an example, the Whiplash has a flowMultcap of 2 and the default sharpness of 2.  So at a normalized density of 1 (for example, flying at sea level at 30 degrees latitude at "3 am" or 45min after midnight under Kerbin time) at a speed of exactly Mach 3, the supposed thrust mult of 5.8 would get transformed like so:

trueThrustMult = 2 + (5.8 - 2) / (2 + (5.8 - 2) / 2) = 2.97

 

EDIT: As an additional note, the atmCurve is not based on pressure, but on normalized density, with a normalized value of 1 = 1.225 km/m^3, which is the sea level density under the ISA (International Standard Atmosphere) and occurs at a temperature of 15C.  Interestingly, this results in jet engines not providing their full advertised thrust on the runway, as it is significantly warmer than 15C on Kerbin's equator, and so the atmosphere is less dense.

Edited by Lt_Duckweed
Link to comment
Share on other sites

  On 2/27/2024 at 9:16 PM, Lt_Duckweed said:

The actual trueThrustMult you get once above the flowMultCap is:

trueThrustMult = flowMultCap + (thrustMult - flowMultCap) / (flowMultCapSharpness + (thrustMult - flowMultCap)/flowMultCap)

By default, flowMultCapSharpness is 2 and no stock engine has an alternate value

Working an example, the Whiplash has a flowMultcap of 2 and the default sharpness of 2.  So at a normalized density of 1 (for example, flying at sea level at 30 degrees latitude at "3 am" or 45min after midnight under Kerbin time) at a speed of exactly Mach 3, the supposed thrust mult of 5.8 would get transformed like so:

trueThrustMult = 2 + (5.8 - 2) / (2 + (5.8 - 2) / 2) = 2.97

 

EDIT: As an additional note, the atmCurve is not based on pressure, but on normalized density, with a normalized value of 1 = 1.225 km/m^3, which is the sea level density under the ISA (International Standard Atmosphere) and occurs at a temperature of 15C.  Interestingly, this results in jet engines not providing their full advertised thrust on the runway, as it is significantly warmer than 15C on Kerbin's equator, and so the atmosphere is less dense.

Expand  

Thank you for your answer and your correction, I have plotted your values with the correction and it is a lot closer that said there is still a slight discrepency.  Your equation does not take into account the Isp of the engine and I am a bit confused by that. If i were to change the Isp to a very high number wont that mean that my fuel consumption is near zero for the thrust i am getting and thus this clamp should not engage? 

This is that same theoretical data and the corrected value plotted:
KtyOyvP.png
With experimental/theoretical plotted:
8sFcaU2.png
and with the new thrust-mach curve at these conditions:
fwA5255.png
I am wondering of the discrepency is due to the atmospheric conditions which are near the normalized density of 1 but not quite it or if there is a small error?

For completeness here is the matlab code for the thrust (please ignore my poor coding standards):
 

function Thrust = whiplash(M,atmp,tht)
baseThrust = 130 ;

% logic checks and input conditioning
M = abs(M);
lc = double(M >= 0).*double(M <= 5.5).*double(atmp > 0);



% velocity curve
velCurve = [...
		    0       1       0               0 ;
		    0.2     0.98    0               0 ;
		    0.72    1.716   2.433527        2.433527;
		    1.36    3.2     1.986082        1.986082;
			2.15    4.9     1.452677        1.452677;
			3       5.8     0.0005786046    0.0005786046;
			4.5     3      -4.279616       -4.279616 ;
			5.5     0      -0.02420209      0];

atmCurve = [...
    		0       0       0           0 ;
			0.045   0.166   4.304647    4.304647 ;
			0.16    0.5     0.5779132   0.5779132 ;
			0.5     0.6     0.4809403   0.4809403 ;
		    1       1       1.013946    0 ;];


% velocity multiplier
vel_mult = ppval(spline(velCurve(:, 1), [velCurve(1, 3); velCurve(:, 2); velCurve(end, 4)]), M);

% atmospheric multiplier
atm_mult = ppval(spline(atmCurve(:, 1), [atmCurve(1, 3); atmCurve(:, 2); atmCurve(end, 4)]), atmp);



% thrust multiplier 
Thrust_MP = atm_mult.*vel_mult.*tht.*lc ;
Thrust = baseThrust.*tht.*flowmult(2,Thrust_MP) ;
end

with the thrust  flowmultiplier coded as follows:

% thrust flow limit logic
function truemult = flowmult(cap,val)
sharp = 2 ;
truemult = cap + (val-cap)./(sharp + (val - cap)./(cap)) ;
end

all in matlab
 

Link to comment
Share on other sites

  On 2/28/2024 at 3:35 PM, CastleBravoTest said:

Your equation does not take into account the Isp of the engine and I am a bit confused by that. If i were to change the Isp to a very high number wont that mean that my fuel consumption is near zero for the thrust i am getting and thus this clamp should not engage? 

Expand  

I also just checked that statement by tweaking the ISP to an absurldy large number (100 000 s) and my fuel consumption was predictably near zero but the clamp engaged so i guess it isnt dependent on the fuel flow itself. Strange way to code it

Link to comment
Share on other sites

  On 2/28/2024 at 3:45 PM, CastleBravoTest said:

I also just checked that statement by tweaking the ISP to an absurldy large number (100 000 s) and my fuel consumption was predictably near zero but the clamp engaged so i guess it isnt dependent on the fuel flow itself. Strange way to code it

Expand  

Yes the the mults and clamps are just dimensionless constants.  You start with a value of 1, then take the density and velocity mults and go from there

Link to comment
Share on other sites

  On 2/27/2024 at 9:16 PM, Lt_Duckweed said:

trueThrustMult = flowMultCap + (thrustMult - flowMultCap) / (flowMultCapSharpness + (thrustMult - flowMultCap)/flowMultCap)

Expand  

That said there is still something that is not quite right about that especially at higher altitudes. Here is data at high altitude (somewhere when the clamp should not be engaged):

mpz4f5l.png

The error is significant. That said if i were to remove the clamp and plot this all again:

TdtYmoc.png

Here the data is almost striking. There must be something about that mathematics that we are missing that makes the clamp tend towards 1 at higher altitudes?

Also for this data I used a data logger mod that outputs a CSV file.  As this file has almost 100 rather poorly formatted rows I shall not post it here for brevity. Furthermore I used the normalized density as you corrected me to do. 

Link to comment
Share on other sites

I just also noticed that the last two graphs, the lables are back to front. The fashed line is the "theoretical value"  and the solid line is the experimental value. The point however remains. If the clamp is a piece of mathematics that is on constantly and in the state presented then the theoretical data does not fit the experimental one. The model is slighly off.

This is a computer simulation so the clamp function once all of its details are discovered I would expect to converge almost perfectly with the recorded data.

Basically this model has some flaws and some more investigation is needed. Perhaps the next step is to decompile the game and see if i can make sense of any of it (i am no expert in computer science so it remains to be seen if i can manage that)

Link to comment
Share on other sites

This thread is quite old. Please consider starting a new thread rather than reviving this one.

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.
Note: Your post will require moderator approval before it will be visible.

Guest
Reply to this topic...

×   Pasted as rich text.   Paste as plain text instead

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

×
×
  • Create New...