From 39cb2a4d875770682f8d73aac119032376b7fa4b Mon Sep 17 00:00:00 2001 From: Michael Zingale Date: Wed, 29 May 2024 12:34:57 -0400 Subject: [PATCH] unify the SDC integrator code paths (#1546) This removes a lot of the redundancy in the various actual_integrator_sdc.H files --- .../BackwardEuler/actual_integrator_sdc.H | 223 +-------------- integration/Make.package | 1 + integration/RKC/actual_integrator_sdc.H | 221 +-------------- integration/VODE/actual_integrator_sdc.H | 227 +-------------- integration/_parameters | 3 - integration/integrator_setup_sdc.H | 265 ++++++++++++++++++ 6 files changed, 288 insertions(+), 652 deletions(-) create mode 100644 integration/integrator_setup_sdc.H diff --git a/integration/BackwardEuler/actual_integrator_sdc.H b/integration/BackwardEuler/actual_integrator_sdc.H index e81eaf5641..7f117325f1 100644 --- a/integration/BackwardEuler/actual_integrator_sdc.H +++ b/integration/BackwardEuler/actual_integrator_sdc.H @@ -1,14 +1,13 @@ #ifndef actual_integrator_H #define actual_integrator_H -#include +#include +#include -#include +#include #include #include -#include -#include template AMREX_GPU_HOST_DEVICE AMREX_INLINE @@ -16,223 +15,15 @@ void actual_integrator (BurnT& state, const amrex::Real dt, bool is_retry=false) { constexpr int int_neqs = integrator_neqs(); - be_t be; - - // Start off by assuming a successful burn. - - state.success = true; - - // Initialize the integration time. - - be.t = 0.0; - be.tout = dt; - - // set the Jacobian type - if (is_retry && integrator_rp::retry_swap_jacobian) { - be.jacobian_type = (jacobian == 1) ? 2 : 1; - } else { - be.jacobian_type = jacobian; - } - - // Fill in the initial integration state. - - burn_to_int(state, be); - - // Save the initial composition and temperature for our later diagnostics. - -#ifndef AMREX_USE_GPU - amrex::Real xn_in[NumSpec]; - for (int n = 0; n < NumSpec; ++n) { - xn_in[n] = state.y[SFS+n] / state.y[SRHO]; - } - // we are assuming that the temperature was valid on input - amrex::Real T_in = state.T; -#ifdef AUX_THERMO - amrex::Real aux_in[NumAux]; - for (int n = 0; n < NumAux; ++n) { - aux_in[n] = state.y[SFX+n] / state.y[SRHO]; - } -#endif - amrex::Real rhoe_in = state.y[SEINT]; -#endif - - - // Set the tolerances. - - amrex::Real sdc_tol_fac = std::pow(integrator_rp::sdc_burn_tol_factor, state.num_sdc_iters - state.sdc_iter - 1); - - // we use 1-based indexing inside of BackwardEuler, so we need to shift the - // indices SRHO, SFS, etc by 1 - - amrex::Real sdc_min_density = amrex::min(state.rho, state.rho_orig + state.ydot_a[SRHO] * dt); - - if (!is_retry) { - - be.atol_enuc = sdc_min_density * atol_enuc * sdc_tol_fac; - be.rtol_enuc = rtol_enuc * sdc_tol_fac; - - // Note: we define the input atol for species to refer only to the - // mass fraction part, and we multiply by a representative density - // so that atol becomes an absolutely tolerance on (rho X) - - be.atol_spec = sdc_min_density * atol_spec * sdc_tol_fac; - be.rtol_spec = rtol_spec * sdc_tol_fac; - - } else { - - be.atol_enuc = sdc_min_density * retry_atol_enuc * sdc_tol_fac; - be.rtol_enuc = retry_rtol_enuc * sdc_tol_fac; - - // Note: we define the input atol for species to refer only to the - // mass fraction part, and we multiply by a representative density - // so that atol becomes an absolutely tolerance on (rho X) - - be.atol_spec = sdc_min_density * retry_atol_spec * sdc_tol_fac; - be.rtol_spec = retry_rtol_spec * sdc_tol_fac; - - } - - if (scale_system) { - // the absolute tol for energy needs to reflect the scaled - // energy the integrator sees - be.atol_enuc /= state.e_scale; - } + auto be_state = integrator_setup>(state, dt, is_retry); + auto state_save = integrator_backup(state); // Call the integration routine. - int istate = be_integrator(state, be); + int istate = be_integrator(state, be_state); state.error_code = istate; - // Get the number of RHS and Jacobian evaluations. - - state.n_rhs = be.n_rhs; - state.n_jac = be.n_jac; - state.n_step = be.n_step; - - // Copy the integration data back to the burn state. - // This will also update the aux state from X if we are using NSE - - int_to_burn(be.t, be, state); - - // we only evolved (rho e), not (rho E), so we need to update the - // total energy now to ensure we are conservative - - amrex::Real rho_Sdot = 0.0_rt; - if (state.time > 0) { - rho_Sdot = (state.y[SEINT] - state.rhoe_orig) / state.time - state.ydot_a[SEINT]; - } - - state.y[SEDEN] += state.time * (state.ydot_a[SEDEN] + rho_Sdot); - - // also momentum - - state.y[SMX] += state.time * state.ydot_a[SMX]; - state.y[SMY] += state.time * state.ydot_a[SMY]; - state.y[SMZ] += state.time * state.ydot_a[SMZ]; - - // normalize the abundances on exit. We'll assume that the driver - // calling this is making use of the conserved state (state.y[]), - // so that is what will be normalized. - - normalize_abundances_sdc_burn(state); - - // BE does not always fail even though it can lead to unphysical states. - // Add some checks that indicate a burn fail even if VODE thinks the - // integration was successful. - - if (istate != IERR_SUCCESS) { - state.success = false; - } - - if (state.y[SEINT] < 0.0_rt) { - state.success = false; - } - - for (int n = 0; n < NumSpec; ++n) { - if (state.y[SFS+n] / state.rho < -species_failure_tolerance) { - state.success = false; - } - - if (state.y[SFS+n] / state.rho > 1.0_rt + species_failure_tolerance) { - state.success = false; - } - } - - -#ifndef AMREX_USE_GPU - if (burner_verbose) { - // Print out some integration statistics, if desired. - std::cout << "integration summary: " << std::endl; - std::cout << "dens: " << state.rho << " temp: " << state.T << std::endl; - std::cout << "energy released: " << state.e << std::endl; - std::cout << "number of steps taken: " << state.n_step << std::endl; - std::cout << "number of f evaluations: " << state.n_rhs << std::endl; - } -#endif - - // If we failed, print out the current state of the integration. - - if (!state.success) { - if (istate != IERR_ENTERED_NSE) { -#ifndef AMREX_USE_GPU - std::cout << amrex::Font::Bold << amrex::FGColor::Red << "[ERROR] integration failed in net" << amrex::ResetDisplay << std::endl; - std::cout << "istate = " << istate << std::endl; - if (istate == IERR_SUCCESS) { - std::cout << " BE exited successfully, but a check on the data values failed" << std::endl; - } - std::cout << "zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; - std::cout << "time = " << state.time << std::endl; - std::cout << "dt = " << std::setprecision(16) << dt << std::endl; - std::cout << "dens start = " << std::setprecision(16) << state.rho_orig << std::endl; - std::cout << "temp start = " << std::setprecision(16) << T_in << std::endl; - std::cout << "rhoe start = " << std::setprecision(16) << rhoe_in << std::endl; - std::cout << "xn start = "; - for (const auto X : xn_in) { - std::cout << std::setprecision(16) << X << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "aux start = "; - for (const auto aux : aux_in) { - std::cout << std::setprecision(16) << aux << " "; - } - std::cout << std::endl; -#endif - std::cout << "dens current = " << std::setprecision(16) << state.rho << std::endl; - std::cout << "temp current = " << std::setprecision(16) << state.T << std::endl; - std::cout << "xn current = "; - for (int n = 0; n < NumSpec; ++n) { - std::cout << std::setprecision(16) << state.xn[n] << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "aux current = "; - for (int n = 0; n < NumAux; ++n) { - std::cout << std::setprecision(16) << state.aux[n] << " "; - } - std::cout << std::endl; -#endif - std::cout << "A(rho) = " << std::setprecision(16) << state.ydot_a[SRHO] << std::endl; - std::cout << "A(rho e) = " << std::setprecision(16) << state.ydot_a[SEINT] << std::endl; - std::cout << "A(rho X_k) = "; - for (int n = 0; n < NumSpec; n++) { - std::cout << std::setprecision(16) << state.ydot_a[SFS+n] << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "A(rho aux_k) = "; - for (int n = 0; n < NumAux; n++) { - std::cout << std::setprecision(16) << state.ydot_a[SFX+n] << " "; - } - std::cout << std::endl; -#endif -#endif - } else { -#ifndef AMREX_USE_GPU - std::cout << "burn entered NSE during integration (after " << state.n_step << " steps), zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; -#endif - } - } + integrator_cleanup(be_state, state, istate, state_save, dt); } diff --git a/integration/Make.package b/integration/Make.package index b08b6b118d..534310c29d 100644 --- a/integration/Make.package +++ b/integration/Make.package @@ -19,6 +19,7 @@ CEXE_headers += integrator_type.H ifeq ($(USE_ALL_SDC), TRUE) CEXE_headers += integrator_rhs_sdc.H CEXE_headers += integrator_type_sdc.H + CEXE_headers += integrator_setup_sdc.H else CEXE_headers += integrator_type_strang.H CEXE_headers += integrator_rhs_strang.H diff --git a/integration/RKC/actual_integrator_sdc.H b/integration/RKC/actual_integrator_sdc.H index 4a1b338de9..103136306d 100644 --- a/integration/RKC/actual_integrator_sdc.H +++ b/integration/RKC/actual_integrator_sdc.H @@ -1,16 +1,11 @@ #ifndef actual_integrator_H #define actual_integrator_H -#include - -#include - #include #include -#include -#include -#include -#include + +#include + #include #include @@ -20,220 +15,18 @@ template AMREX_GPU_HOST_DEVICE AMREX_INLINE void actual_integrator (BurnT& state, amrex::Real dt, bool is_retry=false) { - constexpr int int_neqs = integrator_neqs(); - - rkc_t rkc_state{}; - - // Start off by assuming a successful burn. - - state.success = true; - - // Initialize the integration time. - - rkc_state.t = 0.0_rt; - rkc_state.tout = dt; - - // Fill in the initial integration state. - - burn_to_int(state, rkc_state); - - // Save the initial composition and temperature for our later diagnostics. - -#ifndef AMREX_USE_GPU - amrex::Real xn_in[NumSpec]; - for (int n = 0; n < NumSpec; ++n) { - xn_in[n] = state.y[SFS+n] / state.y[SRHO]; - } - // we are assuming that the temperature was valid on input - amrex::Real T_in = state.T; -#ifdef AUX_THERMO - amrex::Real aux_in[NumAux]; - for (int n = 0; n < NumAux; ++n) { - aux_in[n] = state.y[SFX+n] / state.y[SRHO]; - } -#endif - amrex::Real rhoe_in = state.y[SEINT]; -#endif - - - // Set the tolerances. - - amrex::Real sdc_tol_fac = std::pow(sdc_burn_tol_factor, state.num_sdc_iters - state.sdc_iter - 1); - - // we use 1-based indexing inside of RKC, so we need to shift the - // indices SRHO, SFS, etc by 1 - - amrex::Real sdc_min_density = amrex::min(state.rho, state.rho_orig + state.ydot_a[SRHO] * dt); - - if (!is_retry) { - - rkc_state.atol_enuc = sdc_min_density * atol_enuc * sdc_tol_fac; - rkc_state.rtol_enuc = rtol_enuc * sdc_tol_fac; - - // Note: we define the input atol for species to refer only to the - // mass fraction part, and we multiply by a representative density - // so that atol becomes an absolutely tolerance on (rho X) - - rkc_state.atol_spec = sdc_min_density * atol_spec * sdc_tol_fac; - rkc_state.rtol_spec = rtol_spec * sdc_tol_fac; - - } else { - - rkc_state.atol_enuc = sdc_min_density * retry_atol_enuc * sdc_tol_fac; - rkc_state.rtol_enuc = retry_rtol_enuc * sdc_tol_fac; - - // Note: we define the input atol for species to refer only to the - // mass fraction part, and we multiply by a representative density - // so that atol becomes an absolutely tolerance on (rho X) - - rkc_state.atol_spec = sdc_min_density * retry_atol_spec * sdc_tol_fac; - rkc_state.rtol_spec = retry_rtol_spec * sdc_tol_fac; - } - - if (scale_system) { - // the absolute tol for energy needs to reflect the scaled - // energy the integrator sees - rkc_state.atol_enuc /= state.e_scale; - } + constexpr int int_neqs = integrator_neqs(); + auto rkc_state = integrator_setup>(state, dt, is_retry); + auto state_save = integrator_backup(state); // Call the integration routine. int istate = rkc(state, rkc_state); state.error_code = istate; - // Get the number of RHS and Jacobian evaluations. - - state.n_rhs = rkc_state.n_rhs; - state.n_jac = 0; - state.n_step = rkc_state.n_step; - - // Copy the integration data back to the burn state. - // This will also update the aux state from X if we are using NSE - - int_to_burn(rkc_state.t, rkc_state, state); - - // we only evolved (rho e), not (rho E), so we need to update the - // total energy now to ensure we are conservative - - amrex::Real rho_Sdot = 0.0_rt; - if (state.time > 0) { - rho_Sdot = (state.y[SEINT] - state.rhoe_orig) / state.time - state.ydot_a[SEINT]; - } - - state.y[SEDEN] += state.time * (state.ydot_a[SEDEN] + rho_Sdot); - - // also momentum - - state.y[SMX] += state.time * state.ydot_a[SMX]; - state.y[SMY] += state.time * state.ydot_a[SMY]; - state.y[SMZ] += state.time * state.ydot_a[SMZ]; - - // normalize the abundances on exit. We'll assume that the driver - // calling this is making use of the conserved state (state.y[]), - // so that is what will be normalized. - - normalize_abundances_sdc_burn(state); - - // RKC does not always fail even though it can lead to unphysical states. - // Add some checks that indicate a burn fail even if RKC thinks the - // integration was successful. - - if (istate != IERR_SUCCESS) { - state.success = false; - } - - if (state.y[SEINT] < 0.0_rt) { - state.success = false; - } - - for (int n = 0; n < NumSpec; ++n) { - if (state.y[SFS+n] / state.rho < -species_failure_tolerance) { - state.success = false; - } - - if (state.y[SFS+n] / state.rho > 1.0_rt + species_failure_tolerance) { - state.success = false; - } - } - - -#ifndef AMREX_USE_GPU - if (burner_verbose) { - // Print out some integration statistics, if desired. - std::cout << "integration summary: " << std::endl; - std::cout << "dens: " << state.rho << " temp: " << state.T << std::endl; - std::cout << " energy released: " << state.e << std::endl; - std::cout << "number of steps taken: " << state.n_step << std::endl; - std::cout << "number of f evaluations: " << state.n_rhs << std::endl; - } -#endif - - // If we failed, print out the current state of the integration. - - if (!state.success) { - if (istate != IERR_ENTERED_NSE) { -#ifndef AMREX_USE_GPU - std::cout << amrex::Font::Bold << amrex::FGColor::Red << "[ERROR] integration failed in net" << amrex::ResetDisplay << std::endl; - std::cout << "istate = " << istate << std::endl; - if (istate == IERR_SUCCESS) { - std::cout << " RKC exited successfully, but a check on the data values failed" << std::endl; - } - std::cout << "zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; - std::cout << "time = " << state.time << std::endl; - std::cout << "dt = " << std::setprecision(16) << dt << std::endl; - std::cout << "dens start = " << std::setprecision(16) << state.rho_orig << std::endl; - std::cout << "temp start = " << std::setprecision(16) << T_in << std::endl; - std::cout << "rhoe start = " << std::setprecision(16) << rhoe_in << std::endl; - std::cout << "xn start = "; - for (auto X : xn_in) { - std::cout << std::setprecision(16) << X << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "aux start = "; - for (auto aux : aux_in) { - std::cout << std::setprecision(16) << aux << " "; - } - std::cout << std::endl; -#endif - std::cout << "dens current = " << std::setprecision(16) << state.rho << std::endl; - std::cout << "temp current = " << std::setprecision(16) << state.T << std::endl; - std::cout << "xn current = "; - for (int n = 0; n < NumSpec; ++n) { - std::cout << std::setprecision(16) << state.xn[n] << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "aux current = "; - for (int n = 0; n < NumAux; ++n) { - std::cout << std::setprecision(16) << state.aux[n] << " "; - } - std::cout << std::endl; -#endif - std::cout << "A(rho) = " << std::setprecision(16) << state.ydot_a[SRHO] << std::endl; - std::cout << "A(rho e) = " << std::setprecision(16) << state.ydot_a[SEINT] << std::endl; - std::cout << "A(rho X_k) = "; - for (int n = 0; n < NumSpec; n++) { - std::cout << std::setprecision(16) << state.ydot_a[SFS+n] << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "A(rho aux_k) = "; - for (int n = 0; n < NumAux; n++) { - std::cout << std::setprecision(16) << state.ydot_a[SFX+n] << " "; - } - std::cout << std::endl; -#endif -#endif - } else { -#ifndef AMREX_USE_GPU - std::cout << "burn entered NSE during integration (after " << state.n_step << " steps), zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; -#endif - } - } + integrator_cleanup(rkc_state, state, istate, state_save, dt); } - #endif diff --git a/integration/VODE/actual_integrator_sdc.H b/integration/VODE/actual_integrator_sdc.H index f034e219e5..8cc855c75d 100644 --- a/integration/VODE/actual_integrator_sdc.H +++ b/integration/VODE/actual_integrator_sdc.H @@ -4,15 +4,11 @@ // Common variables and routines for burners // that use VODE for their integration. -#include - -#include - #include #include -#include -#include -#include + +#include + #include #include @@ -22,226 +18,19 @@ template AMREX_GPU_HOST_DEVICE AMREX_INLINE void actual_integrator (BurnT& state, amrex::Real dt, bool is_retry=false) { - constexpr int int_neqs = integrator_neqs(); - - dvode_t vode_state{}; - - // Start off by assuming a successful burn. - - state.success = true; - - // Initialize the integration time. - - vode_state.t = 0.0_rt; - vode_state.tout = dt; - - - // set the Jacobian type - if (is_retry && integrator_rp::retry_swap_jacobian) { - vode_state.jacobian_type = (jacobian == 1) ? 2 : 1; - } else { - vode_state.jacobian_type = jacobian; - } - - // Fill in the initial integration state. - - burn_to_int(state, vode_state); - - // Save the initial composition and temperature for our later diagnostics. - -#ifndef AMREX_USE_GPU - amrex::Real xn_in[NumSpec]; - for (int n = 0; n < NumSpec; ++n) { - xn_in[n] = state.y[SFS+n] / state.y[SRHO]; - } - // we are assuming that the temperature was valid on input - amrex::Real T_in = state.T; -#ifdef AUX_THERMO - amrex::Real aux_in[NumAux]; - for (int n = 0; n < NumAux; ++n) { - aux_in[n] = state.y[SFX+n] / state.y[SRHO]; - } -#endif - amrex::Real rhoe_in = state.y[SEINT]; -#endif - - - // Set the tolerances. - - amrex::Real sdc_tol_fac = std::pow(integrator_rp::sdc_burn_tol_factor, state.num_sdc_iters - state.sdc_iter - 1); - - // we use 1-based indexing inside of VODE, so we need to shift the - // indices SRHO, SFS, etc by 1 - - amrex::Real sdc_min_density = amrex::min(state.rho, state.rho_orig + state.ydot_a[SRHO] * dt); - - if (!is_retry) { - - vode_state.atol_enuc = sdc_min_density * atol_enuc * sdc_tol_fac; - vode_state.rtol_enuc = rtol_enuc * sdc_tol_fac; - - // Note: we define the input atol for species to refer only to the - // mass fraction part, and we multiply by a representative density - // so that atol becomes an absolutely tolerance on (rho X) - - vode_state.atol_spec = sdc_min_density * atol_spec * sdc_tol_fac; - vode_state.rtol_spec = rtol_spec * sdc_tol_fac; - - } else { - vode_state.atol_enuc = sdc_min_density * retry_atol_enuc * sdc_tol_fac; - vode_state.rtol_enuc = retry_rtol_enuc * sdc_tol_fac; - - // Note: we define the input atol for species to refer only to the - // mass fraction part, and we multiply by a representative density - // so that atol becomes an absolutely tolerance on (rho X) - - vode_state.atol_spec = sdc_min_density * retry_atol_spec * sdc_tol_fac; - vode_state.rtol_spec = retry_rtol_spec * sdc_tol_fac; - - } + constexpr int int_neqs = integrator_neqs(); - if (scale_system) { - // the absolute tol for energy needs to reflect the scaled - // energy the integrator sees - vode_state.atol_enuc /= state.e_scale; - } + auto vode_state = integrator_setup>(state, dt, is_retry); + auto state_save = integrator_backup(state); // Call the integration routine. - int istate = dvode(state, vode_state); + auto istate = dvode(state, vode_state); state.error_code = istate; - // Get the number of RHS and Jacobian evaluations. - - state.n_rhs = vode_state.n_rhs; - state.n_jac = vode_state.n_jac; - state.n_step = vode_state.n_step; - - // Copy the integration data back to the burn state. - // This will also update the aux state from X if we are using NSE - - int_to_burn(vode_state.t, vode_state, state); - - // we only evolved (rho e), not (rho E), so we need to update the - // total energy now to ensure we are conservative - - amrex::Real rho_Sdot = 0.0_rt; - if (state.time > 0) { - rho_Sdot = (state.y[SEINT] - state.rhoe_orig) / state.time - state.ydot_a[SEINT]; - } - - state.y[SEDEN] += state.time * (state.ydot_a[SEDEN] + rho_Sdot); - - // also momentum - - state.y[SMX] += state.time * state.ydot_a[SMX]; - state.y[SMY] += state.time * state.ydot_a[SMY]; - state.y[SMZ] += state.time * state.ydot_a[SMZ]; - - // normalize the abundances on exit. We'll assume that the driver - // calling this is making use of the conserved state (state.y[]), - // so that is what will be normalized. - - normalize_abundances_sdc_burn(state); + integrator_cleanup(vode_state, state, istate, state_save, dt); - // VODE does not always fail even though it can lead to unphysical states. - // Add some checks that indicate a burn fail even if VODE thinks the - // integration was successful. - - if (istate != IERR_SUCCESS) { - state.success = false; - } - - if (state.y[SEINT] < 0.0_rt) { - state.success = false; - } - - for (int n = 0; n < NumSpec; ++n) { - if (state.y[SFS+n] / state.rho < -species_failure_tolerance) { - state.success = false; - } - - if (state.y[SFS+n] / state.rho > 1.0_rt + species_failure_tolerance) { - state.success = false; - } - } - - -#ifndef AMREX_USE_GPU - if (burner_verbose) { - // Print out some integration statistics, if desired. - std::cout << "integration summary: " << std::endl; - std::cout << "dens: " << state.rho << " temp: " << state.T << std::endl; - std::cout << " energy released: " << state.e << std::endl; - std::cout << "number of steps taken: " << state.n_step << std::endl; - std::cout << "number of f evaluations: " << state.n_rhs << std::endl; - } -#endif - - // If we failed, print out the current state of the integration. - - if (!state.success) { - if (istate != IERR_ENTERED_NSE) { -#ifndef AMREX_USE_GPU - std::cout << amrex::Font::Bold << amrex::FGColor::Red << "[ERROR] integration failed in net" << amrex::ResetDisplay << std::endl; - std::cout << "istate = " << istate << std::endl; - if (istate == IERR_SUCCESS) { - std::cout << " VODE exited successfully, but a check on the data values failed" << std::endl; - } - std::cout << "zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; - std::cout << "time = " << state.time << std::endl; - std::cout << "dt = " << std::setprecision(16) << dt << std::endl; - std::cout << "dens start = " << std::setprecision(16) << state.rho_orig << std::endl; - std::cout << "temp start = " << std::setprecision(16) << T_in << std::endl; - std::cout << "rhoe start = " << std::setprecision(16) << rhoe_in << std::endl; - std::cout << "xn start = "; - for (const auto X : xn_in) { - std::cout << std::setprecision(16) << X << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "aux start = "; - for (const auto aux : aux_in) { - std::cout << std::setprecision(16) << aux << " "; - } - std::cout << std::endl; -#endif - std::cout << "dens current = " << std::setprecision(16) << state.rho << std::endl; - std::cout << "temp current = " << std::setprecision(16) << state.T << std::endl; - std::cout << "xn current = "; - for (int n = 0; n < NumSpec; ++n) { - std::cout << std::setprecision(16) << state.xn[n] << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "aux current = "; - for (int n = 0; n < NumAux; ++n) { - std::cout << std::setprecision(16) << state.aux[n] << " "; - } - std::cout << std::endl; -#endif - std::cout << "A(rho) = " << std::setprecision(16) << state.ydot_a[SRHO] << std::endl; - std::cout << "A(rho e) = " << std::setprecision(16) << state.ydot_a[SEINT] << std::endl; - std::cout << "A(rho X_k) = "; - for (int n = 0; n < NumSpec; n++) { - std::cout << std::setprecision(16) << state.ydot_a[SFS+n] << " "; - } - std::cout << std::endl; -#ifdef AUX_THERMO - std::cout << "A(rho aux_k) = "; - for (int n = 0; n < NumAux; n++) { - std::cout << std::setprecision(16) << state.ydot_a[SFX+n] << " "; - } - std::cout << std::endl; -#endif -#endif - } else { -#ifndef AMREX_USE_GPU - std::cout << "burn entered NSE during integration (after " << state.n_step << " steps), zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; -#endif - } - } } diff --git a/integration/_parameters b/integration/_parameters index e7423504da..1c85504e41 100644 --- a/integration/_parameters +++ b/integration/_parameters @@ -84,9 +84,6 @@ use_number_densities bool 0 # flag for tuning on the subtraction of internal energy subtract_internal_energy bool 1 -# SDC iteration tolerance adjustment factor -sdc_burn_tol_factor real 1.e0 - # do we scale the ODE system we integrate to make it O(1)? # for Strang, this simply means scaling e by the initial energy? scale_system bool 0 diff --git a/integration/integrator_setup_sdc.H b/integration/integrator_setup_sdc.H new file mode 100644 index 0000000000..6ebce35203 --- /dev/null +++ b/integration/integrator_setup_sdc.H @@ -0,0 +1,265 @@ +#ifndef INTEGRATOR_SETUP_SDC_H +#define INTEGRATOR_SETUP_SDC_H + +#include +#include + +#include + +#include +#include +#include +#include +#include + +#include + +struct state_backup_t { + amrex::Real T_in{}; + amrex::Real rhoe_in{}; +#ifndef AMREX_USE_GPU + amrex::Real xn_in[NumSpec]{}; +#ifdef AUX_THERMO + amrex::Real aux_in[NumAux]{}; +#endif +#endif +}; + +template +AMREX_GPU_HOST_DEVICE AMREX_INLINE +IntegratorT integrator_setup (BurnT& state, amrex::Real dt, bool is_retry) +{ + + IntegratorT int_state{}; + + // Start off by assuming a successful burn. + + state.success = true; + + // Initialize the integration time. + + int_state.t = 0.0_rt; + int_state.tout = dt; + + + // set the Jacobian type + if (is_retry && integrator_rp::retry_swap_jacobian) { + int_state.jacobian_type = (integrator_rp::jacobian == 1) ? 2 : 1; + } else { + int_state.jacobian_type = integrator_rp::jacobian; + } + + // Fill in the initial integration state. + + burn_to_int(state, int_state); + + // Set the tolerances. + + amrex::Real sdc_min_density = + amrex::min(state.rho, + state.rho_orig + state.ydot_a[SRHO] * dt); + + if (!is_retry) { + + int_state.atol_enuc = sdc_min_density * integrator_rp::atol_enuc; + int_state.rtol_enuc = integrator_rp::rtol_enuc; + + // Note: we define the input atol for species to refer only to the + // mass fraction part, and we multiply by a representative density + // so that atol becomes an absolutely tolerance on (rho X) + + int_state.atol_spec = sdc_min_density * integrator_rp::atol_spec; + int_state.rtol_spec = integrator_rp::rtol_spec; + + } else { + + int_state.atol_enuc = sdc_min_density * integrator_rp::retry_atol_enuc; + int_state.rtol_enuc = integrator_rp::retry_rtol_enuc; + + // Note: we define the input atol for species to refer only to the + // mass fraction part, and we multiply by a representative density + // so that atol becomes an absolutely tolerance on (rho X) + + int_state.atol_spec = sdc_min_density * integrator_rp::retry_atol_spec; + int_state.rtol_spec = integrator_rp::retry_rtol_spec; + + } + + if (integrator_rp::scale_system) { + // the absolute tol for energy needs to reflect the scaled + // energy the integrator sees + int_state.atol_enuc /= state.e_scale; + } + + return int_state; +} + +template +AMREX_GPU_HOST_DEVICE AMREX_INLINE +state_backup_t integrator_backup (const BurnT& state) { + + // Save the initial composition and temperature for our later + // diagnostics. + + state_backup_t state_save; + +#ifndef AMREX_USE_GPU + for (int n = 0; n < NumSpec; ++n) { + state_save.xn_in[n] = state.y[SFS+n] / state.y[SRHO]; + } + // we are assuming that the temperature was valid on input + state_save.T_in = state.T; +#ifdef AUX_THERMO + for (int n = 0; n < NumAux; ++n) { + state_save.aux_in[n] = state.y[SFX+n] / state.y[SRHO]; + } +#endif + state_save.rhoe_in = state.y[SEINT]; +#endif + + return state_save; + +} + + +template +AMREX_GPU_HOST_DEVICE AMREX_INLINE +void integrator_cleanup (IntegratorT& int_state, BurnT& state, + int istate, const state_backup_t& state_save, amrex::Real dt) +{ + + // Copy the integration data back to the burn state. + + // Get the number of RHS and Jacobian evaluations. + + state.n_rhs = int_state.n_rhs; + state.n_jac = int_state.n_jac; + state.n_step = int_state.n_step; + + // Copy the integration data back to the burn state. + // This will also update the aux state from X if we are using NSE + + int_to_burn(int_state.t, int_state, state); + + // we only evolved (rho e), not (rho E), so we need to update the + // total energy now to ensure we are conservative + + amrex::Real rho_Sdot = 0.0_rt; + if (state.time > 0) { + rho_Sdot = (state.y[SEINT] - state.rhoe_orig) / state.time - state.ydot_a[SEINT]; + } + + state.y[SEDEN] += state.time * (state.ydot_a[SEDEN] + rho_Sdot); + + // also momentum + + state.y[SMX] += state.time * state.ydot_a[SMX]; + state.y[SMY] += state.time * state.ydot_a[SMY]; + state.y[SMZ] += state.time * state.ydot_a[SMZ]; + + // normalize the abundances on exit. We'll assume that the driver + // calling this is making use of the conserved state (state.y[]), + // so that is what will be normalized. + + normalize_abundances_sdc_burn(state); + + // The integrator may not always fail even though it can lead to + // unphysical states. Add some checks that indicate a burn fail + // even if Vthe integrator thinks the integration was successful. + + if (istate != IERR_SUCCESS) { + state.success = false; + } + + if (state.y[SEINT] < 0.0_rt) { + state.success = false; + } + + for (int n = 0; n < NumSpec; ++n) { + if (state.y[SFS+n] / state.rho < -species_failure_tolerance) { + state.success = false; + } + + if (state.y[SFS+n] / state.rho > 1.0_rt + species_failure_tolerance) { + state.success = false; + } + } + + +#ifndef AMREX_USE_GPU + if (integrator_rp::burner_verbose) { + // Print out some integration statistics, if desired. + std::cout << "integration summary: " << std::endl; + std::cout << "dens: " << state.rho << " temp: " << state.T << std::endl; + std::cout << " energy released: " << state.e << std::endl; + std::cout << "number of steps taken: " << state.n_step << std::endl; + std::cout << "number of f evaluations: " << state.n_rhs << std::endl; + } +#endif + + // If we failed, print out the current state of the integration. + + if (!state.success) { + if (istate != IERR_ENTERED_NSE) { +#ifndef AMREX_USE_GPU + std::cout << amrex::Font::Bold << amrex::FGColor::Red << "[ERROR] integration failed in net" << amrex::ResetDisplay << std::endl; + std::cout << "istate = " << istate << std::endl; + if (istate == IERR_SUCCESS) { + std::cout << " integrator exited successfully, but a check on the data values failed" << std::endl; + } + std::cout << "zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; + std::cout << "time = " << state.time << std::endl; + std::cout << "dt = " << std::setprecision(16) << dt << std::endl; + std::cout << "dens start = " << std::setprecision(16) << state.rho_orig << std::endl; + std::cout << "temp start = " << std::setprecision(16) << state_save.T_in << std::endl; + std::cout << "rhoe start = " << std::setprecision(16) << state_save.rhoe_in << std::endl; + std::cout << "xn start = "; + for (const auto X : state_save.xn_in) { + std::cout << std::setprecision(16) << X << " "; + } + std::cout << std::endl; +#ifdef AUX_THERMO + std::cout << "aux start = "; + for (const auto aux : state_save.aux_in) { + std::cout << std::setprecision(16) << aux << " "; + } + std::cout << std::endl; +#endif + std::cout << "dens current = " << std::setprecision(16) << state.rho << std::endl; + std::cout << "temp current = " << std::setprecision(16) << state.T << std::endl; + std::cout << "xn current = "; + for (int n = 0; n < NumSpec; ++n) { + std::cout << std::setprecision(16) << state.xn[n] << " "; + } + std::cout << std::endl; +#ifdef AUX_THERMO + std::cout << "aux current = "; + for (int n = 0; n < NumAux; ++n) { + std::cout << std::setprecision(16) << state.aux[n] << " "; + } + std::cout << std::endl; +#endif + std::cout << "A(rho) = " << std::setprecision(16) << state.ydot_a[SRHO] << std::endl; + std::cout << "A(rho e) = " << std::setprecision(16) << state.ydot_a[SEINT] << std::endl; + std::cout << "A(rho X_k) = "; + for (int n = 0; n < NumSpec; n++) { + std::cout << std::setprecision(16) << state.ydot_a[SFS+n] << " "; + } + std::cout << std::endl; +#ifdef AUX_THERMO + std::cout << "A(rho aux_k) = "; + for (int n = 0; n < NumAux; n++) { + std::cout << std::setprecision(16) << state.ydot_a[SFX+n] << " "; + } + std::cout << std::endl; +#endif +#endif + } else { +#ifndef AMREX_USE_GPU + std::cout << "burn entered NSE during integration (after " << state.n_step << " steps), zone = (" << state.i << ", " << state.j << ", " << state.k << ")" << std::endl; +#endif + } + } + +} +#endif