From 6d39c05f8b1993aa0bafaf2f9aca4618529a791b Mon Sep 17 00:00:00 2001 From: Denis Freund Date: Mon, 24 Oct 2022 21:25:43 +0200 Subject: [PATCH] adjust law, edit fmgc flightmodel --- src-a339/fbw/src/model/PitchNormalLaw.cpp | 2056 +++++++++++++++++ .../fmgc/src/guidance/vnav/FlightModel.ts | 287 +++ 2 files changed, 2343 insertions(+) create mode 100644 src-a339/fbw/src/model/PitchNormalLaw.cpp create mode 100644 src-a339/fmgc/src/guidance/vnav/FlightModel.ts diff --git a/src-a339/fbw/src/model/PitchNormalLaw.cpp b/src-a339/fbw/src/model/PitchNormalLaw.cpp new file mode 100644 index 000000000..d79fbd627 --- /dev/null +++ b/src-a339/fbw/src/model/PitchNormalLaw.cpp @@ -0,0 +1,2056 @@ +#include "PitchNormalLaw.h" +#include "rtwtypes.h" +#include +#include "look1_binlxpw.h" +#include "look2_binlxpw.h" + +const uint8_T PitchNormalLaw_IN_Flare_Reduce_Theta_c{ 1U }; + +const uint8_T PitchNormalLaw_IN_Flare_Set_Rate{ 2U }; + +const uint8_T PitchNormalLaw_IN_Flare_Store_Theta_c_deg{ 3U }; + +const uint8_T PitchNormalLaw_IN_Flight_High{ 4U }; + +const uint8_T PitchNormalLaw_IN_Flight_Low{ 5U }; + +const uint8_T PitchNormalLaw_IN_Ground{ 6U }; + +const uint8_T PitchNormalLaw_IN_NO_ACTIVE_CHILD{ 0U }; + +const uint8_T PitchNormalLaw_IN_frozen{ 1U }; + +const uint8_T PitchNormalLaw_IN_running{ 2U }; + +const uint8_T PitchNormalLaw_IN_automatic{ 1U }; + +const uint8_T PitchNormalLaw_IN_manual{ 2U }; + +const uint8_T PitchNormalLaw_IN_reset{ 3U }; + +const uint8_T PitchNormalLaw_IN_tracking{ 4U }; + +const uint8_T PitchNormalLaw_IN_flight_clean{ 1U }; + +const uint8_T PitchNormalLaw_IN_flight_flaps{ 2U }; + +const uint8_T PitchNormalLaw_IN_ground{ 3U }; + +const uint8_T PitchNormalLaw_IN_OFF{ 1U }; + +const uint8_T PitchNormalLaw_IN_ON{ 2U }; + +PitchNormalLaw::Parameters_PitchNormalLaw_T PitchNormalLaw::PitchNormalLaw_rtP{ + + { 0.0, 50.0, 100.0, 200.0 }, + + + { 0.0, 50.0, 100.0, 200.0 }, + + + { 0.0, 100.0, 150.0, 200.0, 250.0, 300.0, 400.0 }, + + + { 0.0, 50.0, 100.0, 200.0 }, + + + { 0.0, 50.0, 100.0, 200.0 }, + + + { 0.0, 50.0, 100.0, 200.0 }, + + + { 0.0, 90.0, 200, 400.0 }, + + + { 0.0, 0.06, 0.1, 0.2, 1.0 }, + + + { 0.0, 90.0, 200, 400.0 }, + + 0.05, + + 0.3, + + 5.0, + + 0.3, + + 5.0, + + 1.6, + + 1.0, + + 2.0, + + 2.0, + + 2.0, + + 2.0, + + 0.3, + + 5.0, + + 0.3, + + 5.0, + + 0.3, + + 5.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 1.0, + + 0.0, + + 0.0, + + 2.0, + + 0.0, + + 0.0, + + 30.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + -30.0, + + -30.0, + + + { 0.0, 0.0, -30.0, -30.0 }, + + + { 0.0, 0.0, -30.0, -30.0 }, + + + { 0.1, 0.1, 0.15, 0.2, 0.3, 0.5, 0.5 }, + + + { 0.0, 0.0, -30.0, -30.0 }, + + + { 0.0, 0.0, -30.0, -30.0 }, + + + { 0.0, 0.0, -30.0, -30.0 }, + + + { 1.0, 1.0, 0.5, 0.3 }, + + + { 1.0, 1.0, 0.5, 0.3 }, + + + { 1.0, 1.0, 0.5, 0.3 }, + + 17.0, + + 30.0, + + -0.2, + + -0.5, + + -0.5, + + -0.5, + + -1.0, + + -1.0, + + -0.25, + + -2.0, + + -1.0, + + -0.5, + + -0.33333333333333331, + + -4.0, + + -0.5, + + -45.0, + + 0.2, + + 0.5, + + 0.5, + + 0.5, + + 1.0, + + 1.0, + + 4.0, + + 2.0, + + 1.0, + + 0.5, + + 4.0, + + 2.0, + + 0.5, + + 45.0, + + true, + + 3.5, + + -11.0, + + 0.0, + + -0.4, + + -0.4, + + 1.4, + + -0.1, + + -0.6, + + 0.2, + + 0.75, + + -0.75, + + 1.5, + + 0.0, + + 0.0, + + + { -2.0, 0.0, 1.5 }, + + + { -1.0, 0.0, 1.0 }, + + 0.0, + + 1.0, + + 1.0, + + 0.0, + + 0.06, + + 0.0, + + -0.8, + + 1.0, + + 1.0, + + 0.0, + + 0.0, + + 30.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 1.0, + + 1.0, + + 0.0, + + 1.0, + + 0.0, + + -30.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.0, + + 0.2, + + 0.2, + + 1.0, + + -0.25, + + + { -2.0, 0.0, 1.5 }, + + + { -1.0, 0.0, 1.0 }, + + 0.017453292519943295, + + 0.017453292519943295, + + + { 100.0, 100.0, 100.0, 100.0, 100.0, 100.0, 100.0 }, + + + { 100.0, 150.0, 200.0, 250.0, 300.0, 350.0, 400.0 }, + + 2000.0, + + 100.0, + + 0.51444444444444448, + + 1.0, + + 0.017453292519943295, + + 100.0, + + 0.1019367991845056, + + + { 13.5, 13.5 }, + + + { 0.0, 350.0 }, + + 1.0, + + + { 0.5, 0.5 }, + + + { 0.0, 350.0 }, + + 1.0, + + -1.0, + + 0.06, + + 1.0, + + -1.0, + + 30.0, + + -30.0, + + 1.0, + + 0.017453292519943295, + + 100.0, + + 0.1019367991845056, + + + { 100.0, 100.0, 100.0, 100.0, 100.0, 100.0, 100.0 }, + + + { 100.0, 150.0, 200.0, 250.0, 300.0, 350.0, 400.0 }, + + 2000.0, + + 100.0, + + 0.51444444444444448, + + 1.0, + + + { 13.5, 13.5 }, + + + { 0.0, 350.0 }, + + + { 0.5, 0.5 }, + + + { 0.0, 350.0 }, + + 1.0, + + -1.0, + + 0.06, + + 1.0, + + -1.0, + + 30.0, + + -30.0, + + 0.0, + + 2.0, + + 0.0, + + -1.0, + + -1.2, + + 0.0, + + 2.0, + + 0.0, + + -0.8, + + 1.0, + + 1.0, + + 4.0, + + -4.0, + + 1.0, + + 0.0, + + 1.0, + + 1.0, + + 0.017453292519943295, + + 100.0, + + 0.1019367991845056, + + + { 100.0, 100.0, 100.0, 100.0, 100.0, 100.0, 100.0 }, + + + { 100.0, 150.0, 200.0, 250.0, 300.0, 350.0, 400.0 }, + + 2000.0, + + 100.0, + + 0.51444444444444448, + + 1.0, + + 0.0, + + 2.0, + + 0.0, + + 0.0, + + 2.0, + + 0.0, + + 33.0, + + -33.0, + + 0.017453292519943295, + + + { 13.5, 13.5 }, + + + { 0.0, 350.0 }, + + + { 0.5, 0.5 }, + + + { 0.0, 350.0 }, + + 1.0, + + -1.0, + + 0.06, + + 1.0, + + -1.0, + + 30.0, + + -30.0, + + 1.0, + + 0.017453292519943295, + + 100.0, + + 0.1019367991845056, + + + { 100.0, 100.0, 100.0, 100.0, 100.0, 100.0, 100.0 }, + + + { 100.0, 150.0, 200.0, 250.0, 300.0, 350.0, 400.0 }, + + 2000.0, + + 100.0, + + 0.51444444444444448, + + 1.0, + + + { 13.5, 13.5 }, + + + { 0.0, 350.0 }, + + + { 0.5, 0.5 }, + + + { 0.0, 350.0 }, + + 1.0, + + -1.0, + + 0.06, + + 1.0, + + -1.0, + + 30.0, + + -30.0, + + 1.0, + + 0.017453292519943295, + + 100.0, + + 0.1019367991845056, + + + { 100.0, 100.0, 100.0, 100.0, 100.0, 100.0, 100.0 }, + + + { 100.0, 150.0, 200.0, 250.0, 300.0, 350.0, 400.0 }, + + 2000.0, + + 100.0, + + 0.51444444444444448, + + 1.0, + + -15.0, + + 0.2, + + 0.25, + + -1.0, + + + { -2.0, 0.0, 1.5 }, + + + { -1.0, 0.0, 1.0 }, + + + { 13.5, 13.5 }, + + + { 0.0, 350.0 }, + + + { 0.5, 0.5 }, + + + { 0.0, 350.0 }, + + 1.0, + + -1.0, + + 0.06, + + 1.0, + + -1.0, + + 30.0, + + -30.0, + + 0.0, + + 0.076923076923076927, + + 1.0, + + 0.0, + + 1.0, + + 1.0, + + + { 0.0, 0.0, 0.0, -0.75, -0.75, 0.0, 0.0, 0.0, -0.75, -0.75, 0.0, 0.0, 0.0, 0.0, -0.75, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, + 0.0, 0.0, 0.0, 0.0 }, + + + { -1.0, 0.0, 6.0, 11.0, 15.0 }, + + + { -1.0, 8.0, 18.0, 28.0, 40.0 }, + + 1.0, + + 0.0, + + + { -9.0, 0.0, 9.0 }, + + + { -1.0, 0.0, 1.0 }, + + -2.0, + + 4.0, + + 0.2, + + 1.0, + + 2.5, + + -1.5, + + 2.0, + + 0.0, + + 1.0, + + 0.0, + + 1.0, + + 17.0, + + -30.0, + + + { 4U, 4U }, + + 1U, + + 1U +}; + +void PitchNormalLaw::PitchNormalLaw_LagFilter_Reset(rtDW_LagFilter_PitchNormalLaw_T *localDW) +{ + localDW->pY_not_empty = false; + localDW->pU_not_empty = false; +} + +void PitchNormalLaw::PitchNormalLaw_LagFilter(const real_T *rtu_U, real_T rtu_C1, const real_T *rtu_dt, real_T *rty_Y, + rtDW_LagFilter_PitchNormalLaw_T *localDW) +{ + real_T ca; + real_T denom_tmp; + if ((!localDW->pY_not_empty) || (!localDW->pU_not_empty)) { + localDW->pU = *rtu_U; + localDW->pU_not_empty = true; + localDW->pY = *rtu_U; + localDW->pY_not_empty = true; + } + + denom_tmp = *rtu_dt * rtu_C1; + ca = denom_tmp / (denom_tmp + 2.0); + *rty_Y = (2.0 - denom_tmp) / (denom_tmp + 2.0) * localDW->pY + (*rtu_U * ca + localDW->pU * ca); + localDW->pY = *rty_Y; + localDW->pU = *rtu_U; +} + +void PitchNormalLaw::PitchNormalLaw_RateLimiter_Reset(rtDW_RateLimiter_PitchNormalLaw_T *localDW) +{ + localDW->pY_not_empty = false; +} + +void PitchNormalLaw::PitchNormalLaw_RateLimiter(real_T rtu_u, real_T rtu_up, real_T rtu_lo, const real_T *rtu_Ts, real_T + rtu_init, real_T *rty_Y, rtDW_RateLimiter_PitchNormalLaw_T *localDW) +{ + if (!localDW->pY_not_empty) { + localDW->pY = rtu_init; + localDW->pY_not_empty = true; + } + + localDW->pY += std::fmax(std::fmin(rtu_u - localDW->pY, std::abs(rtu_up) * *rtu_Ts), -std::abs(rtu_lo) * *rtu_Ts); + *rty_Y = localDW->pY; +} + +void PitchNormalLaw::PitchNormalLaw_eta_trim_limit_lofreeze_Reset(rtDW_eta_trim_limit_lofreeze_PitchNormalLaw_T *localDW) +{ + localDW->frozen_eta_trim_not_empty = false; +} + +void PitchNormalLaw::PitchNormalLaw_eta_trim_limit_lofreeze(const real_T *rtu_eta_trim, const boolean_T *rtu_trigger, + real_T *rty_y, rtDW_eta_trim_limit_lofreeze_PitchNormalLaw_T *localDW) +{ + if ((!*rtu_trigger) || (!localDW->frozen_eta_trim_not_empty)) { + localDW->frozen_eta_trim = *rtu_eta_trim; + localDW->frozen_eta_trim_not_empty = true; + } + + *rty_y = localDW->frozen_eta_trim; +} + +void PitchNormalLaw::PitchNormalLaw_LagFilter_i_Reset(rtDW_LagFilter_PitchNormalLaw_d_T *localDW) +{ + localDW->pY_not_empty = false; + localDW->pU_not_empty = false; +} + +void PitchNormalLaw::PitchNormalLaw_LagFilter_n(real_T rtu_U, real_T rtu_C1, const real_T *rtu_dt, real_T *rty_Y, + rtDW_LagFilter_PitchNormalLaw_d_T *localDW) +{ + real_T ca; + real_T denom_tmp; + if ((!localDW->pY_not_empty) || (!localDW->pU_not_empty)) { + localDW->pU = rtu_U; + localDW->pU_not_empty = true; + localDW->pY = rtu_U; + localDW->pY_not_empty = true; + } + + denom_tmp = *rtu_dt * rtu_C1; + ca = denom_tmp / (denom_tmp + 2.0); + *rty_Y = (2.0 - denom_tmp) / (denom_tmp + 2.0) * localDW->pY + (rtu_U * ca + localDW->pU * ca); + localDW->pY = *rty_Y; + localDW->pU = rtu_U; +} + +void PitchNormalLaw::PitchNormalLaw_WashoutFilter_Reset(rtDW_WashoutFilter_PitchNormalLaw_T *localDW) +{ + localDW->pY_not_empty = false; + localDW->pU_not_empty = false; +} + +void PitchNormalLaw::PitchNormalLaw_WashoutFilter(real_T rtu_U, real_T rtu_C1, const real_T *rtu_dt, real_T *rty_Y, + rtDW_WashoutFilter_PitchNormalLaw_T *localDW) +{ + real_T ca; + real_T denom_tmp; + if ((!localDW->pY_not_empty) || (!localDW->pU_not_empty)) { + localDW->pU = rtu_U; + localDW->pU_not_empty = true; + localDW->pY = rtu_U; + localDW->pY_not_empty = true; + } + + denom_tmp = *rtu_dt * rtu_C1; + ca = 2.0 / (denom_tmp + 2.0); + *rty_Y = (2.0 - denom_tmp) / (denom_tmp + 2.0) * localDW->pY + (rtu_U * ca - localDW->pU * ca); + localDW->pY = *rty_Y; + localDW->pU = rtu_U; +} + +void PitchNormalLaw::PitchNormalLaw_RateLimiter_l_Reset(rtDW_RateLimiter_PitchNormalLaw_o_T *localDW) +{ + localDW->pY_not_empty = false; +} + +void PitchNormalLaw::PitchNormalLaw_RateLimiter_c(const real_T *rtu_u, real_T rtu_up, real_T rtu_lo, const real_T + *rtu_Ts, real_T rtu_init, real_T *rty_Y, rtDW_RateLimiter_PitchNormalLaw_o_T *localDW) +{ + if (!localDW->pY_not_empty) { + localDW->pY = rtu_init; + localDW->pY_not_empty = true; + } + + localDW->pY += std::fmax(std::fmin(*rtu_u - localDW->pY, std::abs(rtu_up) * *rtu_Ts), -std::abs(rtu_lo) * *rtu_Ts); + *rty_Y = localDW->pY; +} + +void PitchNormalLaw::PitchNormalLaw_VoterAttitudeProtection(real_T rtu_input, real_T rtu_input_l, real_T rtu_input_o, + real_T *rty_vote) +{ + real_T rtb_TmpSignalConversionAtSFunctionInport1[3]; + int32_T rtu_input_0; + rtb_TmpSignalConversionAtSFunctionInport1[0] = rtu_input; + rtb_TmpSignalConversionAtSFunctionInport1[1] = rtu_input_l; + rtb_TmpSignalConversionAtSFunctionInport1[2] = rtu_input_o; + if (rtu_input < rtu_input_l) { + if (rtu_input_l < rtu_input_o) { + rtu_input_0 = 1; + } else if (rtu_input < rtu_input_o) { + rtu_input_0 = 2; + } else { + rtu_input_0 = 0; + } + } else if (rtu_input < rtu_input_o) { + rtu_input_0 = 0; + } else if (rtu_input_l < rtu_input_o) { + rtu_input_0 = 2; + } else { + rtu_input_0 = 1; + } + + *rty_vote = rtb_TmpSignalConversionAtSFunctionInport1[rtu_input_0]; +} + +void PitchNormalLaw::init(void) +{ + PitchNormalLaw_DWork.Delay_DSTATE = PitchNormalLaw_rtP.RateLimiterDynamicVariableTs_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_h = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_n = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_c = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_l = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_l; + PitchNormalLaw_DWork.Delay_DSTATE_k = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_o; + PitchNormalLaw_DWork.Delay_DSTATE_d = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_d; + PitchNormalLaw_DWork.Delay_DSTATE_f = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_h; + PitchNormalLaw_DWork.Delay_DSTATE_g = PitchNormalLaw_rtP.Delay_InitialCondition; + PitchNormalLaw_DWork.Delay1_DSTATE = PitchNormalLaw_rtP.Delay1_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_j = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_m; + PitchNormalLaw_DWork.Delay_DSTATE_ca = PitchNormalLaw_rtP.Delay_InitialCondition_e; + PitchNormalLaw_DWork.Delay1_DSTATE_i = PitchNormalLaw_rtP.Delay1_InitialCondition_g; + PitchNormalLaw_DWork.Delay_DSTATE_e = PitchNormalLaw_rtP.RateLimiterVariableTs5_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_kd = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_j; + PitchNormalLaw_DWork.Delay_DSTATE_b = PitchNormalLaw_rtP.RateLimiterVariableTs3_InitialCondition_e; + PitchNormalLaw_DWork.Delay_DSTATE_ku = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_f; + PitchNormalLaw_DWork.Delay_DSTATE_gl = PitchNormalLaw_rtP.Delay_InitialCondition_c; + PitchNormalLaw_DWork.Delay1_DSTATE_l = PitchNormalLaw_rtP.Delay1_InitialCondition_gf; + PitchNormalLaw_DWork.Delay_DSTATE_m = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_e; + PitchNormalLaw_DWork.Delay_DSTATE_k2 = PitchNormalLaw_rtP.Delay_InitialCondition_h; + PitchNormalLaw_DWork.Delay1_DSTATE_n = PitchNormalLaw_rtP.Delay1_InitialCondition_e; + PitchNormalLaw_DWork.Delay_DSTATE_mz = PitchNormalLaw_rtP.RateLimiterVariableTs4_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_jh = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_a; + PitchNormalLaw_DWork.Delay_DSTATE_dy = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_di; + PitchNormalLaw_DWork.Delay_DSTATE_e5 = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_f; + PitchNormalLaw_DWork.Delay_DSTATE_gz = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_g; + PitchNormalLaw_DWork.Delay_DSTATE_lf = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_c; + PitchNormalLaw_DWork.Delay_DSTATE_ho = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_g; + PitchNormalLaw_DWork.Delay_DSTATE_ds = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_h; + PitchNormalLaw_DWork.Delay_DSTATE_jt = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_a; + PitchNormalLaw_DWork.icLoad = true; + PitchNormalLaw_DWork.Delay_DSTATE_ej = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_b; + PitchNormalLaw_DWork.Delay_DSTATE_e4 = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_p; + PitchNormalLaw_DWork.icLoad_p = true; +} + +void PitchNormalLaw::reset(void) +{ + real_T rtb_nz_limit_up_g; + real_T rtb_nz_limit_lo_g; + real_T rtb_in_flare; + PitchNormalLaw_DWork.Delay_DSTATE = PitchNormalLaw_rtP.RateLimiterDynamicVariableTs_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_h = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_n = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_c = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_l = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_l; + PitchNormalLaw_DWork.Delay_DSTATE_k = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_o; + PitchNormalLaw_DWork.Delay_DSTATE_d = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_d; + PitchNormalLaw_DWork.Delay_DSTATE_f = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_h; + PitchNormalLaw_DWork.Delay_DSTATE_g = PitchNormalLaw_rtP.Delay_InitialCondition; + PitchNormalLaw_DWork.Delay1_DSTATE = PitchNormalLaw_rtP.Delay1_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_j = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_m; + PitchNormalLaw_DWork.Delay_DSTATE_ca = PitchNormalLaw_rtP.Delay_InitialCondition_e; + PitchNormalLaw_DWork.Delay1_DSTATE_i = PitchNormalLaw_rtP.Delay1_InitialCondition_g; + PitchNormalLaw_DWork.Delay_DSTATE_e = PitchNormalLaw_rtP.RateLimiterVariableTs5_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_kd = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_j; + PitchNormalLaw_DWork.Delay_DSTATE_b = PitchNormalLaw_rtP.RateLimiterVariableTs3_InitialCondition_e; + PitchNormalLaw_DWork.Delay_DSTATE_ku = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_f; + PitchNormalLaw_DWork.Delay_DSTATE_gl = PitchNormalLaw_rtP.Delay_InitialCondition_c; + PitchNormalLaw_DWork.Delay1_DSTATE_l = PitchNormalLaw_rtP.Delay1_InitialCondition_gf; + PitchNormalLaw_DWork.Delay_DSTATE_m = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_e; + PitchNormalLaw_DWork.Delay_DSTATE_k2 = PitchNormalLaw_rtP.Delay_InitialCondition_h; + PitchNormalLaw_DWork.Delay1_DSTATE_n = PitchNormalLaw_rtP.Delay1_InitialCondition_e; + PitchNormalLaw_DWork.Delay_DSTATE_mz = PitchNormalLaw_rtP.RateLimiterVariableTs4_InitialCondition; + PitchNormalLaw_DWork.Delay_DSTATE_jh = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_a; + PitchNormalLaw_DWork.Delay_DSTATE_dy = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_di; + PitchNormalLaw_DWork.Delay_DSTATE_e5 = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_f; + PitchNormalLaw_DWork.Delay_DSTATE_gz = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_g; + PitchNormalLaw_DWork.Delay_DSTATE_lf = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_c; + PitchNormalLaw_DWork.Delay_DSTATE_ho = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_InitialCondition_g; + PitchNormalLaw_DWork.Delay_DSTATE_ds = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_h; + PitchNormalLaw_DWork.Delay_DSTATE_jt = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_InitialCondition_a; + PitchNormalLaw_DWork.icLoad = true; + PitchNormalLaw_DWork.Delay_DSTATE_ej = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_b; + PitchNormalLaw_DWork.Delay_DSTATE_e4 = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_InitialCondition_p; + PitchNormalLaw_DWork.icLoad_p = true; + PitchNormalLaw_LagFilter_Reset(&PitchNormalLaw_DWork.sf_LagFilter); + PitchNormalLaw_DWork.is_active_c2_PitchNormalLaw = 0U; + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_NO_ACTIVE_CHILD; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = 0.0; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = 0.0; + PitchNormalLaw_DWork.is_active_c9_PitchNormalLaw = 0U; + PitchNormalLaw_DWork.is_c9_PitchNormalLaw = PitchNormalLaw_IN_NO_ACTIVE_CHILD; + PitchNormalLaw_DWork.is_active_c8_PitchNormalLaw = 0U; + PitchNormalLaw_DWork.is_c8_PitchNormalLaw = PitchNormalLaw_IN_NO_ACTIVE_CHILD; + PitchNormalLaw_DWork.is_active_c7_PitchNormalLaw = 0U; + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_NO_ACTIVE_CHILD; + rtb_nz_limit_up_g = 0.0; + rtb_nz_limit_lo_g = 0.0; + PitchNormalLaw_RateLimiter_Reset(&PitchNormalLaw_DWork.sf_RateLimiter); + PitchNormalLaw_DWork.is_active_c6_PitchNormalLaw = 0U; + PitchNormalLaw_DWork.is_c6_PitchNormalLaw = PitchNormalLaw_IN_NO_ACTIVE_CHILD; + PitchNormalLaw_RateLimiter_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_p); + PitchNormalLaw_RateLimiter_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_c); + PitchNormalLaw_RateLimiter_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_n); + PitchNormalLaw_eta_trim_limit_lofreeze_Reset(&PitchNormalLaw_DWork.sf_eta_trim_limit_lofreeze); + PitchNormalLaw_eta_trim_limit_lofreeze_Reset(&PitchNormalLaw_DWork.sf_eta_trim_limit_upfreeze); + PitchNormalLaw_RateLimiter_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_o); + PitchNormalLaw_LagFilter_i_Reset(&PitchNormalLaw_DWork.sf_LagFilter_k); + PitchNormalLaw_WashoutFilter_Reset(&PitchNormalLaw_DWork.sf_WashoutFilter_k); + PitchNormalLaw_LagFilter_i_Reset(&PitchNormalLaw_DWork.sf_LagFilter_g3); + PitchNormalLaw_WashoutFilter_Reset(&PitchNormalLaw_DWork.sf_WashoutFilter_c); + PitchNormalLaw_RateLimiter_l_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_nx); + PitchNormalLaw_LagFilter_Reset(&PitchNormalLaw_DWork.sf_LagFilter_m); + PitchNormalLaw_WashoutFilter_Reset(&PitchNormalLaw_DWork.sf_WashoutFilter_h); + PitchNormalLaw_RateLimiter_l_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_d); + PitchNormalLaw_RateLimiter_l_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_c2); + PitchNormalLaw_RateLimiter_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_l); + PitchNormalLaw_LagFilter_i_Reset(&PitchNormalLaw_DWork.sf_LagFilter_i); + PitchNormalLaw_WashoutFilter_Reset(&PitchNormalLaw_DWork.sf_WashoutFilter_l); + PitchNormalLaw_LagFilter_i_Reset(&PitchNormalLaw_DWork.sf_LagFilter_g); + PitchNormalLaw_WashoutFilter_Reset(&PitchNormalLaw_DWork.sf_WashoutFilter_d); + PitchNormalLaw_LagFilter_i_Reset(&PitchNormalLaw_DWork.sf_LagFilter_n); + PitchNormalLaw_WashoutFilter_Reset(&PitchNormalLaw_DWork.sf_WashoutFilter); + PitchNormalLaw_RateLimiter_l_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_ct); + PitchNormalLaw_LagFilter_i_Reset(&PitchNormalLaw_DWork.sf_LagFilter_f); + PitchNormalLaw_RateLimiter_Reset(&PitchNormalLaw_DWork.sf_RateLimiter_b); +} + +void PitchNormalLaw::step(const real_T *rtu_In_time_dt, const real_T *rtu_In_nz_g, const real_T *rtu_In_Theta_deg, const + real_T *rtu_In_Phi_deg, const real_T *rtu_In_qk_deg_s, const real_T *rtu_In_qk_dot_deg_s2, const real_T + *rtu_In_eta_deg, const real_T *rtu_In_eta_trim_deg, const real_T *rtu_In_alpha_deg, const real_T *rtu_In_V_ias_kn, + const real_T *rtu_In_V_tas_kn, const real_T *rtu_In_H_radio_ft, const real_T *rtu_In_flaps_handle_index, const real_T * + rtu_In_spoilers_left_pos, const real_T *rtu_In_spoilers_right_pos, const real_T *rtu_In_thrust_lever_1_pos, const + real_T *rtu_In_thrust_lever_2_pos, const boolean_T *rtu_In_tailstrike_protection_on, const real_T *rtu_In_VLS_kn, + const real_T *rtu_In_delta_eta_pos, const boolean_T *rtu_In_on_ground, const real_T *rtu_In_in_flight, const boolean_T + *rtu_In_tracking_mode_on, const boolean_T *rtu_In_high_aoa_prot_active, const boolean_T *rtu_In_high_speed_prot_active, + const real_T *rtu_In_alpha_prot, const real_T *rtu_In_alpha_max, const real_T *rtu_In_high_speed_prot_high_kn, const + real_T *rtu_In_high_speed_prot_low_kn, const real_T *rtu_In_ap_theta_c_deg, const boolean_T *rtu_In_any_ap_engaged, + real_T *rty_Out_eta_deg, real_T *rty_Out_eta_trim_dot_deg_s, real_T *rty_Out_eta_trim_limit_lo, real_T + *rty_Out_eta_trim_limit_up) +{ + real_T rtb_nz_limit_up_g; + real_T rtb_nz_limit_lo_g; + real_T rtb_in_flare; + real_T rtb_Bias_o; + real_T rtb_Cos; + real_T rtb_Divide; + real_T rtb_Divide_a0; + real_T rtb_Divide_an; + real_T rtb_Divide_b4; + real_T rtb_Divide_cq; + real_T rtb_Divide_kq; + real_T rtb_Divide_l; + real_T rtb_Divide_m; + real_T rtb_Divide_o; + real_T rtb_Divide_o2; + real_T rtb_Gain; + real_T rtb_Gain1_e; + real_T rtb_Gain1_ft; + real_T rtb_Gain5_gq; + real_T rtb_Gain_bs; + real_T rtb_Gain_f; + real_T rtb_Gain_g; + real_T rtb_Gain_ll; + real_T rtb_Gain_ot; + real_T rtb_Gain_px; + real_T rtb_Loaddemand2; + real_T rtb_Loaddemand2_l; + real_T rtb_ManualSwitch; + real_T rtb_Product1_ck; + real_T rtb_Product1_dm; + real_T rtb_Product_k; + real_T rtb_Product_kz; + real_T rtb_Product_n3; + real_T rtb_Saturation3; + real_T rtb_Sum10; + real_T rtb_Sum1_d4; + real_T rtb_Sum_j4; + real_T rtb_Y_aj; + real_T rtb_Y_am; + real_T rtb_Y_b; + real_T rtb_Y_cm; + real_T rtb_Y_ll; + real_T rtb_Y_n; + real_T rtb_Y_p; + real_T rtb_alpha_err_gain; + real_T rtb_eta_trim_deg_rate_limit_lo_deg_s; + real_T rtb_eta_trim_deg_rate_limit_up_deg_s; + real_T rtb_qk_dot_gain; + real_T rtb_qk_gain; + real_T rtb_uDLookupTable; + real_T rtb_v_target; + real_T rtb_y; + int32_T rtb_in_rotation; + boolean_T rtb_OR; + boolean_T rtb_eta_trim_deg_should_freeze; + PitchNormalLaw_LagFilter(rtu_In_Theta_deg, PitchNormalLaw_rtP.LagFilter_C1, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_LagFilter); + if (PitchNormalLaw_rtP.ManualSwitch_CurrentSetting == 1) { + rtb_ManualSwitch = PitchNormalLaw_rtP.Constant1_Value_k; + } else { + rtb_ManualSwitch = PitchNormalLaw_rtP.Constant_Value_p; + } + + if (PitchNormalLaw_DWork.is_active_c2_PitchNormalLaw == 0U) { + PitchNormalLaw_DWork.is_active_c2_PitchNormalLaw = 1U; + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Ground; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } else { + switch (PitchNormalLaw_DWork.is_c2_PitchNormalLaw) { + case PitchNormalLaw_IN_Flare_Reduce_Theta_c: + if (*rtu_In_in_flight == 0.0) { + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Ground; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } else if ((*rtu_In_H_radio_ft > 50.0) && (rtb_ManualSwitch == 0.0)) { + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flight_Low; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } else { + rtb_in_flare = 1.0; + PitchNormalLaw_B.flare_Theta_c_deg = -2.0; + } + break; + + case PitchNormalLaw_IN_Flare_Set_Rate: + if (PitchNormalLaw_rtP.ManualSwitch1_CurrentSetting == 1) { + rtb_Y_b = PitchNormalLaw_rtP.Constant1_Value_k; + } else { + rtb_Y_b = PitchNormalLaw_rtP.Constant_Value_p; + } + + if ((*rtu_In_H_radio_ft <= 30.0) || (rtb_Y_b == 1.0)) { + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flare_Reduce_Theta_c; + rtb_in_flare = 1.0; + PitchNormalLaw_B.flare_Theta_c_deg = -2.0; + } else if ((*rtu_In_H_radio_ft > 50.0) && (rtb_ManualSwitch == 0.0)) { + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flight_Low; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } else { + rtb_in_flare = 1.0; + } + break; + + case PitchNormalLaw_IN_Flare_Store_Theta_c_deg: + if ((*rtu_In_H_radio_ft > 50.0) && (rtb_ManualSwitch == 0.0)) { + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flight_Low; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } else { + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -(rtb_Sum_j4 + 2.0) / 8.0; + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flare_Set_Rate; + rtb_in_flare = 1.0; + } + break; + + case PitchNormalLaw_IN_Flight_High: + if ((*rtu_In_H_radio_ft <= 50.0) || (rtb_ManualSwitch == 1.0)) { + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flare_Store_Theta_c_deg; + rtb_in_flare = 1.0; + } else { + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } + break; + + case PitchNormalLaw_IN_Flight_Low: + if (*rtu_In_H_radio_ft > 50.0) { + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flight_High; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } else { + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } + break; + + default: + if (*rtu_In_in_flight == 1.0) { + PitchNormalLaw_DWork.is_c2_PitchNormalLaw = PitchNormalLaw_IN_Flight_Low; + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } else { + rtb_in_flare = 0.0; + PitchNormalLaw_B.flare_Theta_c_deg = rtb_Sum_j4; + PitchNormalLaw_B.flare_Theta_c_rate_deg_s = -3.0; + } + break; + } + } + + if (PitchNormalLaw_DWork.is_active_c9_PitchNormalLaw == 0U) { + PitchNormalLaw_DWork.is_active_c9_PitchNormalLaw = 1U; + PitchNormalLaw_DWork.is_c9_PitchNormalLaw = PitchNormalLaw_IN_running; + rtb_eta_trim_deg_should_freeze = false; + } else if (PitchNormalLaw_DWork.is_c9_PitchNormalLaw == PitchNormalLaw_IN_frozen) { + if ((rtb_in_flare == 0.0) && (*rtu_In_nz_g < 1.25) && (*rtu_In_nz_g > 0.5) && (std::abs(*rtu_In_Phi_deg) <= 30.0)) { + PitchNormalLaw_DWork.is_c9_PitchNormalLaw = PitchNormalLaw_IN_running; + rtb_eta_trim_deg_should_freeze = false; + } else { + rtb_eta_trim_deg_should_freeze = true; + } + } else if ((rtb_in_flare != 0.0) || (*rtu_In_nz_g >= 1.25) || (*rtu_In_nz_g <= 0.5) || (std::abs(*rtu_In_Phi_deg) > + 30.0)) { + PitchNormalLaw_DWork.is_c9_PitchNormalLaw = PitchNormalLaw_IN_frozen; + rtb_eta_trim_deg_should_freeze = true; + } else { + rtb_eta_trim_deg_should_freeze = false; + } + + if (PitchNormalLaw_DWork.is_active_c8_PitchNormalLaw == 0U) { + PitchNormalLaw_DWork.is_active_c8_PitchNormalLaw = 1U; + PitchNormalLaw_DWork.is_c8_PitchNormalLaw = PitchNormalLaw_IN_manual; + } else { + switch (PitchNormalLaw_DWork.is_c8_PitchNormalLaw) { + case PitchNormalLaw_IN_automatic: + if (*rtu_In_in_flight == 0.0) { + PitchNormalLaw_DWork.is_c8_PitchNormalLaw = PitchNormalLaw_IN_reset; + } else if (*rtu_In_tracking_mode_on) { + PitchNormalLaw_DWork.is_c8_PitchNormalLaw = PitchNormalLaw_IN_tracking; + } + break; + + case PitchNormalLaw_IN_manual: + if (*rtu_In_in_flight != 0.0) { + PitchNormalLaw_DWork.is_c8_PitchNormalLaw = PitchNormalLaw_IN_automatic; + } + break; + + case PitchNormalLaw_IN_reset: + if ((*rtu_In_in_flight == 0.0) && (*rtu_In_eta_trim_deg == 0.0)) { + PitchNormalLaw_DWork.is_c8_PitchNormalLaw = PitchNormalLaw_IN_manual; + } + break; + + default: + if (!*rtu_In_tracking_mode_on) { + PitchNormalLaw_DWork.is_c8_PitchNormalLaw = PitchNormalLaw_IN_automatic; + } + break; + } + } + + if (PitchNormalLaw_DWork.is_active_c7_PitchNormalLaw == 0U) { + PitchNormalLaw_DWork.is_active_c7_PitchNormalLaw = 1U; + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_ground; + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.7; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.7; + rtb_nz_limit_up_g = 2.0; + rtb_nz_limit_lo_g = 0.0; + } else { + switch (PitchNormalLaw_DWork.is_c7_PitchNormalLaw) { + case PitchNormalLaw_IN_flight_clean: + if (*rtu_In_flaps_handle_index != 0.0) { + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_flight_flaps; + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.7; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.7; + rtb_nz_limit_up_g = 2.0; + rtb_nz_limit_lo_g = 0.0; + } else if (*rtu_In_in_flight == 0.0) { + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_ground; + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.7; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.7; + rtb_nz_limit_up_g = 2.0; + rtb_nz_limit_lo_g = 0.0; + } else { + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.3; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.3; + rtb_nz_limit_up_g = 2.5; + rtb_nz_limit_lo_g = -1.0; + } + break; + + case PitchNormalLaw_IN_flight_flaps: + if (*rtu_In_flaps_handle_index == 0.0) { + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_flight_clean; + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.3; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.3; + rtb_nz_limit_up_g = 2.5; + rtb_nz_limit_lo_g = -1.0; + } else if (*rtu_In_in_flight == 0.0) { + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_ground; + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.7; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.7; + rtb_nz_limit_up_g = 2.0; + rtb_nz_limit_lo_g = 0.0; + } else { + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.7; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.7; + rtb_nz_limit_up_g = 2.0; + rtb_nz_limit_lo_g = 0.0; + } + break; + + default: + if ((*rtu_In_in_flight != 0.0) && (*rtu_In_flaps_handle_index == 0.0)) { + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_flight_clean; + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.3; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.3; + rtb_nz_limit_up_g = 2.5; + rtb_nz_limit_lo_g = -1.0; + } else if ((*rtu_In_in_flight != 0.0) && (*rtu_In_flaps_handle_index != 0.0)) { + PitchNormalLaw_DWork.is_c7_PitchNormalLaw = PitchNormalLaw_IN_flight_flaps; + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.7; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.7; + rtb_nz_limit_up_g = 2.0; + rtb_nz_limit_lo_g = 0.0; + } else { + rtb_eta_trim_deg_rate_limit_up_deg_s = 0.7; + rtb_eta_trim_deg_rate_limit_lo_deg_s = -0.7; + rtb_nz_limit_up_g = 2.0; + rtb_nz_limit_lo_g = 0.0; + } + break; + } + } + + rtb_uDLookupTable = *rtu_In_in_flight; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation_UpperSat_c) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_UpperSat_c; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation_LowerSat_n) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_LowerSat_n; + } + + PitchNormalLaw_RateLimiter(rtb_uDLookupTable, PitchNormalLaw_rtP.RateLimiterVariableTs_up, + PitchNormalLaw_rtP.RateLimiterVariableTs_lo, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs_InitialCondition, &rtb_Y_b, &PitchNormalLaw_DWork.sf_RateLimiter); + if (PitchNormalLaw_DWork.is_active_c6_PitchNormalLaw == 0U) { + PitchNormalLaw_DWork.is_active_c6_PitchNormalLaw = 1U; + PitchNormalLaw_DWork.is_c6_PitchNormalLaw = PitchNormalLaw_IN_OFF; + rtb_in_rotation = 0; + } else if (PitchNormalLaw_DWork.is_c6_PitchNormalLaw == PitchNormalLaw_IN_OFF) { + if ((rtb_Y_b < 1.0) && (*rtu_In_V_tas_kn > 70.0) && ((*rtu_In_thrust_lever_1_pos >= 35.0) || + (*rtu_In_thrust_lever_2_pos >= 35.0))) { + PitchNormalLaw_DWork.is_c6_PitchNormalLaw = PitchNormalLaw_IN_ON; + rtb_in_rotation = 1; + } else { + rtb_in_rotation = 0; + } + } else if ((rtb_Y_b == 1.0) || (*rtu_In_H_radio_ft > 400.0) || ((*rtu_In_V_tas_kn < 70.0) && + ((*rtu_In_thrust_lever_1_pos < 35.0) || (*rtu_In_thrust_lever_2_pos < 35.0)))) { + PitchNormalLaw_DWork.is_c6_PitchNormalLaw = PitchNormalLaw_IN_OFF; + rtb_in_rotation = 0; + } else { + rtb_in_rotation = 1; + } + + if (rtb_in_rotation > PitchNormalLaw_rtP.Saturation1_UpperSat) { + rtb_Gain = PitchNormalLaw_rtP.Saturation1_UpperSat; + } else if (rtb_in_rotation < PitchNormalLaw_rtP.Saturation1_LowerSat) { + rtb_Gain = PitchNormalLaw_rtP.Saturation1_LowerSat; + } else { + rtb_Gain = rtb_in_rotation; + } + + PitchNormalLaw_RateLimiter(rtb_Gain, PitchNormalLaw_rtP.RateLimiterVariableTs1_up, + PitchNormalLaw_rtP.RateLimiterVariableTs1_lo, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs1_InitialCondition, &rtb_ManualSwitch, + &PitchNormalLaw_DWork.sf_RateLimiter_p); + rtb_Gain = PitchNormalLaw_rtP.Gain_Gain_a * *rtu_In_delta_eta_pos; + PitchNormalLaw_RateLimiter(rtb_nz_limit_up_g, PitchNormalLaw_rtP.RateLimiterVariableTs2_up, + PitchNormalLaw_rtP.RateLimiterVariableTs2_lo, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs2_InitialCondition, &rtb_Y_p, &PitchNormalLaw_DWork.sf_RateLimiter_c); + PitchNormalLaw_RateLimiter(rtb_nz_limit_lo_g, PitchNormalLaw_rtP.RateLimiterVariableTs3_up, + PitchNormalLaw_rtP.RateLimiterVariableTs3_lo, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs3_InitialCondition, &rtb_Y_ll, &PitchNormalLaw_DWork.sf_RateLimiter_n); + rtb_Loaddemand2_l = std::abs(PitchNormalLaw_B.flare_Theta_c_rate_deg_s) * *rtu_In_time_dt; + rtb_Loaddemand2 = *rtu_In_time_dt * PitchNormalLaw_B.flare_Theta_c_rate_deg_s; + PitchNormalLaw_DWork.Delay_DSTATE += std::fmax(std::fmin(PitchNormalLaw_B.flare_Theta_c_deg - + PitchNormalLaw_DWork.Delay_DSTATE, rtb_Loaddemand2_l), rtb_Loaddemand2); + PitchNormalLaw_eta_trim_limit_lofreeze(rtu_In_eta_trim_deg, rtu_In_high_aoa_prot_active, &rtb_Y_am, + &PitchNormalLaw_DWork.sf_eta_trim_limit_lofreeze); + if (*rtu_In_high_aoa_prot_active) { + *rty_Out_eta_trim_limit_lo = rtb_Y_am; + } else { + *rty_Out_eta_trim_limit_lo = PitchNormalLaw_rtP.Constant3_Value; + } + + PitchNormalLaw_eta_trim_limit_lofreeze(rtu_In_eta_trim_deg, rtu_In_high_speed_prot_active, &rtb_Y_am, + &PitchNormalLaw_DWork.sf_eta_trim_limit_upfreeze); + if (*rtu_In_high_speed_prot_active) { + *rty_Out_eta_trim_limit_up = rtb_Y_am; + } else { + *rty_Out_eta_trim_limit_up = PitchNormalLaw_rtP.Constant2_Value; + } + + rtb_Loaddemand2_l = *rtu_In_V_ias_kn; + rtb_Gain_ot = *rtu_In_flaps_handle_index; + rtb_Gain_px = *rtu_In_VLS_kn; + if (rtb_Gain_ot == 5.0) { + rtb_in_rotation = 25; + } else { + rtb_in_rotation = 30; + } + + PitchNormalLaw_RateLimiter(static_cast(rtb_in_rotation) - std::fmin(5.0, std::fmax(0.0, 5.0 - + (rtb_Loaddemand2_l - (rtb_Gain_px + 5.0)) * 0.25)), PitchNormalLaw_rtP.RateLimiterVariableTs6_up, + PitchNormalLaw_rtP.RateLimiterVariableTs6_lo, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs6_InitialCondition, &rtb_Y_am, &PitchNormalLaw_DWork.sf_RateLimiter_o); + rtb_Loaddemand2 = PitchNormalLaw_rtP.Gain1_Gain * *rtu_In_Theta_deg; + rtb_Loaddemand2_l = PitchNormalLaw_rtP.Gain1_Gain_c * *rtu_In_Theta_deg; + rtb_Cos = std::cos(rtb_Loaddemand2_l); + rtb_Loaddemand2_l = PitchNormalLaw_rtP.Gain1_Gain_l * *rtu_In_Phi_deg; + rtb_Y_cm = rtb_Cos / std::cos(rtb_Loaddemand2_l); + rtb_Y_n = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.uDLookupTable_bp01Data, + PitchNormalLaw_rtP.uDLookupTable_tableData, 6U); + rtb_uDLookupTable = *rtu_In_V_tas_kn; + rtb_Loaddemand2_l = PitchNormalLaw_rtP.Gain1_Gain_e * *rtu_In_qk_deg_s; + rtb_Gain_g = *rtu_In_nz_g - rtb_Y_cm; + rtb_Divide_kq = PitchNormalLaw_rtP.Gain2_Gain * rtb_Y_am - rtb_Loaddemand2; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation3_UpperSat) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_UpperSat; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation3_LowerSat) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_LowerSat; + } + + if (rtb_Divide_kq > PitchNormalLaw_rtP.Saturation1_UpperSat_i) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation1_UpperSat_i; + } else if (rtb_Divide_kq < PitchNormalLaw_rtP.Saturation1_LowerSat_h) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation1_LowerSat_h; + } + + rtb_Saturation3 = (PitchNormalLaw_rtP.Gain_Gain_c * PitchNormalLaw_rtP.Vm_currentms_Value * rtb_Loaddemand2_l + + rtb_Gain_g) - (rtb_Y_n / (PitchNormalLaw_rtP.Gain5_Gain * rtb_uDLookupTable) + + PitchNormalLaw_rtP.Bias_Bias) * ((rtb_Y_cm + look1_binlxpw(rtb_Divide_kq, PitchNormalLaw_rtP.Loaddemand1_bp01Data, + PitchNormalLaw_rtP.Loaddemand1_tableData, 2U)) - rtb_Y_cm); + rtb_Y_n = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.PLUT_bp01Data, PitchNormalLaw_rtP.PLUT_tableData, 1U); + rtb_Product1_dm = rtb_Saturation3 * rtb_Y_n; + rtb_Loaddemand2_l = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_Gain * *rtu_In_qk_deg_s; + rtb_Divide = (rtb_Loaddemand2_l - PitchNormalLaw_DWork.Delay_DSTATE_h) / *rtu_In_time_dt; + rtb_Y_n = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.DLUT_bp01Data, PitchNormalLaw_rtP.DLUT_tableData, 1U); + rtb_Gain_px = rtb_Saturation3 * rtb_Y_n * PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_Gain; + rtb_Divide_o = (rtb_Gain_px - PitchNormalLaw_DWork.Delay_DSTATE_n) / *rtu_In_time_dt; + rtb_Gain_ot = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_Gain * *rtu_In_V_tas_kn; + rtb_Divide_an = (rtb_Gain_ot - PitchNormalLaw_DWork.Delay_DSTATE_c) / *rtu_In_time_dt; + PitchNormalLaw_LagFilter_n(rtb_Divide_an, PitchNormalLaw_rtP.LagFilter_C1_i, rtu_In_time_dt, &rtb_Y_am, + &PitchNormalLaw_DWork.sf_LagFilter_k); + if (rtb_Y_am > PitchNormalLaw_rtP.SaturationV_dot_UpperSat) { + rtb_Divide_a0 = PitchNormalLaw_rtP.SaturationV_dot_UpperSat; + } else if (rtb_Y_am < PitchNormalLaw_rtP.SaturationV_dot_LowerSat) { + rtb_Divide_a0 = PitchNormalLaw_rtP.SaturationV_dot_LowerSat; + } else { + rtb_Divide_a0 = rtb_Y_am; + } + + rtb_Gain5_gq = std::fmin(*rtu_In_spoilers_left_pos, *rtu_In_spoilers_right_pos); + PitchNormalLaw_WashoutFilter(rtb_Gain5_gq, PitchNormalLaw_rtP.WashoutFilter_C1, rtu_In_time_dt, &rtb_Y_am, + &PitchNormalLaw_DWork.sf_WashoutFilter_k); + rtb_Y_n = look1_binlxpw(*rtu_In_H_radio_ft, PitchNormalLaw_rtP.ScheduledGain_BreakpointsForDimension1, + PitchNormalLaw_rtP.ScheduledGain_Table, 3U); + if (rtb_Y_am > PitchNormalLaw_rtP.SaturationSpoilers_UpperSat) { + rtb_Y_am = PitchNormalLaw_rtP.SaturationSpoilers_UpperSat; + } else if (rtb_Y_am < PitchNormalLaw_rtP.SaturationSpoilers_LowerSat) { + rtb_Y_am = PitchNormalLaw_rtP.SaturationSpoilers_LowerSat; + } + + rtb_Product_kz = rtb_Y_am * rtb_Y_n; + rtb_Divide_an = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_Gain_i * *rtu_In_qk_deg_s; + rtb_Divide_cq = (rtb_Divide_an - PitchNormalLaw_DWork.Delay_DSTATE_l) / *rtu_In_time_dt; + rtb_Gain1_ft = PitchNormalLaw_rtP.Gain1_Gain_o * *rtu_In_qk_deg_s; + rtb_Saturation3 = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.uDLookupTable_bp01Data_o, + PitchNormalLaw_rtP.uDLookupTable_tableData_e, 6U); + rtb_uDLookupTable = *rtu_In_V_tas_kn; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation3_UpperSat_a) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_UpperSat_a; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation3_LowerSat_l) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_LowerSat_l; + } + + rtb_Saturation3 = (PitchNormalLaw_rtP.Gain_Gain_al * PitchNormalLaw_rtP.Vm_currentms_Value_e * rtb_Gain1_ft + + rtb_Gain_g) - (rtb_Saturation3 / (PitchNormalLaw_rtP.Gain5_Gain_d * rtb_uDLookupTable) + + PitchNormalLaw_rtP.Bias_Bias_a) * (rtb_Y_p - rtb_Y_cm); + rtb_Y_n = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.PLUT_bp01Data_b, PitchNormalLaw_rtP.PLUT_tableData_b, 1U); + rtb_Product1_ck = rtb_Saturation3 * rtb_Y_n; + rtb_Y_n = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.DLUT_bp01Data_h, PitchNormalLaw_rtP.DLUT_tableData_p, 1U); + rtb_Gain1_ft = rtb_Saturation3 * rtb_Y_n * PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_Gain_j; + rtb_Divide_l = (rtb_Gain1_ft - PitchNormalLaw_DWork.Delay_DSTATE_k) / *rtu_In_time_dt; + rtb_Gain_bs = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_Gain_e * *rtu_In_V_tas_kn; + rtb_Divide_m = (rtb_Gain_bs - PitchNormalLaw_DWork.Delay_DSTATE_d) / *rtu_In_time_dt; + PitchNormalLaw_LagFilter_n(rtb_Divide_m, PitchNormalLaw_rtP.LagFilter_C1_p, rtu_In_time_dt, &rtb_Y_am, + &PitchNormalLaw_DWork.sf_LagFilter_g3); + if (rtb_Y_am > PitchNormalLaw_rtP.SaturationV_dot_UpperSat_j) { + rtb_Gain_f = PitchNormalLaw_rtP.SaturationV_dot_UpperSat_j; + } else if (rtb_Y_am < PitchNormalLaw_rtP.SaturationV_dot_LowerSat_e) { + rtb_Gain_f = PitchNormalLaw_rtP.SaturationV_dot_LowerSat_e; + } else { + rtb_Gain_f = rtb_Y_am; + } + + PitchNormalLaw_WashoutFilter(rtb_Gain5_gq, PitchNormalLaw_rtP.WashoutFilter_C1_n, rtu_In_time_dt, &rtb_Y_am, + &PitchNormalLaw_DWork.sf_WashoutFilter_c); + rtb_Y_n = look1_binlxpw(*rtu_In_H_radio_ft, PitchNormalLaw_rtP.ScheduledGain_BreakpointsForDimension1_n, + PitchNormalLaw_rtP.ScheduledGain_Table_b, 3U); + if (rtb_Y_am > PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_g) { + rtb_Y_am = PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_g; + } else if (rtb_Y_am < PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_j) { + rtb_Y_am = PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_j; + } + + rtb_Product_n3 = rtb_Y_am * rtb_Y_n; + PitchNormalLaw_RateLimiter_c(rtu_In_delta_eta_pos, PitchNormalLaw_rtP.RateLimiterVariableTs2_up_m, + PitchNormalLaw_rtP.RateLimiterVariableTs2_lo_k, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs2_InitialCondition_f, &rtb_Sum_j4, &PitchNormalLaw_DWork.sf_RateLimiter_nx); + rtb_Divide_m = (*rtu_In_alpha_max - *rtu_In_alpha_prot) * rtb_Sum_j4; + PitchNormalLaw_LagFilter(rtu_In_alpha_deg, PitchNormalLaw_rtP.LagFilter1_C1, rtu_In_time_dt, &rtb_Y_am, + &PitchNormalLaw_DWork.sf_LagFilter_m); + rtb_Sum10 = rtb_Y_am - *rtu_In_alpha_prot; + rtb_y = std::fmax(std::fmax(0.0, *rtu_In_Theta_deg - 22.5), std::fmax(0.0, (std::abs(*rtu_In_Phi_deg) - 3.0) / 6.0)); + PitchNormalLaw_WashoutFilter(rtb_y, PitchNormalLaw_rtP.WashoutFilter_C1_b, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_WashoutFilter_h); + rtb_Saturation3 = (rtb_Divide_m - rtb_Sum10) - rtb_Sum_j4; + rtb_Divide_m = PitchNormalLaw_rtP.Subsystem1_Gain * rtb_Saturation3; + rtb_Sum10 = (rtb_Divide_m - PitchNormalLaw_DWork.Delay_DSTATE_f) / *rtu_In_time_dt; + rtb_Y_n = *rtu_In_time_dt * PitchNormalLaw_rtP.Subsystem1_C1; + rtb_Y_aj = rtb_Y_n + PitchNormalLaw_rtP.Constant_Value_f; + PitchNormalLaw_DWork.Delay1_DSTATE = 1.0 / rtb_Y_aj * (PitchNormalLaw_rtP.Constant_Value_f - rtb_Y_n) * + PitchNormalLaw_DWork.Delay1_DSTATE + (rtb_Sum10 + PitchNormalLaw_DWork.Delay_DSTATE_g) * (rtb_Y_n / rtb_Y_aj); + rtb_alpha_err_gain = PitchNormalLaw_rtP.alpha_err_gain_Gain * rtb_Saturation3; + rtb_y = PitchNormalLaw_rtP.Subsystem3_Gain * *rtu_In_V_ias_kn; + rtb_Divide_o2 = (rtb_y - PitchNormalLaw_DWork.Delay_DSTATE_j) / *rtu_In_time_dt; + rtb_Y_aj = *rtu_In_time_dt * PitchNormalLaw_rtP.Subsystem3_C1; + rtb_Saturation3 = rtb_Y_aj + PitchNormalLaw_rtP.Constant_Value_b; + PitchNormalLaw_DWork.Delay1_DSTATE_i = 1.0 / rtb_Saturation3 * (PitchNormalLaw_rtP.Constant_Value_b - rtb_Y_aj) * + PitchNormalLaw_DWork.Delay1_DSTATE_i + (rtb_Divide_o2 + PitchNormalLaw_DWork.Delay_DSTATE_ca) * (rtb_Y_aj / + rtb_Saturation3); + rtb_qk_gain = PitchNormalLaw_rtP.qk_gain_Gain * *rtu_In_qk_deg_s; + rtb_qk_dot_gain = PitchNormalLaw_rtP.qk_dot_gain_Gain * *rtu_In_qk_dot_deg_s2; + rtb_Y_n = *rtu_In_high_aoa_prot_active; + rtb_Sum_j4 = PitchNormalLaw_rtP.RateLimiterVariableTs5_up * *rtu_In_time_dt; + rtb_Y_n = std::fmin(rtb_Y_n - PitchNormalLaw_DWork.Delay_DSTATE_e, rtb_Sum_j4); + rtb_Sum_j4 = *rtu_In_time_dt * PitchNormalLaw_rtP.RateLimiterVariableTs5_lo; + PitchNormalLaw_DWork.Delay_DSTATE_e += std::fmax(rtb_Y_n, rtb_Sum_j4); + if (PitchNormalLaw_DWork.Delay_DSTATE_e > PitchNormalLaw_rtP.Saturation_UpperSat_eo) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_UpperSat_eo; + } else if (PitchNormalLaw_DWork.Delay_DSTATE_e < PitchNormalLaw_rtP.Saturation_LowerSat_h) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_LowerSat_h; + } else { + rtb_Sum_j4 = PitchNormalLaw_DWork.Delay_DSTATE_e; + } + + rtb_uDLookupTable = (((PitchNormalLaw_rtP.precontrol_gain_Gain * PitchNormalLaw_DWork.Delay1_DSTATE + + rtb_alpha_err_gain) + PitchNormalLaw_rtP.v_dot_gain_Gain * PitchNormalLaw_DWork.Delay1_DSTATE_i) + rtb_qk_gain) + + rtb_qk_dot_gain; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation3_UpperSat_f) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_UpperSat_f; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation3_LowerSat_c) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_LowerSat_c; + } + + rtb_Product_k = rtb_uDLookupTable * rtb_Sum_j4; + rtb_Sum1_d4 = PitchNormalLaw_rtP.Constant_Value_fe - rtb_Sum_j4; + rtb_alpha_err_gain = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_Gain_m * *rtu_In_qk_deg_s; + rtb_Divide_kq = (rtb_alpha_err_gain - PitchNormalLaw_DWork.Delay_DSTATE_kd) / *rtu_In_time_dt; + rtb_Gain1_e = PitchNormalLaw_rtP.Gain1_Gain_en * *rtu_In_qk_deg_s; + rtb_Sum_j4 = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.uDLookupTable_bp01Data_b, + PitchNormalLaw_rtP.uDLookupTable_tableData_h, 6U); + rtb_uDLookupTable = *rtu_In_V_tas_kn; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation3_UpperSat_b) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_UpperSat_b; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation3_LowerSat_e) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_LowerSat_e; + } + + rtb_Bias_o = rtb_Sum_j4 / (PitchNormalLaw_rtP.Gain5_Gain_e * rtb_uDLookupTable) + PitchNormalLaw_rtP.Bias_Bias_f; + PitchNormalLaw_RateLimiter_c(rtu_In_ap_theta_c_deg, PitchNormalLaw_rtP.RateLimiterVariableTs1_up_d, + PitchNormalLaw_rtP.RateLimiterVariableTs1_lo_g, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs1_InitialCondition_l, &rtb_Y_p, &PitchNormalLaw_DWork.sf_RateLimiter_d); + rtb_uDLookupTable = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.ScheduledGain_BreakpointsForDimension1_h, + PitchNormalLaw_rtP.ScheduledGain_Table_j, 6U); + PitchNormalLaw_RateLimiter_c(rtu_In_delta_eta_pos, PitchNormalLaw_rtP.RateLimiterVariableTs_up_n, + PitchNormalLaw_rtP.RateLimiterVariableTs_lo_c, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs_InitialCondition_o, &rtb_Y_am, &PitchNormalLaw_DWork.sf_RateLimiter_c2); + rtb_Y_aj = *rtu_In_delta_eta_pos - PitchNormalLaw_DWork.Delay_DSTATE_b; + rtb_Saturation3 = PitchNormalLaw_rtP.RateLimiterVariableTs3_up_i * *rtu_In_time_dt; + rtb_Y_aj = std::fmin(rtb_Y_aj, rtb_Saturation3); + rtb_Saturation3 = *rtu_In_time_dt * PitchNormalLaw_rtP.RateLimiterVariableTs3_lo_b; + PitchNormalLaw_DWork.Delay_DSTATE_b += std::fmax(rtb_Y_aj, rtb_Saturation3); + rtb_v_target = std::fmax((*rtu_In_high_speed_prot_low_kn - *rtu_In_high_speed_prot_high_kn) * + PitchNormalLaw_DWork.Delay_DSTATE_b, 0.0) + *rtu_In_high_speed_prot_low_kn; + rtb_qk_gain = PitchNormalLaw_rtP.Subsystem2_Gain * rtb_v_target; + rtb_qk_dot_gain = (rtb_qk_gain - PitchNormalLaw_DWork.Delay_DSTATE_ku) / *rtu_In_time_dt; + rtb_Sum_j4 = *rtu_In_time_dt * PitchNormalLaw_rtP.Subsystem2_C1; + rtb_Y_aj = rtb_Sum_j4 + PitchNormalLaw_rtP.Constant_Value_ja; + PitchNormalLaw_DWork.Delay1_DSTATE_l = 1.0 / rtb_Y_aj * (PitchNormalLaw_rtP.Constant_Value_ja - rtb_Sum_j4) * + PitchNormalLaw_DWork.Delay1_DSTATE_l + (rtb_qk_dot_gain + PitchNormalLaw_DWork.Delay_DSTATE_gl) * (rtb_Sum_j4 / + rtb_Y_aj); + rtb_Gain_ll = PitchNormalLaw_rtP.Subsystem_Gain * *rtu_In_V_ias_kn; + rtb_Divide_b4 = (rtb_Gain_ll - PitchNormalLaw_DWork.Delay_DSTATE_m) / *rtu_In_time_dt; + rtb_Sum_j4 = *rtu_In_time_dt * PitchNormalLaw_rtP.Subsystem_C1; + rtb_Y_aj = rtb_Sum_j4 + PitchNormalLaw_rtP.Constant_Value_jj; + PitchNormalLaw_DWork.Delay1_DSTATE_n = 1.0 / rtb_Y_aj * (PitchNormalLaw_rtP.Constant_Value_jj - rtb_Sum_j4) * + PitchNormalLaw_DWork.Delay1_DSTATE_n + (rtb_Divide_b4 + PitchNormalLaw_DWork.Delay_DSTATE_k2) * (rtb_Sum_j4 / + rtb_Y_aj); + rtb_in_rotation = *rtu_In_high_speed_prot_active; + rtb_Saturation3 = PitchNormalLaw_rtP.RateLimiterVariableTs4_up * *rtu_In_time_dt; + rtb_Y_aj = std::fmin(static_cast(rtb_in_rotation) - PitchNormalLaw_DWork.Delay_DSTATE_mz, rtb_Saturation3); + rtb_Saturation3 = *rtu_In_time_dt * PitchNormalLaw_rtP.RateLimiterVariableTs4_lo; + PitchNormalLaw_DWork.Delay_DSTATE_mz += std::fmax(rtb_Y_aj, rtb_Saturation3); + PitchNormalLaw_RateLimiter(rtb_in_flare, PitchNormalLaw_rtP.RateLimiterVariableTs6_up_n, + PitchNormalLaw_rtP.RateLimiterVariableTs6_lo_p, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs6_InitialCondition_f, &rtb_Y_n, &PitchNormalLaw_DWork.sf_RateLimiter_l); + if (*rtu_In_any_ap_engaged) { + rtb_Sum_j4 = rtb_Y_p - *rtu_In_Theta_deg; + rtb_Sum_j4 *= rtb_uDLookupTable; + } else { + rtb_Sum_j4 = look1_binlxpw(rtb_Y_am, PitchNormalLaw_rtP.Loaddemand_bp01Data, PitchNormalLaw_rtP.Loaddemand_tableData, + 2U); + if (rtb_Y_n > PitchNormalLaw_rtP.Saturation_UpperSat) { + rtb_Y_n = PitchNormalLaw_rtP.Saturation_UpperSat; + } else if (rtb_Y_n < PitchNormalLaw_rtP.Saturation_LowerSat) { + rtb_Y_n = PitchNormalLaw_rtP.Saturation_LowerSat; + } + + rtb_Y_p = PitchNormalLaw_DWork.Delay_DSTATE - *rtu_In_Theta_deg; + rtb_uDLookupTable = PitchNormalLaw_rtP.Gain_Gain * rtb_Y_p; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation_UpperSat_g) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_UpperSat_g; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation_LowerSat_d) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_LowerSat_d; + } + + rtb_Y_p = (PitchNormalLaw_rtP.Constant_Value_g - rtb_Y_n) * PitchNormalLaw_rtP.Constant_Value_l + rtb_uDLookupTable * + rtb_Y_n; + if (PitchNormalLaw_DWork.Delay_DSTATE_mz > PitchNormalLaw_rtP.Saturation_UpperSat_e) { + rtb_Y_am = PitchNormalLaw_rtP.Saturation_UpperSat_e; + } else if (PitchNormalLaw_DWork.Delay_DSTATE_mz < PitchNormalLaw_rtP.Saturation_LowerSat_m) { + rtb_Y_am = PitchNormalLaw_rtP.Saturation_LowerSat_m; + } else { + rtb_Y_am = PitchNormalLaw_DWork.Delay_DSTATE_mz; + } + + if (rtb_in_rotation > PitchNormalLaw_rtP.Switch2_Threshold) { + rtb_Y_n = PitchNormalLaw_rtP.qk_dot_gain1_Gain * *rtu_In_qk_dot_deg_s2; + rtb_uDLookupTable = PitchNormalLaw_rtP.qk_gain_HSP_Gain * *rtu_In_qk_deg_s; + rtb_Y_aj = rtb_v_target - *rtu_In_V_ias_kn; + rtb_uDLookupTable = ((((PitchNormalLaw_rtP.precontrol_gain_HSP_Gain * PitchNormalLaw_DWork.Delay1_DSTATE_l + + PitchNormalLaw_rtP.Gain6_Gain * rtb_Y_aj) + PitchNormalLaw_rtP.v_dot_gain_HSP_Gain * + PitchNormalLaw_DWork.Delay1_DSTATE_n) + rtb_uDLookupTable) + rtb_Y_n) * PitchNormalLaw_rtP.HSP_gain_Gain; + if (rtb_Sum_j4 > PitchNormalLaw_rtP.Saturation8_UpperSat) { + rtb_Y_n = PitchNormalLaw_rtP.Saturation8_UpperSat; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.Saturation8_LowerSat) { + rtb_Y_n = PitchNormalLaw_rtP.Saturation8_LowerSat; + } else { + rtb_Y_n = rtb_Sum_j4; + } + + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation4_UpperSat) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation4_UpperSat; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation4_LowerSat) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation4_LowerSat; + } + + rtb_Y_n += rtb_uDLookupTable; + } else { + rtb_Y_n = PitchNormalLaw_rtP.Constant1_Value; + } + + rtb_Sum_j4 = ((PitchNormalLaw_rtP.Constant_Value_m - rtb_Y_am) * rtb_Sum_j4 + rtb_Y_n * rtb_Y_am) + rtb_Y_p; + } + + rtb_uDLookupTable = *rtu_In_Phi_deg; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation_UpperSat_f1) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_UpperSat_f1; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation_LowerSat_o1) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_LowerSat_o1; + } + + rtb_Sum_j4 = (PitchNormalLaw_rtP.Gain_Gain_b * PitchNormalLaw_rtP.Vm_currentms_Value_h * rtb_Gain1_e + rtb_Gain_g) - + ((rtb_Cos / std::cos(PitchNormalLaw_rtP.Gain1_Gain_lm * rtb_uDLookupTable) + rtb_Sum_j4) - rtb_Y_cm) * rtb_Bias_o; + rtb_Y_aj = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.PLUT_bp01Data_f, PitchNormalLaw_rtP.PLUT_tableData_k, 1U); + rtb_Y_n = rtb_Sum_j4 * rtb_Y_aj; + rtb_Y_aj = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.DLUT_bp01Data_m, PitchNormalLaw_rtP.DLUT_tableData_a, 1U); + rtb_Cos = rtb_Sum_j4 * rtb_Y_aj * PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_Gain_b; + rtb_uDLookupTable = (rtb_Cos - PitchNormalLaw_DWork.Delay_DSTATE_jh) / *rtu_In_time_dt; + rtb_Y_p = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_Gain_c * *rtu_In_V_tas_kn; + rtb_Sum_j4 = PitchNormalLaw_DWork.Delay_DSTATE_dy; + rtb_Y_am = (rtb_Y_p - PitchNormalLaw_DWork.Delay_DSTATE_dy) / *rtu_In_time_dt; + PitchNormalLaw_LagFilter_n(rtb_Y_am, PitchNormalLaw_rtP.LagFilter_C1_pt, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_LagFilter_i); + if (rtb_Sum_j4 > PitchNormalLaw_rtP.SaturationV_dot_UpperSat_b) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationV_dot_UpperSat_b; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.SaturationV_dot_LowerSat_m) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationV_dot_LowerSat_m; + } + + rtb_Y_am = PitchNormalLaw_rtP.Gain_Gain_f * rtb_Sum_j4; + PitchNormalLaw_WashoutFilter(rtb_Gain5_gq, PitchNormalLaw_rtP.WashoutFilter_C1_l, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_WashoutFilter_l); + rtb_Y_aj = look1_binlxpw(*rtu_In_H_radio_ft, PitchNormalLaw_rtP.ScheduledGain_BreakpointsForDimension1_c, + PitchNormalLaw_rtP.ScheduledGain_Table_g, 3U); + if (rtb_Sum_j4 > PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_o) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_o; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_jl) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_jl; + } + + rtb_uDLookupTable = (((PitchNormalLaw_rtP.Gain3_Gain_c * rtb_Divide_kq + rtb_Y_n) + rtb_uDLookupTable) + rtb_Y_am) + + rtb_Sum_j4 * rtb_Y_aj; + rtb_Y_n = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_Gain_c * *rtu_In_qk_deg_s; + rtb_Saturation3 = (rtb_Y_n - PitchNormalLaw_DWork.Delay_DSTATE_e5) / *rtu_In_time_dt; + rtb_Y_am = PitchNormalLaw_rtP.Gain1_Gain_b * *rtu_In_qk_deg_s; + rtb_Sum_j4 = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.uDLookupTable_bp01Data_a, + PitchNormalLaw_rtP.uDLookupTable_tableData_p, 6U); + rtb_Divide_kq = *rtu_In_V_tas_kn; + if (rtb_Divide_kq > PitchNormalLaw_rtP.Saturation3_UpperSat_n) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation3_UpperSat_n; + } else if (rtb_Divide_kq < PitchNormalLaw_rtP.Saturation3_LowerSat_a) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation3_LowerSat_a; + } + + rtb_Sum_j4 = (PitchNormalLaw_rtP.Gain_Gain_p * PitchNormalLaw_rtP.Vm_currentms_Value_p * rtb_Y_am + rtb_Gain_g) - + (rtb_Sum_j4 / (PitchNormalLaw_rtP.Gain5_Gain_n * rtb_Divide_kq) + PitchNormalLaw_rtP.Bias_Bias_ai) * (rtb_Y_ll - + rtb_Y_cm); + rtb_Y_aj = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.PLUT_bp01Data_a, PitchNormalLaw_rtP.PLUT_tableData_o, 1U); + rtb_Gain1_e = rtb_Sum_j4 * rtb_Y_aj; + rtb_Y_aj = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.DLUT_bp01Data_k, PitchNormalLaw_rtP.DLUT_tableData_e, 1U); + rtb_Y_ll = rtb_Sum_j4 * rtb_Y_aj * PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_Gain_p; + rtb_Bias_o = (rtb_Y_ll - PitchNormalLaw_DWork.Delay_DSTATE_gz) / *rtu_In_time_dt; + rtb_Y_am = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_Gain_a * *rtu_In_V_tas_kn; + rtb_Sum_j4 = PitchNormalLaw_DWork.Delay_DSTATE_lf; + rtb_Y_aj = (rtb_Y_am - PitchNormalLaw_DWork.Delay_DSTATE_lf) / *rtu_In_time_dt; + PitchNormalLaw_LagFilter_n(rtb_Y_aj, PitchNormalLaw_rtP.LagFilter_C1_l, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_LagFilter_g); + if (rtb_Sum_j4 > PitchNormalLaw_rtP.SaturationV_dot_UpperSat_m) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationV_dot_UpperSat_m; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.SaturationV_dot_LowerSat_ek) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationV_dot_LowerSat_ek; + } + + rtb_v_target = PitchNormalLaw_rtP.Gain_Gain_k * rtb_Sum_j4; + PitchNormalLaw_WashoutFilter(rtb_Gain5_gq, PitchNormalLaw_rtP.WashoutFilter_C1_h, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_WashoutFilter_d); + rtb_Y_aj = look1_binlxpw(*rtu_In_H_radio_ft, PitchNormalLaw_rtP.ScheduledGain_BreakpointsForDimension1_f, + PitchNormalLaw_rtP.ScheduledGain_Table_h, 3U); + rtb_Divide_kq = (((PitchNormalLaw_rtP.Gain3_Gain_m * rtb_Divide_cq + rtb_Product1_ck) + rtb_Divide_l) + + PitchNormalLaw_rtP.Gain_Gain_j * rtb_Gain_f) + rtb_Product_n3; + if (rtb_Sum_j4 > PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_h) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_h; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_l) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_l; + } + + rtb_Sum_j4 = (((PitchNormalLaw_rtP.Gain3_Gain_b * rtb_Saturation3 + rtb_Gain1_e) + rtb_Bias_o) + rtb_v_target) + + rtb_Sum_j4 * rtb_Y_aj; + if (rtb_Divide_kq > PitchNormalLaw_rtP.Saturation_UpperSat_hc) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation_UpperSat_hc; + } else if (rtb_Divide_kq < PitchNormalLaw_rtP.Saturation_LowerSat_a) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation_LowerSat_a; + } + + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation_UpperSat_k) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_UpperSat_k; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation_LowerSat_p1) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_LowerSat_p1; + } + + if (rtb_Sum_j4 > PitchNormalLaw_rtP.Saturation_UpperSat_j) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_UpperSat_j; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.Saturation_LowerSat_dw) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_LowerSat_dw; + } + + PitchNormalLaw_VoterAttitudeProtection(rtb_Divide_kq, rtb_Product_k + rtb_Sum1_d4 * rtb_uDLookupTable, rtb_Sum_j4, + &rtb_Saturation3); + rtb_Divide_cq = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs1_Gain_k * *rtu_In_qk_deg_s; + rtb_Product1_ck = (rtb_Divide_cq - PitchNormalLaw_DWork.Delay_DSTATE_ho) / *rtu_In_time_dt; + rtb_Divide_l = PitchNormalLaw_rtP.Gain1_Gain_lk * *rtu_In_qk_deg_s; + rtb_Sum_j4 = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.uDLookupTable_bp01Data_m, + PitchNormalLaw_rtP.uDLookupTable_tableData_a, 6U); + rtb_uDLookupTable = *rtu_In_V_tas_kn; + rtb_Divide_kq = PitchNormalLaw_rtP.Gain3_Gain_g * PitchNormalLaw_rtP.Theta_max3_Value - rtb_Loaddemand2; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation3_UpperSat_e) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_UpperSat_e; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation3_LowerSat_k) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation3_LowerSat_k; + } + + if (rtb_Divide_kq > PitchNormalLaw_rtP.Saturation2_UpperSat) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation2_UpperSat; + } else if (rtb_Divide_kq < PitchNormalLaw_rtP.Saturation2_LowerSat) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation2_LowerSat; + } + + rtb_Sum_j4 = (PitchNormalLaw_rtP.Gain_Gain_jq * PitchNormalLaw_rtP.Vm_currentms_Value_b * rtb_Divide_l + rtb_Gain_g) - + (rtb_Sum_j4 / (PitchNormalLaw_rtP.Gain5_Gain_m * rtb_uDLookupTable) + PitchNormalLaw_rtP.Bias_Bias_m) * ((rtb_Y_cm + + look1_binlxpw(rtb_Divide_kq, PitchNormalLaw_rtP.Loaddemand2_bp01Data, PitchNormalLaw_rtP.Loaddemand2_tableData, 2U)) + - rtb_Y_cm); + rtb_Y_aj = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.PLUT_bp01Data_e, PitchNormalLaw_rtP.PLUT_tableData_g, 1U); + rtb_Y_cm = rtb_Sum_j4 * rtb_Y_aj; + rtb_Y_aj = look1_binlxpw(*rtu_In_V_tas_kn, PitchNormalLaw_rtP.DLUT_bp01Data_hw, PitchNormalLaw_rtP.DLUT_tableData_l, + 1U); + rtb_Loaddemand2 = rtb_Sum_j4 * rtb_Y_aj * PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_Gain_c; + rtb_Divide_l = (rtb_Loaddemand2 - PitchNormalLaw_DWork.Delay_DSTATE_ds) / *rtu_In_time_dt; + rtb_Gain_g = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs2_Gain_p * *rtu_In_V_tas_kn; + rtb_Sum_j4 = PitchNormalLaw_DWork.Delay_DSTATE_jt; + rtb_uDLookupTable = (rtb_Gain_g - PitchNormalLaw_DWork.Delay_DSTATE_jt) / *rtu_In_time_dt; + PitchNormalLaw_LagFilter_n(rtb_uDLookupTable, PitchNormalLaw_rtP.LagFilter_C1_f, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_LagFilter_n); + if (rtb_Sum_j4 > PitchNormalLaw_rtP.SaturationV_dot_UpperSat_j2) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationV_dot_UpperSat_j2; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.SaturationV_dot_LowerSat_n) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationV_dot_LowerSat_n; + } + + rtb_Gain_f = PitchNormalLaw_rtP.Gain_Gain_l0 * rtb_Sum_j4; + PitchNormalLaw_WashoutFilter(rtb_Gain5_gq, PitchNormalLaw_rtP.WashoutFilter_C1_j, rtu_In_time_dt, &rtb_Sum_j4, + &PitchNormalLaw_DWork.sf_WashoutFilter); + rtb_Y_aj = look1_binlxpw(*rtu_In_H_radio_ft, PitchNormalLaw_rtP.ScheduledGain_BreakpointsForDimension1_b, + PitchNormalLaw_rtP.ScheduledGain_Table_e, 3U); + rtb_uDLookupTable = (((PitchNormalLaw_rtP.Gain3_Gain * rtb_Divide + rtb_Product1_dm) + rtb_Divide_o) + + PitchNormalLaw_rtP.Gain_Gain_l * rtb_Divide_a0) + rtb_Product_kz; + if (rtb_Sum_j4 > PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_m) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationSpoilers_UpperSat_m; + } else if (rtb_Sum_j4 < PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_d) { + rtb_Sum_j4 = PitchNormalLaw_rtP.SaturationSpoilers_LowerSat_d; + } + + rtb_Divide_kq = (((PitchNormalLaw_rtP.Gain3_Gain_n * rtb_Product1_ck + rtb_Y_cm) + rtb_Divide_l) + rtb_Gain_f) + + rtb_Sum_j4 * rtb_Y_aj; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation_UpperSat_h) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_UpperSat_h; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation_LowerSat_o) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_LowerSat_o; + } + + if (rtb_Divide_kq > PitchNormalLaw_rtP.Saturation_UpperSat_a) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation_UpperSat_a; + } else if (rtb_Divide_kq < PitchNormalLaw_rtP.Saturation_LowerSat_k) { + rtb_Divide_kq = PitchNormalLaw_rtP.Saturation_LowerSat_k; + } + + PitchNormalLaw_VoterAttitudeProtection(rtb_uDLookupTable, rtb_Saturation3, rtb_Divide_kq, &rtb_Y_aj); + rtb_Sum_j4 = look1_binlxpw(*rtu_In_V_ias_kn, PitchNormalLaw_rtP.ScheduledGain1_BreakpointsForDimension1, + PitchNormalLaw_rtP.ScheduledGain1_Table, 4U); + rtb_Y_cm = rtb_Y_aj * rtb_Sum_j4; + rtb_Sum_j4 = look1_binlxpw(*rtu_In_time_dt, PitchNormalLaw_rtP.ScheduledGain_BreakpointsForDimension1_d, + PitchNormalLaw_rtP.ScheduledGain_Table_hh, 4U); + rtb_Sum_j4 = rtb_Y_cm * rtb_Sum_j4 * PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_Gain * *rtu_In_time_dt; + if (*rtu_In_in_flight > PitchNormalLaw_rtP.Switch_Threshold) { + rtb_Y_aj = *rtu_In_eta_deg; + } else { + rtb_Y_aj = rtb_Gain; + } + + rtb_OR = ((rtb_Y_b == 0.0) || (*rtu_In_tracking_mode_on)); + PitchNormalLaw_DWork.icLoad = (rtb_OR || PitchNormalLaw_DWork.icLoad); + if (PitchNormalLaw_DWork.icLoad) { + PitchNormalLaw_DWork.Delay_DSTATE_o = rtb_Y_aj - rtb_Sum_j4; + } + + PitchNormalLaw_DWork.Delay_DSTATE_o += rtb_Sum_j4; + if (PitchNormalLaw_DWork.Delay_DSTATE_o > PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_UpperLimit) { + PitchNormalLaw_DWork.Delay_DSTATE_o = PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_UpperLimit; + } else if (PitchNormalLaw_DWork.Delay_DSTATE_o < PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_LowerLimit) { + PitchNormalLaw_DWork.Delay_DSTATE_o = PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_LowerLimit; + } + + if (rtb_eta_trim_deg_should_freeze == PitchNormalLaw_rtP.CompareToConstant_const) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Constant_Value; + } else { + rtb_Sum_j4 = PitchNormalLaw_DWork.Delay_DSTATE_o; + } + + rtb_Sum_j4 *= PitchNormalLaw_rtP.Gain_Gain_cy; + if (rtb_Sum_j4 > rtb_eta_trim_deg_rate_limit_up_deg_s) { + *rty_Out_eta_trim_dot_deg_s = rtb_eta_trim_deg_rate_limit_up_deg_s; + } else if (rtb_Sum_j4 < rtb_eta_trim_deg_rate_limit_lo_deg_s) { + *rty_Out_eta_trim_dot_deg_s = rtb_eta_trim_deg_rate_limit_lo_deg_s; + } else { + *rty_Out_eta_trim_dot_deg_s = rtb_Sum_j4; + } + + if (rtb_Y_b > PitchNormalLaw_rtP.Saturation_UpperSat_l) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_UpperSat_l; + } else if (rtb_Y_b < PitchNormalLaw_rtP.Saturation_LowerSat_kp) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_LowerSat_kp; + } else { + rtb_Sum_j4 = rtb_Y_b; + } + + rtb_Product1_dm = PitchNormalLaw_DWork.Delay_DSTATE_o * rtb_Sum_j4; + rtb_Divide = PitchNormalLaw_rtP.Constant_Value_o1 - rtb_Sum_j4; + PitchNormalLaw_RateLimiter_c(rtu_In_delta_eta_pos, PitchNormalLaw_rtP.RateLimiterVariableTs_up_na, + PitchNormalLaw_rtP.RateLimiterVariableTs_lo_i, rtu_In_time_dt, + PitchNormalLaw_rtP.RateLimiterVariableTs_InitialCondition_m, &rtb_Y_cm, &PitchNormalLaw_DWork.sf_RateLimiter_ct); + rtb_Sum_j4 = look2_binlxpw(*rtu_In_Theta_deg, *rtu_In_H_radio_ft, PitchNormalLaw_rtP.uDLookupTable_bp01Data_l, + PitchNormalLaw_rtP.uDLookupTable_bp02Data, PitchNormalLaw_rtP.uDLookupTable_tableData_e5, + PitchNormalLaw_rtP.uDLookupTable_maxIndex, 5U); + rtb_Y_aj = *rtu_In_tailstrike_protection_on; + if (rtb_Y_cm > PitchNormalLaw_rtP.Saturation3_UpperSat_l) { + rtb_eta_trim_deg_rate_limit_up_deg_s = PitchNormalLaw_rtP.Saturation3_UpperSat_l; + } else if (rtb_Y_cm < PitchNormalLaw_rtP.Saturation3_LowerSat_h) { + rtb_eta_trim_deg_rate_limit_up_deg_s = PitchNormalLaw_rtP.Saturation3_LowerSat_h; + } else { + rtb_eta_trim_deg_rate_limit_up_deg_s = rtb_Y_cm; + } + + rtb_Sum_j4 = look1_binlxpw(rtb_Y_aj * rtb_Sum_j4 * rtb_eta_trim_deg_rate_limit_up_deg_s + rtb_Y_cm, + PitchNormalLaw_rtP.PitchRateDemand_bp01Data, PitchNormalLaw_rtP.PitchRateDemand_tableData, 2U); + rtb_eta_trim_deg_rate_limit_up_deg_s = PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_Gain_j3 * rtb_Sum_j4; + rtb_Y_cm = (rtb_eta_trim_deg_rate_limit_up_deg_s - PitchNormalLaw_DWork.Delay_DSTATE_ej) / *rtu_In_time_dt; + rtb_Y_aj = *rtu_In_qk_deg_s - rtb_Sum_j4; + rtb_Divide_o = PitchNormalLaw_rtP.Gain_Gain_pt * rtb_Y_aj; + rtb_eta_trim_deg_rate_limit_lo_deg_s = PitchNormalLaw_rtP.Gain1_Gain_d * rtb_Y_aj * + PitchNormalLaw_rtP.DiscreteDerivativeVariableTs_Gain_g; + rtb_Y_aj = PitchNormalLaw_DWork.Delay_DSTATE_e4; + rtb_Divide_a0 = (rtb_eta_trim_deg_rate_limit_lo_deg_s - PitchNormalLaw_DWork.Delay_DSTATE_e4) / *rtu_In_time_dt; + rtb_Gain5_gq = PitchNormalLaw_rtP.Gain5_Gain_h * *rtu_In_qk_dot_deg_s2; + rtb_Gain5_gq += *rtu_In_qk_deg_s; + PitchNormalLaw_LagFilter_n(rtb_Gain5_gq, PitchNormalLaw_rtP.LagFilter_C1_k, rtu_In_time_dt, &rtb_Y_aj, + &PitchNormalLaw_DWork.sf_LagFilter_f); + rtb_Gain5_gq = PitchNormalLaw_rtP.Gain6_Gain_g * *rtu_In_qk_dot_deg_s2; + rtb_Y_cm = (((rtb_Divide_o + rtb_Divide_a0) * PitchNormalLaw_rtP.Gain1_Gain_a + PitchNormalLaw_rtP.Gain3_Gain_e * + rtb_Y_cm) + (rtb_Y_aj - rtb_Sum_j4) * PitchNormalLaw_rtP.Gain4_Gain) + rtb_Gain5_gq; + rtb_Sum_j4 = look1_binlxpw(*rtu_In_V_ias_kn, PitchNormalLaw_rtP.ScheduledGain1_BreakpointsForDimension1_h, + PitchNormalLaw_rtP.ScheduledGain1_Table_c, 4U); + rtb_Sum_j4 = (PitchNormalLaw_rtP.Constant2_Value_k - rtb_Y_b) * (rtb_Y_cm * rtb_Sum_j4) * + PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_Gain_j * *rtu_In_time_dt; + rtb_eta_trim_deg_should_freeze = (*rtu_In_delta_eta_pos <= PitchNormalLaw_rtP.Constant_Value_o); + rtb_eta_trim_deg_should_freeze = (rtb_eta_trim_deg_should_freeze && (*rtu_In_on_ground)); + rtb_eta_trim_deg_should_freeze = (rtb_eta_trim_deg_should_freeze || (rtb_ManualSwitch == 0.0) || + (*rtu_In_tracking_mode_on)); + PitchNormalLaw_DWork.icLoad_p = (rtb_eta_trim_deg_should_freeze || PitchNormalLaw_DWork.icLoad_p); + if (PitchNormalLaw_DWork.icLoad_p) { + PitchNormalLaw_DWork.Delay_DSTATE_cl = PitchNormalLaw_rtP.Constant_Value_jk - rtb_Sum_j4; + } + + PitchNormalLaw_DWork.Delay_DSTATE_cl += rtb_Sum_j4; + if (PitchNormalLaw_DWork.Delay_DSTATE_cl > PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_UpperLimit_p) { + PitchNormalLaw_DWork.Delay_DSTATE_cl = PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_UpperLimit_p; + } else if (PitchNormalLaw_DWork.Delay_DSTATE_cl < PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_LowerLimit_h) { + PitchNormalLaw_DWork.Delay_DSTATE_cl = PitchNormalLaw_rtP.DiscreteTimeIntegratorVariableTs_LowerLimit_h; + } + + if (*rtu_In_on_ground) { + if (rtb_Gain > PitchNormalLaw_rtP.Saturation_UpperSat_f) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_UpperSat_f; + } else if (rtb_Gain < PitchNormalLaw_rtP.Saturation_LowerSat_p) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_LowerSat_p; + } else { + rtb_Sum_j4 = rtb_Gain; + } + } else { + rtb_Sum_j4 = PitchNormalLaw_rtP.Constant1_Value_h; + } + + rtb_Y_b = PitchNormalLaw_DWork.Delay_DSTATE_cl + rtb_Sum_j4; + if (rtb_ManualSwitch > PitchNormalLaw_rtP.Saturation_UpperSat_m) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_UpperSat_m; + } else if (rtb_ManualSwitch < PitchNormalLaw_rtP.Saturation_LowerSat_b) { + rtb_Sum_j4 = PitchNormalLaw_rtP.Saturation_LowerSat_b; + } else { + rtb_Sum_j4 = rtb_ManualSwitch; + } + + rtb_uDLookupTable = ((PitchNormalLaw_rtP.Constant_Value_h - rtb_Sum_j4) * rtb_Gain + rtb_Y_b * rtb_Sum_j4) * + rtb_Divide + rtb_Product1_dm; + if (rtb_uDLookupTable > PitchNormalLaw_rtP.Saturation_UpperSat_kp) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_UpperSat_kp; + } else if (rtb_uDLookupTable < PitchNormalLaw_rtP.Saturation_LowerSat_a4) { + rtb_uDLookupTable = PitchNormalLaw_rtP.Saturation_LowerSat_a4; + } + + PitchNormalLaw_RateLimiter(rtb_uDLookupTable, PitchNormalLaw_rtP.RateLimitereta_up, + PitchNormalLaw_rtP.RateLimitereta_lo, rtu_In_time_dt, PitchNormalLaw_rtP.RateLimitereta_InitialCondition, + rty_Out_eta_deg, &PitchNormalLaw_DWork.sf_RateLimiter_b); + PitchNormalLaw_DWork.Delay_DSTATE_h = rtb_Loaddemand2_l; + PitchNormalLaw_DWork.Delay_DSTATE_n = rtb_Gain_px; + PitchNormalLaw_DWork.Delay_DSTATE_c = rtb_Gain_ot; + PitchNormalLaw_DWork.Delay_DSTATE_l = rtb_Divide_an; + PitchNormalLaw_DWork.Delay_DSTATE_k = rtb_Gain1_ft; + PitchNormalLaw_DWork.Delay_DSTATE_d = rtb_Gain_bs; + PitchNormalLaw_DWork.Delay_DSTATE_f = rtb_Divide_m; + PitchNormalLaw_DWork.Delay_DSTATE_g = rtb_Sum10; + PitchNormalLaw_DWork.Delay_DSTATE_j = rtb_y; + PitchNormalLaw_DWork.Delay_DSTATE_ca = rtb_Divide_o2; + PitchNormalLaw_DWork.Delay_DSTATE_kd = rtb_alpha_err_gain; + PitchNormalLaw_DWork.Delay_DSTATE_ku = rtb_qk_gain; + PitchNormalLaw_DWork.Delay_DSTATE_gl = rtb_qk_dot_gain; + PitchNormalLaw_DWork.Delay_DSTATE_m = rtb_Gain_ll; + PitchNormalLaw_DWork.Delay_DSTATE_k2 = rtb_Divide_b4; + PitchNormalLaw_DWork.Delay_DSTATE_jh = rtb_Cos; + PitchNormalLaw_DWork.Delay_DSTATE_dy = rtb_Y_p; + PitchNormalLaw_DWork.Delay_DSTATE_e5 = rtb_Y_n; + PitchNormalLaw_DWork.Delay_DSTATE_gz = rtb_Y_ll; + PitchNormalLaw_DWork.Delay_DSTATE_lf = rtb_Y_am; + PitchNormalLaw_DWork.Delay_DSTATE_ho = rtb_Divide_cq; + PitchNormalLaw_DWork.Delay_DSTATE_ds = rtb_Loaddemand2; + PitchNormalLaw_DWork.Delay_DSTATE_jt = rtb_Gain_g; + PitchNormalLaw_DWork.icLoad = false; + PitchNormalLaw_DWork.Delay_DSTATE_ej = rtb_eta_trim_deg_rate_limit_up_deg_s; + PitchNormalLaw_DWork.Delay_DSTATE_e4 = rtb_eta_trim_deg_rate_limit_lo_deg_s; + PitchNormalLaw_DWork.icLoad_p = false; +} + +PitchNormalLaw::PitchNormalLaw(): + PitchNormalLaw_B(), + PitchNormalLaw_DWork() +{ +} + +PitchNormalLaw::~PitchNormalLaw() +{ +} diff --git a/src-a339/fmgc/src/guidance/vnav/FlightModel.ts b/src-a339/fmgc/src/guidance/vnav/FlightModel.ts new file mode 100644 index 000000000..8a7f23f99 --- /dev/null +++ b/src-a339/fmgc/src/guidance/vnav/FlightModel.ts @@ -0,0 +1,287 @@ +import { MathUtils } from '@shared/MathUtils'; +import { Common, FlapConf } from './common'; + +export class FlightModel { + static Cd0 = 0.0200; + + static wingSpan = 209.97; + + static wingArea = 5010; + + static wingEffcyFactor = 0.72; + + static requiredAccelRateKNS = 1.33; // in knots/second + + static requiredAccelRateMS2 = 0.684; // in m/s^2 + + static gravityConstKNS = 19.0626 // in knots/second + + static gravityConstMS2 = 9.806665; // in m/s^2 + + // From https://github.com/flybywiresim/a32nx/pull/6903#issuecomment-1073168320 + static machValues: Mach[] = [0.55, 0.6, 0.65, 0.7, 0.75, 0.8, 0.85] + + static dragCoefficientCorrections: number[] = [0, 0.0002, 0.0003, 0.0004, 0.0008, 0.0015, 0.01] + + /** + * Get lift coefficient at given conditions + * @param weight in pounds + * @param mach self-explanatory + * @param delta pressure at the altitude divided by the pressure at sea level + * @param loadFactor g-Force + * @returns lift coefficient (Cl) + */ + static getLiftCoefficient(weight: number, mach: number, delta: number, loadFactor = 1): number { + return (weight * loadFactor) / (1481.4 * (mach ** 2) * delta * this.wingArea); + } + + static getLiftCoefficientFromEAS(lift: number, eas: number): number { + return (295.369 * lift) / ((eas ** 2) * this.wingArea); + } + + /** + * Get drag coefficient at given conditions + * @param Cl coefficient of lift + * @param spdBrkDeflected whether speedbrake is deflected at half or not + * @param gearExtended whether gear is extended or not + * @param flapConf flap configuration + * @returns drag coefficient (Cd) + */ + static getDragCoefficient(Cl: number, spdBrkDeflected = false, gearExtended = false, flapConf = FlapConf.CLEAN) : number { + // Values taken at mach 0 + let baseDrag; + switch (flapConf) { + case FlapConf.CLEAN: + baseDrag = (-0.1043 * Cl ** 5) + (0.2635 * Cl ** 4) - (0.2319 * Cl ** 3) + (0.1537 * Cl ** 2) - (0.0379 * Cl) + 0.0233; + break; + case FlapConf.CONF_1: + baseDrag = (-0.0207 * Cl ** 5) + (0.0764 * Cl ** 4) - (0.0813 * Cl ** 3) + (0.0912 * Cl ** 2) - (0.0285 * Cl) + 0.0337; + break; + case FlapConf.CONF_2: + baseDrag = (0.0066 * Cl ** 5) - (0.0271 * Cl ** 4) + (0.0615 * Cl ** 3) - (0.0187 * Cl ** 2) + (0.0035 * Cl) + 0.0538; + break; + case FlapConf.CONF_3: + baseDrag = (0.0768 * Cl ** 5) - (0.3979 * Cl ** 4) + (0.8252 * Cl ** 3) - (0.7951 * Cl ** 2) + (0.3851 * Cl) - 0.0107; + break; + case FlapConf.CONF_FULL: + baseDrag = (0.017 * Cl ** 5) - (0.0978 * Cl ** 4) + (0.2308 * Cl ** 3) - (0.2278 * Cl ** 2) + (0.1157 * Cl) + 0.0682; + break; + default: + break; + } + + const spdBrkIncrement = spdBrkDeflected ? 0.00611 : 0; + const gearIncrement = gearExtended ? 0.03 : 0; + return baseDrag + spdBrkIncrement + gearIncrement; + } + + /** + * Get drag at given conditions + * @param weight in pounds + * @param mach self-explanatory + * @param delta pressure at the altitude divided by the pressure at sea level + * @param spdBrkDeflected Whether speedbrake is deflected at half or not + * @param gearExtended whether gear is extended or not + * @param flapConf flap configuration + * @returns drag + */ + static getDrag(weight: number, mach: number, delta: number, spdBrkDeflected: boolean, gearExtended: boolean, flapConf: FlapConf): number { + const Cl = this.getLiftCoefficient(weight, mach, delta); + const Cd = this.getDragCoefficient(Cl, spdBrkDeflected, gearExtended, flapConf); + const deltaCd = this.getMachCorrection(mach, flapConf); + + return 1481.4 * (mach ** 2) * delta * this.wingArea * (Cd + deltaCd); + } + + static getMachCorrection(mach: Mach, flapConf: FlapConf): number { + if (flapConf !== FlapConf.CLEAN) { + return 0; + } + + return this.interpolate(mach, this.machValues, this.dragCoefficientCorrections); + } + + /** + * Interpolates in a list + * @param x The value to look up in in `xs`. + * @param xs The table of x values with known y values + * @param ys The y values corresponding to the x values in `xs` + */ + static interpolate(x: number, xs: number[], ys: number[]) { + if (x <= xs[0]) { + return ys[0]; + } + + for (let i = 0; i < xs.length - 1; i++) { + if (x > xs[i] && x <= xs[i + 1]) { + return Common.interpolate(x, xs[i], xs[i + 1], ys[i], ys[i + 1]); + } + } + + return ys[ys.length - 1]; + } + + // NEW + + /** + * Returns the available climb or descent gradient. + * + * @param thrust the thrust in lbf + * @param drag + * @param weight in lbs + * + * @returns the available gradient in radians + */ + static getAvailableGradient( + thrust: number, + drag: number, + weight: number, + ): number { + return Math.asin((thrust - drag) / weight); + } + + /** + * Returns an acceleration for a given available gradient, fpa and acceleration factor. + * + * @param availableGradient in radians + * @param fpa in radians + * @param accelFactor + * + * @returns the acceleration + */ + static accelerationForGradient( + availableGradient: Radians, + fpa: number, + accelFactor: number, + ): number { + return (Math.sin(availableGradient) - Math.sin(fpa)) * accelFactor; + } + + /** + * Returns an fpa for a given available gradient, acceleration and acceleration factor. + * + * @param availableGradient in radians + * @param acceleration + * @param accelFactor + * + * @returns the fpa in radians + */ + static fpaForGradient( + availableGradient: Radians, + acceleration: number, + accelFactor: number, + ): number { + return Math.asin(Math.sin(availableGradient) - (acceleration / accelFactor)); + } + + // END NEW + + static getConstantThrustPathAngle( + thrust: number, + weight: number, + drag: number, + accelFactor: number, + ): number { + return Math.asin(((thrust - drag) / weight) / accelFactor); + } + + static getConstantThrustPathAngleFromCoefficients( + thrust: number, + weight: number, + Cl: number, + Cd: number, + accelFactor: number, + ): number { + return Math.asin(((thrust / weight) - (Cd / Cl)) / accelFactor); + } + + static getThrustFromConstantPathAngle( + fpa: number, + weight: number, + drag: number, + accelFactor: number, + ): number { + // fpa is in degrees + return weight * (accelFactor * Math.sin(fpa * MathUtils.DEGREES_TO_RADIANS)) + drag; + } + + static getThrustFromConstantPathAngleCoefficients( + fpa: number, + weight: number, + Cl: number, + Cd: number, + accelFactor: number, + ): number { + // fpa is in degrees + return weight * (accelFactor * Math.sin(fpa * MathUtils.DEGREES_TO_RADIANS) + (Cd / Cl)); + } + + static getSpeedChangePathAngle( + thrust: number, + weight: number, + drag: number, + ): number { + return Math.asin(((thrust - drag) / weight) - (1 / FlightModel.gravityConstMS2) * FlightModel.requiredAccelRateMS2); + } + + static getSpeedChangePathAngleFromCoefficients( + thrust: number, + weight: number, + Cl: number, + Cd: number, + ): number { + return Math.asin(((thrust / weight) - (Cd / Cl)) - (1 / FlightModel.gravityConstMS2) * FlightModel.requiredAccelRateMS2); + } + + static getAccelRateFromIdleGeoPath( + thrust: number, + weight: number, + drag: number, + fpaDeg: number, + ): number { + // fpa is in degrees + const fpaRad = fpaDeg * MathUtils.DEGREES_TO_RADIANS; + return FlightModel.gravityConstKNS * ((thrust - drag) / weight - Math.sin(fpaRad)); + } + + static getAccelRateFromIdleGeoPathCoefficients( + thrust: number, + weight: number, + Cl: number, + Cd: number, + fpaDeg: number, + ): number { + // fpa is in degrees + const fpaRad = fpaDeg * MathUtils.DEGREES_TO_RADIANS; + return FlightModel.gravityConstKNS * (((thrust / weight) - (Cd / Cl)) - Math.sin(fpaRad)); + } + + /** + * Gets distance required to accelerate/decelerate + * @param thrust + * @param drag + * @param weight in pounds + * @param initialSpeed + * @param targetSpeed + * @param fpa flight path angle, default value 0 for level segments + * @param accelFactor acceleration factor, default value 0 for level segments + * @returns distance to accel/decel + */ + static getAccelerationDistance( + thrust: number, + drag: number, + weight: number, + initialSpeed: number, + targetSpeed: number, + fpa = 0, + accelFactor = 0, + ): number { + const sign = Math.sign(fpa); + const force = thrust - drag + (sign * weight * Math.sin(fpa * (Math.PI / 180))) * accelFactor; + + const accel = force / weight; // TODO: Check units + const timeToAccel = (targetSpeed - initialSpeed) / accel; + const distanceToAccel = (initialSpeed * timeToAccel) + (0.5 * accel * (timeToAccel ** 2)); // TODO: Check units + return distanceToAccel; + } +}