Port pke_params, pke_th_rhs, pke_linearize, and all five controllers to Julia. sim_sanity.jl reproduces the MATLAB main.m operation-mode scenario (100%->80% Q_sg step) and matches final state to 3 decimals across n, T_f, T_avg, T_cold, u. reach_operation.jl is a stub: ReachabilityAnalysis.jl (LGG09, GLGM06, BFFPSV18) numerically explodes on the raw stiff system — envelopes of 1e14 K to 1e37 K instead of the known-tight 0.03 K. Almost certainly a state-scaling issue: precursors C_i ~ 1e5, temperatures ~ 300, eigvals span 5000x. Diagonal scaling + retry is planned; left for the next pass since the hand-rolled MATLAB reach already discharges the operation-mode obligation. Project.toml pins OrdinaryDiffEq >= 6.111 (the one that precompiled cleanly on first instantiate). Manifest gitignored. Hacker-Split: Julia path open, reach side needs a scaling pass. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
50 lines
1.5 KiB
Julia
50 lines
1.5 KiB
Julia
"""
|
|
Mode controllers — signatures match the MATLAB side:
|
|
u = ctrl_<mode>(t, x, plant, ref)
|
|
|
|
Pure functions. `ref` can be any NamedTuple of setpoints; unused fields
|
|
are ignored. For heatup, `ref` must provide `T_start`, `T_target`, `ramp_rate`.
|
|
"""
|
|
|
|
ctrl_null(t, x, plant, ref) = 0.0
|
|
|
|
ctrl_shutdown(t, x, plant, ref) = -5.0 * plant.beta
|
|
ctrl_scram(t, x, plant, ref) = -8.0 * plant.beta
|
|
|
|
function ctrl_operation(t, x, plant, ref)
|
|
Kp = 1e-4
|
|
T_avg = x[9]
|
|
return Kp * (ref.T_avg - T_avg)
|
|
end
|
|
|
|
function ctrl_heatup(t, x, plant, ref)
|
|
Kp = 1e-4
|
|
T_f = x[8]
|
|
T_avg = x[9]
|
|
u_ff = -plant.alpha_f * (T_f - plant.T_f0) -
|
|
plant.alpha_c * (T_avg - plant.T_c0)
|
|
T_ref = min(ref.T_start + ref.ramp_rate * t, ref.T_target)
|
|
u_unsat = u_ff + Kp * (T_ref - T_avg)
|
|
u_min = get(ref, :u_min, -5 * plant.beta)
|
|
u_max = get(ref, :u_max, 0.5 * plant.beta)
|
|
return clamp(u_unsat, u_min, u_max)
|
|
end
|
|
|
|
"""
|
|
ctrl_operation_lqr_factory(plant; Q_lqr, R_lqr)
|
|
|
|
Returns a closure `ctrl(t, x, plant_ignored, ref_ignored)` with the LQR
|
|
gain baked in. Pattern chosen so the user can specify Q/R from the
|
|
calling script and get a pure function to pass to the ODE solver.
|
|
|
|
Depends on MatrixEquations.jl for `arec` (algebraic Riccati).
|
|
"""
|
|
function ctrl_operation_lqr_factory(plant, A, B; Q_lqr, R_lqr)
|
|
x_op = pke_initial_conditions(plant)
|
|
X_ric, _, _ = MatrixEquations.arec(A, B, R_lqr, Q_lqr)
|
|
K = (R_lqr \ B') * X_ric # 1x10
|
|
return function (t, x, plant_ignored, ref_ignored)
|
|
return -(K * (x - x_op))[1]
|
|
end, K, x_op
|
|
end
|