15 template <
int dim,
typename Number>
16 std::tuple<Number, bool>
23 const auto view = hyperbolic_system.view<dim, Number>();
28 constexpr ScalarNumber eps = std::numeric_limits<ScalarNumber>::epsilon();
29 const auto small = view.vacuum_state_relaxation_small();
30 const auto large = view.vacuum_state_relaxation_large();
41 const auto &rho_U = view.density(U);
42 const auto &rho_P = view.density(P);
44 const auto &rho_min = std::get<0>(bounds);
45 const auto &rho_max = std::get<1>(bounds);
51 const auto test_min = view.filter_vacuum_density(
52 std::max(Number(0.), rho_U - relax * rho_max));
53 const auto test_max = view.filter_vacuum_density(
54 std::max(Number(0.), rho_min - relax * rho_U));
55 if (!(test_min == Number(0.) && test_max == Number(0.))) {
57 std::cout << std::fixed << std::setprecision(16);
58 std::cout <<
"Bounds violation: low-order density (critical)!"
59 <<
"\n\t\trho min: " << rho_min
60 <<
"\n\t\trho min (delta): "
62 <<
"\n\t\trho: " << rho_U
63 <<
"\n\t\trho max (delta): "
65 <<
"\n\t\trho max: " << rho_max <<
"\n"
71 const Number denominator =
74 constexpr auto lt = dealii::SIMDComparison::less_than;
76 t_r = dealii::compare_and_apply_mask<lt>(
85 (rho_max - rho_U) * denominator,
88 t_r = dealii::compare_and_apply_mask<lt>(
97 (rho_U - rho_min) * denominator,
107 t_r = std::min(t_r, t_max);
108 t_r = std::max(t_r, t_min);
110#ifdef EXPENSIVE_BOUNDS_CHECK
114 const auto rho_new = view.density(U + t_r * P);
115 const auto test_new_min = view.filter_vacuum_density(
116 std::max(Number(0.), rho_new - relax * rho_max));
117 const auto test_new_max = view.filter_vacuum_density(
118 std::max(Number(0.), rho_min - relax * rho_new));
119 if (!(test_new_min == Number(0.) && test_new_max == Number(0.))) {
121 std::cout << std::fixed << std::setprecision(16);
122 std::cout <<
"Bounds violation: high-order density!"
123 <<
"\n\t\trho min: " << rho_min
124 <<
"\n\t\trho min (delta): "
126 <<
"\n\t\trho: " << rho_new
127 <<
"\n\t\trho max (delta): "
129 <<
"\n\t\trho max: " << rho_max <<
"\n"
164 const auto &s_min = std::get<2>(bounds);
166#ifdef DEBUG_OUTPUT_LIMITER
167 std::cout << std::endl;
168 std::cout << std::fixed << std::setprecision(16);
169 std::cout <<
"t_l: (start) " << t_l << std::endl;
170 std::cout <<
"t_r: (start) " << t_r << std::endl;
173 for (
unsigned int n = 0; n < parameters.newton_max_iterations(); ++n) {
175 const auto U_r = U + t_r * P;
176 const auto rho_r = view.density(U_r);
177 const auto rho_r_gamma =
ryujin::pow(rho_r, gamma);
178 const auto rho_e_r = view.internal_energy(U_r);
181 relax_small * rho_r * rho_e_r - s_min * rho_r * rho_r_gamma;
183#ifndef EXPENSIVE_BOUNDS_CHECK
188 t_l = dealii::compare_and_apply_mask<
189 dealii::SIMDComparison::greater_than>(
190 psi_r, Number(0.), t_r, t_l);
208#ifdef DEBUG_OUTPUT_LIMITER
209 std::cout <<
"shortcut: t_l == t_r" << std::endl;
210 std::cout <<
"psi_l: " << psi_l << std::endl;
211 std::cout <<
"psi_r: " << psi_r << std::endl;
212 std::cout <<
"t_l: ( " << n <<
" ) " << t_l << std::endl;
213 std::cout <<
"t_r: ( " << n <<
" ) " << t_r << std::endl;
219 const auto U_l = U + t_l * P;
220 const auto rho_l = view.density(U_l);
221 const auto rho_l_gamma =
ryujin::pow(rho_l, gamma);
222 const auto rho_e_l = view.internal_energy(U_l);
225 relax_small * rho_l * rho_e_l - s_min * rho_l * rho_l_gamma;
231 const auto lower_bound =
232 (
ScalarNumber(1.) - relax) * s_min * rho_l * rho_l_gamma;
234 !(std::min(Number(0.), psi_l - lower_bound) == Number(0.))) {
236 std::cout << std::fixed << std::setprecision(16);
238 <<
"Bounds violation: low-order specific entropy (critical)!\n";
239 std::cout <<
"\t\tPsi left: 0 <= " << psi_l <<
"\n" << std::endl;
244#ifdef EXPENSIVE_BOUNDS_CHECK
249 t_l = dealii::compare_and_apply_mask<
250 dealii::SIMDComparison::greater_than>(
251 psi_r, Number(0.), t_r, t_l);
258 const Number tolerance(parameters.newton_tolerance());
259 if (std::max(Number(0.), t_r - t_l - tolerance) == Number(0.)) {
260#ifdef DEBUG_OUTPUT_LIMITER
261 std::cout <<
"break: t_l and t_r within tolerance" << std::endl;
262 std::cout <<
"psi_l: " << psi_l << std::endl;
263 std::cout <<
"psi_r: " << psi_r << std::endl;
264 std::cout <<
"t_l: ( " << n <<
" ) " << t_l << std::endl;
265 std::cout <<
"t_r: ( " << n <<
" ) " << t_r << std::endl;
272 const auto drho = view.density(P);
273 const auto drho_e_l = view.internal_energy_derivative(U_l) * P;
274 const auto drho_e_r = view.internal_energy_derivative(U_r) * P;
276 rho_l * drho_e_l + (rho_e_l - gp1 * s_min * rho_l_gamma) * drho;
278 rho_r * drho_e_r + (rho_e_r - gp1 * s_min * rho_r_gamma) * drho;
281 t_l, t_r, psi_l, psi_r, dpsi_l, dpsi_r, Number(-1.));
283#ifdef DEBUG_OUTPUT_LIMITER
284 std::cout <<
"psi_l: " << psi_l << std::endl;
285 std::cout <<
"psi_r: " << psi_r << std::endl;
286 std::cout <<
"dpsi_l: " << dpsi_l << std::endl;
287 std::cout <<
"dpsi_r: " << dpsi_r << std::endl;
288 std::cout <<
"t_l: ( " << n <<
" ) " << t_l << std::endl;
289 std::cout <<
"t_r: ( " << n <<
" ) " << t_r << std::endl;
293#ifdef EXPENSIVE_BOUNDS_CHECK
298 const auto U_new = U + t_l * P;
299 const auto rho_new = view.density(U_new);
300 const auto rho_new_gamma =
ryujin::pow(rho_new, gamma);
301 const auto rho_e_new = view.internal_energy(U_new);
303 auto psi_new = relax_small * rho_new * rho_e_new -
304 s_min * rho_new * rho_new_gamma;
306 const auto lower_bound =
307 (
ScalarNumber(1.) - relax) * s_min * rho_new * rho_new_gamma;
309 const bool e_valid = std::min(Number(0.), rho_e_new) == Number(0.);
310 const bool psi_valid =
311 std::min(Number(0.), psi_new - lower_bound) == Number(0.);
313 if (!e_valid || !psi_valid) {
315 std::cout << std::fixed << std::setprecision(16);
316 std::cout <<
"Bounds violation: high-order specific entropy!\n";
317 std::cout <<
"\t\trho e: 0 <= " << rho_e_new <<
"\n";
318 std::cout <<
"\t\tPsi: 0 <= " << psi_new <<
"\n" << std::endl;
326 return {t_l, success};
typename View::ScalarNumber ScalarNumber
typename View::state_type state_type
std::array< Number, n_bounds > Bounds
std::tuple< Number, bool > limit(const Bounds &bounds, const state_type &U, const state_type &P, const Number t_min=Number(0.), const Number t_max=Number(1.))
T pow(const T x, const T b)
DEAL_II_ALWAYS_INLINE Number negative_part(const Number number)
DEAL_II_ALWAYS_INLINE Number positive_part(const Number number)
DEAL_II_ALWAYS_INLINE void quadratic_newton_step(Number &p_1, Number &p_2, const Number phi_p_1, const Number phi_p_2, const Number dphi_p_1, const Number dphi_p_2, const Number sign=Number(1.0))