ryujin 2.1.1 revision 46bf70e400e423a8ffffe8300887eeb35b8dfb2c
initial_state_icf_like.h
Go to the documentation of this file.
1//
2// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
3// [LANL Copyright Statement]
4// Copyright (C) 2024 by the ryujin authors
5// Copyright (C) 2024 by Triad National Security, LLC
6//
7
8#pragma once
9
10#include <compile_time_options.h>
11
13
14namespace ryujin
15{
16 namespace EulerInitialStates
17 {
29 template <typename Description, int dim, typename Number>
30 class ICFLike : public InitialState<Description, dim, Number>
31 {
32 public:
34 using View =
35 typename Description::template HyperbolicSystemView<dim, Number>;
36 using state_type = typename View::state_type;
37
38 ICFLike(const HyperbolicSystem &hyperbolic_system,
39 const std::string subsection)
40 : InitialState<Description, dim, Number>("icf like", subsection)
41 , hyperbolic_system_(hyperbolic_system)
42 {
43 gamma_ = 1.4;
44 if constexpr (!View::have_gamma) {
45 this->add_parameter("gamma", gamma_, "The ratio of specific heats");
46 }
47
48 primitive_inside_[0] = 0.1;
49 primitive_inside_[1] = 0.0;
50 primitive_inside_[2] = 1.0;
51 this->add_parameter(
52 "primitive state inside",
53 primitive_inside_,
54 "Initial primitive state (rho, u, p) inside perturbed interface");
55
56 primitive_outside_[0] = 1.0;
57 primitive_outside_[1] = 0.0;
58 primitive_outside_[2] = 1.0;
59 this->add_parameter(
60 "primitive state outside",
61 primitive_outside_,
62 "Initial primitive state (rho, u, p) outside perturbed interface");
63
64 interface_radius_ = 1.0;
65 this->add_parameter(
66 "interface radius", interface_radius_, "Radius of interface");
67
68 num_modes_ = 8.0;
69 this->add_parameter("number of modes",
70 num_modes_,
71 "Number of modes for pertburation of interface");
72
73 amplitude_ = 0.02;
74 this->add_parameter(
75 "amplitude", amplitude_, "Amplitude for interface pertburation");
76
77 mach_number_ = 3.0;
78 this->add_parameter(
79 "mach number", mach_number_, "Mach number of incoming shock front");
80
81 shock_radius_ = 1.2;
82 this->add_parameter("shock radius",
83 shock_radius_,
84 "Radial location of incoming shock front");
85
86 const auto convert_states = [&]() {
87 const auto view = hyperbolic_system_.template view<dim, Number>();
88 state_inside_ = view.from_initial_state(primitive_inside_);
89 state_outside_ = view.from_initial_state(primitive_outside_);
90 };
91 this->parse_parameters_call_back.connect(convert_states);
92 convert_states();
93 };
94
95 state_type compute(const dealii::Point<dim> &point, Number) final
96 {
97 const auto view = hyperbolic_system_.template view<dim, Number>();
98
99 /* Compute incoming shock state */
100 state_type conserved_shock_state;
101 const auto r_hat = point / point.norm();
102 {
103 auto b = Number(0.);
104 if constexpr (View::have_eos_interpolation_b)
105 b = view.eos_interpolation_b();
106
107 const auto &rho_R = primitive_outside_[0];
108 const auto &u_R = primitive_outside_[1];
109 const auto &p_R = primitive_outside_[2];
110 /* a_R^2 = gamma * p / rho / (1 - b * rho) */
111 const Number a_R = std::sqrt(gamma_ * p_R / rho_R / (1 - b * rho_R));
112 const Number mach_R = u_R / a_R;
113
114 auto S3_ = mach_number_ * a_R;
115 const Number delta_mach = mach_R - mach_number_;
116
117 const Number rho_L =
118 rho_R * (gamma_ + Number(1.)) * delta_mach * delta_mach /
119 ((gamma_ - Number(1.)) * delta_mach * delta_mach + Number(2.));
120 const Number u_L =
121 (Number(1.) - rho_R / rho_L) * S3_ + rho_R / rho_L * u_R;
122 const Number p_L = p_R *
123 (Number(2.) * gamma_ * delta_mach * delta_mach -
124 (gamma_ - Number(1.))) /
125 (gamma_ + Number(1.));
126
127 state_type primitive_shock_state;
128 primitive_shock_state[0] = rho_L;
129
130 for (unsigned int i = 0; i < dim; ++i) {
131 primitive_shock_state[i + 1] = 0.;
132 }
133
134 if (point.norm() > 0.) {
135 for (unsigned int i = 0; i < dim; ++i) {
136 primitive_shock_state[i + 1] = -u_L * r_hat[i];
137 }
138 }
139 primitive_shock_state[1 + dim] = p_L;
140
141 conserved_shock_state =
142 view.from_initial_state(primitive_shock_state);
143 }
144
145 /* Compute polar (and potentially azimuthal) angle */
146 const auto x = point[0];
147 const auto y = dim > 1 ? point[1] : 0.;
148
149 const double theta = std::atan2(y, x);
150
151 double phi = 0.;
152 if constexpr (dim == 3)
153 phi = std::atan2(point[2], std::sqrt(x * x + y * y));
154
155 /* Compute perturbation for interface */
156 const auto omega = num_modes_;
157 const double perturbation =
158 amplitude_ * std::cos(omega * theta) * std::cos(omega * phi);
159
160 /* Compute state depending on location */
161 auto full_state =
162 (point.norm() > interface_radius_ + perturbation ? state_outside_
163 : state_inside_);
164
165 if (point.norm() > shock_radius_) {
166 full_state = conserved_shock_state;
167 }
168
169 /* Set final state */
170 return full_state;
171 }
172
173 private:
174 const HyperbolicSystem &hyperbolic_system_;
175
176 Number gamma_;
177
178 dealii::Tensor<1, 3, Number> primitive_inside_;
179 dealii::Tensor<1, 3, Number> primitive_outside_;
180 state_type state_inside_;
181 state_type state_outside_;
182
183 double interface_radius_;
184 double num_modes_;
185 double amplitude_;
186 double shock_radius_;
187 double mach_number_;
188 };
189
190
191 } // namespace EulerInitialStates
192} // namespace ryujin
ICFLike(const HyperbolicSystem &hyperbolic_system, const std::string subsection)
state_type compute(const dealii::Point< dim > &point, Number) final
typename Description::HyperbolicSystem HyperbolicSystem
typename Description::template HyperbolicSystemView< dim, Number > View
Euler::HyperbolicSystem HyperbolicSystem
Definition: description.h:34