Alamo
Flame.cpp
Go to the documentation of this file.
1#include "Flame.H"
2#include "IO/ParmParse.H"
3#include "BC/Constant.H"
4#include "Numeric/Stencil.H"
5#include "IC/Laminate.H"
6#include "IC/Constant.H"
7#include "IC/PointList.H"
8#include "IC/PSRead.H"
9#include "Numeric/Function.H"
10#include "IC/Expression.H"
11#include "IC/BMP.H"
12#include "IC/PNG.H"
13#include "Base/Mechanics.H"
14#include "Util/Util.H"
18#include <cmath>
19
20namespace Integrator
21{
22
25
27{
28 pp_queryclass(*this);
29}
30
31
32void
34{
35 pp.forbid("pressure.P","use chamber.pressure instead");
36
37 pp.forbid("geometry.x_len","This is specified by geometry.prob_lo/hi");
38 pp.forbid("geometry.y_len","This is specified by geometry.prob_lo/hi");
39 pp.forbid("amr.ghost_cells", "This should not be adjustable ");
40
41 pp.forbid("pf.gamma","use propellant.powerlaw.gamma");
42
43 pp.forbid("pressure.r_ap", "use propellant.powerlaw.r_ap");
44 pp.forbid("pressure.r_htpb", "use propellant.powerlaw.r_htpb");
45 pp.forbid("pressure.r_comb", "use propellant.powerlaw.r_comb");
46 pp.forbid("pressure.n_ap", "use propellant.powerlaw.n_ap");
47 pp.forbid("pressure.n_htpb", "use propellant.powerlaw.n_htpb");
48 pp.forbid("pressure.n_comb", "use propellant.powerlaw.n_comb");
49
50 pp.forbid("thermal.bound", "use thermal.Tref");
51 pp.forbid("thermal.T_fluid", "use thermal.Tfluid (or nothing)");
52 pp.forbid("thermal.m_ap", "use propellant.fullfeedback.m_ap");
53 pp.forbid("thermal.m_htpb", "use propellant.fullfeedback.m_htpb");
54 pp.forbid("thermal.E_ap", "use propellant.fullfeedback.E_ap");
55 pp.forbid("thermal.E_htpb", "use propellant.fullfeedback.E_htpb");
56 pp.forbid("thermal.modeling_ap", "Old debug variable. Should equal 1 ");
57 pp.forbid("thermal.modeling_htpb", "Old debug variable. Should equal 1");
58
59 pp.forbid("pressure.a1", "use propellant.fullfeedback.a1 instead");
60 pp.forbid("pressure.a2", "use propellant.fullfeedback.a2 instead");
61 pp.forbid("pressure.a3", "use propellant.fullfeedback.a3 instead");
62 pp.forbid("pressure.b1", "use propellant.fullfeedback.b1 instead");
63 pp.forbid("pressure.b2", "use propellant.fullfeedback.b2 instead");
64 pp.forbid("pressure.b3", "use propellant.fullfeedback.b3 instead");
65 pp.forbid("pressure.c1", "use propellant.fullfeedback.c1 instead");
66 pp.forbid("pressure.mob_ap", "no longer used");
67 pp.forbid("pressure.dependency", "use propellant.fullfeedback.pressure_dependency");
68 pp.forbid("pressure.h1", "use propellant.homogenize.h1 instead");
69 pp.forbid("pressure.h2", "use propellant.homogenize.h2 instead");
70 pp.forbid("thermal.mlocal_ap", "use propellant.homogenize.mlocal_ap");
71 pp.forbid("thermal.mlocal_comb", "use propellant.homogenize.mlocal_comb");
72 pp.forbid("thermal.mlocal_htpb", "this actually did **nothing** - it was overridden by a hard code using massfraction.");
73
74 pp.forbid("thermal.disperssion1", "use propellant.homogenize.dispersion1");
75 pp.forbid("thermal.disperssion2", "use propellant.homogenize.dispersion2");
76 pp.forbid("thermal.disperssion3", "use propellant.homogenize.dispersion3");
77
78 pp.forbid("thermal.rho_ap", "use propellant.fullfeedback/homogenize.rho_ap ");
79 pp.forbid("thermal.rho_htpb","use propellant.fullfeedback/homogenize.rho_htpb ");
80 pp.forbid("thermal.k_ap", "use propellant.fullfeedback/homogenize.k_ap ");
81 pp.forbid("thermal.k_htpb", "use propellant.fullfeedback/homogenize.k_htpb ");
82 pp.forbid("thermal.cp_ap", "use propellant.fullfeedback/homogenize.cp_ap ");
83 pp.forbid("thermal.cp_htpb","use propellant.fullfeedback/homogenize.cp_htpb ");
84}
85
86
87// [parser]
88void
90{
91 BL_PROFILE("Integrator::Flame::Flame()");
92
93 Forbids(pp);
94
95 // Whether to include extra fields (such as mdot, etc) in the plot output
96 pp.query_default("plot_field",value.plot_field,true);
97
98 //
99 // PHASE FIELD VARIABLES
100 //
101
102 // Burn width thickness
103 pp.query_default("pf.eps", value.pf.eps, "1.0_m", Unit::Length());
104 // Interface energy param
105 pp.query_default("pf.kappa", value.pf.kappa, "0.0_J/m^2", Unit::Energy() / Unit::Area());
106 // Chemical potential multiplier
107 pp.query_default("pf.lambda", value.pf.lambda, "0.0_J/m^2", Unit::Energy()/Unit::Area());
108 // Unburned rest energy
109 pp.query_default("pf.w1", value.pf.w1, "0.0",Unit::Less());
110 // Barrier energy
111 pp.query_default("pf.w12", value.pf.w12, "0.0", Unit::Less());
112 // Burned rest energy
113 pp.query_default("pf.w0", value.pf.w0, "0.0",Unit::Less());
114
115 // Boundary conditions for phase field order params
116 pp.select<BC::Constant>("pf.eta.bc", value.bc_eta, 1 );
117 value.RegisterNewFab(value.eta_mf, value.bc_eta, 1, 2, "eta", true);
118 value.RegisterNewFab(value.eta_old_mf, value.bc_eta, 1, 2, "eta_old", 0);
119
120 // Inital value of eta that doesn't evolve and is used during refiment to set the updated values of eta with voids in the domain.
121 // Used to fix a bug where duirn refinement, a void won't be updated correctly and would be a square, not a circle
122 value.RegisterNewFab(value.eta_0_mf, value.bc_eta, 1, 2, "eta_0", 0);
123
124 // value.RegisterNewFab(value.eta_mf_frozen, value.bc_eta, 1, 2, "eta_frozen", value.plot_field);
125
126 // phase field initial condition
128
129
130 // Select reduced order model to capture heat feedback
134 ("propellant",value.propellant);
135
136
137 // Whether to use the Thermal Transport Model
138 pp_query_default("thermal.on", value.thermal.on, false);
139
140 // Reference temperature
141 // Used to set all other reference temperatures by default.
142 pp_query_default("thermal.Tref", value.thermal.Tref, "300.0_K",Unit::Temperature());
143
144 if (value.thermal.on) {
145
146 // Used to change heat flux units
147 pp_query_default("thermal.hc", value.thermal.hc, "1.0", Unit::Power()/Unit::Area());
148
149 // Effective fluid temperature, temp of the eta = 0 (fluid) region
150 pp_query_default("thermal.Tfluid", value.thermal.Tfluid, value.thermal.Tref);
151
152 // Cutoff value for regression, if T < Tcutoff eta won't evolve/regress
153 pp.query_default("thermal.Tcutoff", value.thermal.Tcutoff, "0.0", Unit::Temperature());
154
155 // Switch time of the improved regridding where eta and the temperature field are both used. It is recommended to make this time ~10x the timestep.
156 // Before this the refinement is based on the gradient of eta which helps the laser IC start correctly. A regrid is forced when this time is reached.
157 pp.query_default("thermal.end_initial_refine_time", value.thermal.end_initial_refine_time, "0.0", Unit::Time());
158
159 // Inital refinement of the phi field based on phi gradient. After time > end_initial_refine_time stops refining at these phi values.
160 pp.query_default("thermal.phi_refinement_criterion_inital", value.thermal.phi_refinement_criterion_inital, 1.0e100);
161
162 //Temperature boundary condition
163 pp.select_default<BC::Constant>("thermal.temp.bc", value.bc_temp, 1, Unit::Temperature());
164
165 value.RegisterNewFab(value.temp_mf, value.bc_temp, 1, 3, "temp", true);
166 value.RegisterNewFab(value.temp_old_mf, value.bc_temp, 1, 3, "temp_old", false);
167 value.RegisterNewFab(value.temps_mf, value.bc_temp, 1, 0, "temps", false);
168
169 value.RegisterNewFab(value.mdot_mf, value.bc_temp, 1, 0, "mdot", value.plot_field);
170 value.RegisterNewFab(value.alpha_mf, value.bc_temp, 1, 0, "alpha", value.plot_field);
171 value.RegisterNewFab(value.heatflux_mf, value.bc_temp, 1, 0, "heatflux", value.plot_field);
172 value.RegisterNewFab(value.laser_mf, value.bc_temp, 1, 0, "laser", value.plot_field);
173
174 value.RegisterIntegratedVariable(&value.chamber.volume, "volume");
175 value.RegisterIntegratedVariable(&value.chamber.area, "area");
176 value.RegisterIntegratedVariable(&value.chamber.massflux, "mass_flux");
177
178 value.RegisterNewFab(value.thermal.has_exceeded_Tcutoff, value.bc_temp, 1, 2, "exceeded_Tcutoff", 0); // Used to determine where regression has started
179
180 // laser initial condition
183 ("laser.ic",value.ic_laser, value.geom, Unit::Power()/Unit::Area());
184
185 // thermal initial condition
188 IC::BMP,
189 IC::PNG >
190 ("temp.ic",value.thermal.ic_temp,value.geom, Unit::Temperature());
191 }
192
193
194 // Constant pressure value
195 pp_query_default("chamber.pressure", value.chamber.pressure, "1.0_MPa", Unit::Pressure());
196
197 // Whether to compute the pressure evolution
198 pp_query_default("variable_pressure", value.variable_pressure, false);
199
200 // Refinement criterion for eta field, if thermal is on, cells will only be tagged for refinement if T>0.9*TCutoff,
201 // and the gradient of eta > m_refinement_criterion at each cell
202 pp_query_default( "amr.refinement_criterion", value.m_refinement_criterion, "0.001",
203 Unit::Less());
204
205 // Refinement criterion for temperature field
206 pp.query_default( "amr.refinement_criterion_temp", value.t_refinement_criterion, "0.001_K",
208
209 // Eta value to restrict the refinament for the temperature field
210 pp.query_default( "amr.refinament_restriction", value.t_refinement_restriction, "0.1",
211 Unit::Less());
212
213 // Refinement criterion for phi field [infinity]
214 pp_query_default("amr.phi_refinement_criterion", value.phi_refinement_criterion, 1.0e100);
215
216 // Minimum allowable threshold for $\eta$
217 pp_query_default("small", value.small, 1.0e-8);
218
219 // Initial condition for $\phi$ field.
221 ("phi.ic",value.ic_phi,value.geom);
222
223 value.RegisterNodalFab(value.phi_mf, 1, 2, "phi", true);
224
225 // Whether to use Neo-hookean Elastic model
226 pp_query_default("elastic.on", value.elastic.on, 0);
227
228 // Body force
229 pp_query_default("elastic.traction", value.elastic.traction, 0.0);
230
231 // Phi refinement criteria
232 pp_query_default("elastic.phirefinement", value.elastic.phirefinement, 1);
233
234 pp.queryclass<Base::Mechanics<model_type>>("elastic",value);
235
236 if (value.m_type != Type::Disable)
237 {
238 // Reference temperature for thermal expansion
239 // (temperature at which the material is strain-free)
240 pp_query_default("Telastic", value.elastic.Telastic, value.thermal.Tref);
241 // elastic model of AP
243 // elastic model of HTPB
245
246 // eta cutoff value to stop applying the traction force and/or chamber pressure to the RHS.
247 // Below this vlaue the RHS is set to 0.0
248 pp.query_default("etacutoff",value.elastic.etacutoff, "0", Unit::Less());
249
250 // Boolean value, when not 0 the chamber.pressure value is applied at the diffuse interface where t>Tcutoff
251 pp.query_default("apply_chamber_pressure",value.elastic.apply_chamber_pressure, "0", Unit::Less());
252
253 // Use our current eta field as the psi field for the solver
254 value.psi_on = false;
255 value.solver.setPsi(value.eta_mf);
256 }
257
258 bool allow_unused;
259 // Set this to true to allow unused inputs without error.
260 // (Not recommended.)
261 pp.query_default("allow_unused",allow_unused,false);
262 if (!allow_unused && pp.AnyUnusedInputs())
263 {
264 Util::Warning(INFO,"The following inputs were specified but not used:");
265 pp.AllUnusedInputs();
266 Util::Exception(INFO,"Aborting. Specify 'allow_unused=True` to ignore this error.");
267 }
268}
269
270void Flame::Initialize(int lev)
271{
272 BL_PROFILE("Integrator::Flame::Initialize");
274
275 ic_eta->Initialize(lev, eta_mf);
277 ic_phi->Initialize(lev, phi_mf);
278 //ic_phicell->Initialize(lev, phicell_mf);
279
280 if (elastic.on) {
281 rhs_mf[lev]->setVal(Set::Vector::Zero());
282 }
283 if (thermal.on) {
284 if (thermal.ic_temp)
285 {
286 thermal.ic_temp->Initialize(lev,temp_mf);
287 thermal.ic_temp->Initialize(lev,temp_old_mf);
288 thermal.ic_temp->Initialize(lev,temps_mf);
289 }
290 else
291 {
292 temp_mf[lev]->setVal(thermal.Tref);
293 temp_old_mf[lev]->setVal(thermal.Tref);
294 temps_mf[lev]->setVal(thermal.Tref);
295 }
296 alpha_mf[lev]->setVal(0.0);
297 mdot_mf[lev]->setVal(0.0);
298 heatflux_mf[lev]->setVal(0.0);
300 }
301 if (variable_pressure) chamber.pressure = 1.0;
302}
303
304void Flame::UpdateModel(int /*a_step*/, Set::Scalar /*a_time*/)
305{
307
308 for (int lev = 0; lev <= finest_level; ++lev)
309 {
310 amrex::Box domain = this->geom[lev].Domain();
311 domain.convert(amrex::IntVect::TheNodeVector());
312 const Set::Scalar* DX = geom[lev].CellSize();
313
314 phi_mf[lev]->FillBoundary();
315 eta_mf[lev]->FillBoundary();
316 temp_mf[lev]->FillBoundary();
317
318 for (MFIter mfi(*model_mf[lev], false); mfi.isValid(); ++mfi)
319 {
320 amrex::Box smallbox = mfi.nodaltilebox();
321 amrex::Box bx = mfi.grownnodaltilebox() & domain;
322 Set::Patch<model_type> model = model_mf.Patch(lev,mfi);
325 Set::Patch<Set::Vector> rhs = rhs_mf.Patch(lev,mfi);
326 Set::Scalar Tcutoff = thermal.Tcutoff;
327
328 if (elastic.on)
329 {
331 amrex::ParallelFor(smallbox, [=] AMREX_GPU_DEVICE(int i, int j, int k)
332
333 {
334 Set::Vector grad_eta = Numeric::CellGradientOnNode(eta, i, j, k, 0, DX);
335
336 if (temp(i,j,k) > Tcutoff && eta(i,j,k) > elastic.etacutoff && elastic.apply_chamber_pressure)
337 {
338 rhs(i, j, k) = (elastic.traction) * grad_eta - chamber.pressure*grad_eta;
339 // std::cout << "Applying chamber pressure" << std::endl;
340 }
341 else if (temp(i,j,k) > Tcutoff && eta(i,j,k) > elastic.etacutoff && !elastic.apply_chamber_pressure)
342 {
343 rhs(i, j, k) = (elastic.traction) * grad_eta;
344 // std::cout << "Applying traction" << std::endl;
345 }
346 else
347 rhs(i, j, k) = 0.0 * grad_eta;
348 // std::cout << "Applying neither" << std::endl;
349
350 });
351 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
352 {
353 Set::Scalar phi_avg = phi(i, j, k, 0);
354 Set::Scalar temp_avg = Numeric::Interpolate::CellToNodeAverage(temp, i, j, k, 0);
355 model_type model_ap = elastic.model_ap;
356 model_ap.F0 -= Set::Matrix::Identity();
357 model_ap.F0 *= (temp_avg - elastic.Telastic);
358 model_ap.F0 += Set::Matrix::Identity();
359 model_type model_htpb = elastic.model_htpb;
360 model_htpb.F0 -= Set::Matrix::Identity();
361 model_htpb.F0 *= (temp_avg - elastic.Telastic);
362 model_htpb.F0 += Set::Matrix::Identity();
363
364 model(i, j, k) = (model_ap * phi_avg + model_htpb * (1. - phi_avg));
365 });
366 }
367 else
368 {
369 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
370 {
371 Set::Scalar phi_avg = Numeric::Interpolate::CellToNodeAverage(phi, i, j, k, 0);
372 //phi_avg = phi(i,j,k,0);
373 model_type model_ap = elastic.model_ap;
374 model_ap.F0 *= Set::Matrix::Zero();
375 model_type model_htpb = elastic.model_htpb;
376 model_htpb.F0 *= Set::Matrix::Zero();
377 model(i, j, k) = (model_ap * phi_avg + model_htpb * (1. - phi_avg));
378 });
379 }
380 }
381 Util::RealFillBoundary(*model_mf[lev], geom[lev]);
382
383 }
384}
385
386void Flame::TimeStepBegin(Set::Scalar a_time, int a_iter)
387{
388 BL_PROFILE("Integrator::Flame::TimeStepBegin");
390 if (thermal.on) {
391 for (int lev = 0; lev <= finest_level; ++lev)
392 ic_laser->Initialize(lev, laser_mf, a_time);
393 }
394
395 if (a_time > thermal.end_initial_refine_time)
396 {
397 if (!end_initial_refine) {
398 for (int lev = 0; lev <= finest_level; ++lev)
399 Flame::Regrid(lev, a_time);
401 }
402
403 prev_finest_ba = grids[finest_level];
404 prev_finest_level = finest_level;
405 }
406}
407
408void Flame::TimeStepComplete(Set::Scalar /*a_time*/, int /*a_iter*/)
409{
410 BL_PROFILE("Integrator::Flame::TimeStepComplete");
411 if (variable_pressure) {
412 //Set::Scalar x_len = geom[0].ProbDomain().length(0);
413 //Set::Scalar y_len = geom[0].ProbDomain().length(1);
414 // Set::Scalar domain_area = x_len * y_len;
415 Util::Message(INFO, "Mass = ", chamber.massflux);
416 Util::Message(INFO, "Pressure = ", chamber.pressure);
417 }
418}
419
421{
422 BL_PROFILE("Integrador::Flame::Advance");
424 const Set::Scalar* DX = geom[lev].CellSize();
425
426 std::swap(eta_old_mf[lev], eta_mf[lev]);
427
428 //
429 // Chamber pressure update
430 //
431 if (variable_pressure) {
432 chamber.pressure = exp(0.00075 * chamber.massflux);
433 if (chamber.pressure > 10.0) {
434 chamber.pressure = 10.0;
435 }
436 else if (chamber.pressure <= 0.99) {
437 chamber.pressure = 0.99;
438 }
439 elastic.traction = chamber.pressure;
440 }
441
442
443 //
444 // Multi-well chemical potential
445 //
447 0.0,
448 -5.0 * pf.w1 + 16.0 * pf.w12 - 11.0 * pf.w0,
449 14.0 * pf.w1 - 32.0 * pf.w12 + 18.0 * pf.w0,
450 -8.0 * pf.w1 + 16.0 * pf.w12 - 8.0 * pf.w0);
452
454
455 for (amrex::MFIter mfi(*eta_mf[lev], true); mfi.isValid(); ++mfi)
456 {
457 const amrex::Box& bx = mfi.tilebox();
458 // Phase fields
459 Set::Patch<Set::Scalar> etanew = eta_mf.Patch(lev,mfi);
462 // Heat transfer fields
464 Set::Patch<Set::Scalar> alpha = alpha_mf.Patch(lev,mfi);
465 Set::Patch<Set::Scalar> laser = laser_mf.Patch(lev,mfi);
466 // Diagnostic fields
467 Set::Patch<Set::Scalar> mdot = mdot_mf.Patch(lev,mfi);
468 Set::Patch<Set::Scalar> heatflux = heatflux_mf.Patch(lev,mfi);
469
470 Set::Patch<Set::Scalar> exceeded_Tcutoff = thermal.has_exceeded_Tcutoff.Patch(lev, mfi);
471 Set::Scalar Tcutoff = thermal.Tcutoff;
472
473
474 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
475 {
476 //
477 // CALCULATE PHI-AVERAGED QUANTITIES
478 //
479 Set::Scalar phi_avg = Numeric::Interpolate::NodeToCellAverage(phi, i, j, k, 0);
480 Set::Scalar T = thermal.on ? temp(i,j,k) : NAN;
481
482 Set::Scalar K = propellant.get_K(phi_avg);
483
485
486 Set::Scalar cp = propellant.get_cp(phi_avg);
487
488 //
489 // CALCULATE MOBILITY
490 //
491 Set::Scalar L = propellant.get_L( phi_avg, T);
492
493 //
494 // EVOLVE PHASE FIELD (ETA)
495 //
496
497 Set::Scalar eta_lap = Numeric::Laplacian(eta, i, j, k, 0, DX);
498 Set::Scalar df_deta = ((pf.lambda / pf.eps) * dw(eta(i, j, k)) - pf.eps * pf.kappa * eta_lap);
499
500 if (df_deta < 0) {
501 // Prevent eta from increasing/healing. A bug was found where if the diffuse thickness was too large compared to a void
502 // (region of eta = 0), eta would heal/increase in a non-physcial way, this statement stops that behavior
503 df_deta = 0.0;
504 }
505 if (thermal.on && T < thermal.Tcutoff) {
506 // If the temperature is lower then the cutoff temperature don't evolve the eta field
507 df_deta = 0.0;
508 }
509 etanew(i, j, k) = eta(i, j, k) - L * dt * df_deta;
510
511 if (etanew(i, j, k) <= small) etanew(i, j, k) = small;
512
513 if (thermal.on)
514 {
515 //
516 // Calculate thermal diffisivity and store for later gradient
517 //
518
519 alpha(i, j, k) = K / rho / cp;
520
521 //
522 // CALCULATE MASS FLUX BASED ON EVOLVING ETA
523 //
524
525 mdot(i, j, k) = rho * fabs(eta(i, j, k) - etanew(i, j, k)) / dt;
526
527 //
528 // CALCULATE HEAT FLUX BASED ON THE CALCULATED MASS FLUX
529 //
530
531 Set::Scalar q0 = propellant.get_qdot(mdot(i,j,k), phi_avg);
532 heatflux(i,j,k) = ( thermal.hc*q0 + laser(i,j,k) ) / K;
533
534 if (temp(i,j,k) > Tcutoff)
535 {
536 exceeded_Tcutoff(i,j,k) = 1;
537 }
538
539 }
540
541 });
542
543 } // MFi For loop
544
545
546 //
547 // THERMAL TRANSPORT
548 //
549 if (thermal.on)
550 {
551 std::swap(temp_old_mf[lev], temp_mf[lev]);
552
553 for (amrex::MFIter mfi(*eta_mf[lev], true); mfi.isValid(); ++mfi)
554 {
555 const amrex::Box& bx = mfi.tilebox();
556
557 Set::Patch<Set::Scalar> tempnew = temp_mf.Patch(lev,mfi);
560
561 Set::Patch<Set::Scalar> temps = temps_mf.Patch(lev,mfi);
562
563
564 // Phase field
565 Set::Patch<Set::Scalar> etanew = (*eta_mf[lev]).array(mfi);
566 Set::Patch<const Set::Scalar> eta = (*eta_old_mf[lev]).array(mfi);
567 // Diagnostic fields
569
570 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
571 {
572 auto sten = Numeric::GetStencil(i, j, k, bx);
573 Set::Vector grad_eta = Numeric::Gradient(eta, i, j, k, 0, DX);
574 Set::Vector grad_temp = Numeric::Gradient(temp, i, j, k, 0, DX);
575 Set::Scalar lap_temp = Numeric::Laplacian(temp, i, j, k, 0, DX);
576 Set::Scalar grad_eta_mag = grad_eta.lpNorm<2>();
577 Set::Vector grad_alpha = Numeric::Gradient(alpha, i, j, k, 0, DX, sten);
578 Set::Scalar dTdt = 0.0;
579 dTdt += grad_eta.dot(grad_temp * alpha(i, j, k));
580 dTdt += grad_alpha.dot(eta(i, j, k) * grad_temp);
581 dTdt += eta(i, j, k) * alpha(i, j, k) * lap_temp;
582 dTdt += alpha(i, j, k) * heatflux(i, j, k) * grad_eta_mag;
583
584 Set::Scalar Tsolid = dTdt + temps(i, j, k) * (etanew(i, j, k) - eta(i, j, k)) / dt;
585 temps(i, j, k) = temps(i, j, k) + dt * Tsolid;
586 tempnew(i, j, k) = etanew(i, j, k) * temps(i, j, k) + (1.0 - etanew(i, j, k)) * thermal.Tfluid;
587 });
588 }
589 }
590
591} //Function
592
593
594void Flame::TagCellsForRefinement(int lev, amrex::TagBoxArray& a_tags, Set::Scalar time, int ngrow)
595{
596 BL_PROFILE("Integrator::Flame::TagCellsForRefinement");
598
599 const Set::Scalar* DX = geom[lev].CellSize();
600 Set::Scalar dr = sqrt(AMREX_D_TERM(DX[0] * DX[0], +DX[1] * DX[1], +DX[2] * DX[2]));
601
602 // Eta criterion for refinement
603 for (amrex::MFIter mfi(*eta_mf[lev], true); mfi.isValid(); ++mfi)
604 {
605 const amrex::Box& bx = mfi.tilebox();
606 amrex::Array4<char> const& tags = a_tags.array(mfi);
609
610 if (thermal.on) {
611 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
612 {
613 Set::Vector gradeta = Numeric::Gradient(eta, i, j, k, 0, DX);
614 if (gradeta.lpNorm<2>() * dr * 2 > m_refinement_criterion && eta(i, j, k) >= t_refinement_restriction && temp(i,j,k) > thermal.Tcutoff*0.9)
615 tags(i, j, k) = amrex::TagBox::SET;
616 });
617
618 } else {
619 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
620 {
621 Set::Vector gradeta = Numeric::Gradient(eta, i, j, k, 0, DX);
622 if (gradeta.lpNorm<2>() * dr * 2 > m_refinement_criterion && eta(i, j, k) >= t_refinement_restriction)
623 tags(i, j, k) = amrex::TagBox::SET;
624 });
625 }
626 }
627
628 // Phi criterion for refinement
629 if (elastic.phirefinement) {
630 for (amrex::MFIter mfi(*eta_mf[lev], true); mfi.isValid(); ++mfi)
631 {
632 const amrex::Box& bx = mfi.tilebox();
633 amrex::Array4<char> const& tags = a_tags.array(mfi);
635
636 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
637 {
638 Set::Vector gradphi = Numeric::Gradient(phi, i, j, k, 0, DX);
639 if (gradphi.lpNorm<2>() * dr >= phi_refinement_criterion)
640 tags(i, j, k) = amrex::TagBox::SET;
641 });
642 }
643 }
644
645
646 // Thermal criterion for refinement
647 if (thermal.on) {
648 for (amrex::MFIter mfi(*temp_mf[lev], true); mfi.isValid(); ++mfi)
649 {
650 const amrex::Box& bx = mfi.tilebox();
651 amrex::Array4<char> const& tags = a_tags.array(mfi);
654 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
655 {
656 Set::Vector tempgrad = Numeric::Gradient(temp, i, j, k, 0, DX);
657 if (tempgrad.lpNorm<2>() * dr > t_refinement_criterion && eta(i, j, k) >= t_refinement_restriction)
658 tags(i, j, k) = amrex::TagBox::SET;
659 });
660 }
661 }
662
663 // Refine at start
664 for (amrex::MFIter mfi(*eta_mf[lev], true); mfi.isValid(); ++mfi)
665 {
666 const amrex::Box& bx = mfi.tilebox();
667 amrex::Array4<char> const& tags = a_tags.array(mfi);
670
671 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
672 {
673 Set::Vector gradeta = Numeric::Gradient(eta, i, j, k, 0, DX);
674 Set::Vector gradphi = Numeric::Gradient(phi, i, j, k, 0, DX);
675 if ((gradeta.lpNorm<2>() * dr * 2 > m_refinement_criterion || gradphi.lpNorm<2>() * dr >= thermal.phi_refinement_criterion_inital) && time < thermal.end_initial_refine_time)
676 tags(i, j, k) = amrex::TagBox::SET;
677 });
678 }
679}
680
681void Flame::Regrid(int lev, Set::Scalar time)
682{
683 BL_PROFILE("Integrator::Flame::Regrid");
684
685 ic_phi->Initialize(lev, phi_mf, time);
686 ic_eta->Initialize(lev, eta_0_mf, time);
687
688 if (thermal.on) {
689 /*
690 This regrid function works by using the "has_exceeded_Tcutoff" field. If the temperature in a cell is greater than Tcutoff,
691 eta will change and when regridding won't use the initial eta field. If T < T_cutoff, when regriding happens it applies the inital
692 eta field condition. This gives at leat a 4x speed improvement in 2D when doing regression with voids. This is because orgionally
693 there was a bug where when regridding, the orgional eta field wouldn't be applied, so there would be "squares" of voids instead of
694 circles/spheres when using .xyzr files as the inital condition.
695 */
696 for (amrex::MFIter mfi(*eta_mf[lev], true); mfi.isValid(); ++mfi)
697 {
698 const amrex::Box &bx = mfi.tilebox();
699 Set::Patch<Set::Scalar> eta = eta_mf.Patch(lev, mfi);
702 Set::Patch<Set::Scalar> exceeded_Tcutoff = thermal.has_exceeded_Tcutoff.Patch(lev, mfi);
703
704 Set::Scalar Tcutoff = thermal.Tcutoff;
705
706 amrex::BoxList boxes_to_update;
707 if (lev == finest_level && prev_finest_level == finest_level)
708 boxes_to_update = amrex::complementIn(bx, prev_finest_ba).boxList();
709 else
710 boxes_to_update.push_back(bx);
711
712 for (const amrex::Box &box : boxes_to_update)
713 amrex::ParallelFor(box, [=] AMREX_GPU_DEVICE(int i, int j, int k)
714 {
715
716 if (!exceeded_Tcutoff(i,j,k) && temp(i,j,k) < Tcutoff)
717 {
718 eta(i, j, k) = eta_0(i, j, k);
719 }
720 });
721 }
722
723 if (lev == finest_level)
724 {
725 prev_finest_ba = grids[finest_level];
726 prev_finest_level = finest_level;
727 }
728 }
729}
730
731void Flame::Integrate(int amrlev, Set::Scalar time, int /*step*/,
732 const amrex::MFIter& mfi, const amrex::Box& box)
733{
734 BL_PROFILE("Flame::Integrate");
735
737
738 const Set::Scalar* DX = geom[amrlev].CellSize();
739 Set::Scalar dv = AMREX_D_TERM(DX[0], *DX[1], *DX[2]);
742 if (variable_pressure) {
743 amrex::ParallelFor(box, [=] AMREX_GPU_DEVICE(int i, int j, int k)
744 {
745 chamber.volume += eta(i, j, k, 0) * dv;
746 Set::Vector grad = Numeric::Gradient(eta, i, j, k, 0, DX);
747 Set::Scalar normgrad = grad.lpNorm<2>();
748 Set::Scalar da = normgrad * dv;
749 chamber.area += da;
750
751 Set::Vector mgrad = Numeric::Gradient(mdot, i, j, k, 0, DX);
752 Set::Scalar mnormgrad = mgrad.lpNorm<2>();
753 Set::Scalar dm = mnormgrad * dv;
754 chamber.massflux += dm;
755
756 });
757 }
758 else {
759 amrex::ParallelFor(box, [=] AMREX_GPU_DEVICE(int i, int j, int k)
760 {
761 chamber.volume += eta(i, j, k, 0) * dv;
762 Set::Vector grad = Numeric::Gradient(eta, i, j, k, 0, DX);
763 Set::Scalar normgrad = grad.lpNorm<2>();
764 Set::Scalar da = normgrad * dv;
765 chamber.area += da;
766 });
767 }
768 // time dependent pressure data from experimenta -> p = 0.0954521220950523 * exp(15.289993148880678 * t)
769}
770} // namespace Integrator
771
#define pp_query_default(...)
Definition ParmParse.H:102
#define pp_queryclass(...)
Definition ParmParse.H:109
#define INFO
Definition Util.H:24
Definition BMP.H:22
void Initialize(const int &a_lev, Set::Field< T > &a_field, Set::Scalar a_time=0.0)
Definition IC.H:39
Initialize Laminates in a matrix.
Definition Laminate.H:16
Definition PNG.H:26
void forbid(std::string name, std::string explanation)
Definition ParmParse.H:160
int AnyUnusedInputs(bool inscopeonly=true, bool verbose=false)
Definition ParmParse.H:895
void select(std::string name, PTRTYPE *&ic_eta, Args &&... args)
Definition ParmParse.H:1056
void queryclass(std::string name, T *value)
Definition ParmParse.H:960
void select_default(std::string name, PTRTYPE *&ic_eta, Args &&... args)
Definition ParmParse.H:1095
static int AllUnusedInputs()
Definition ParmParse.H:933
int query_default(std::string name, T &value, T defaultvalue)
Definition ParmParse.H:293
virtual void TimeStepBegin(Set::Scalar a_time, int a_step) override
Definition Mechanics.H:170
void Integrate(int amrlev, Set::Scalar, int, const amrex::MFIter &mfi, const amrex::Box &a_box) override
Definition Mechanics.H:391
void Initialize(int lev) override
Use the #ic object to initialize::Temp.
Definition Mechanics.H:150
void Advance(int lev, Set::Scalar time, Set::Scalar dt) override
Definition Mechanics.H:251
Solver::Nonlocal::Newton< MODEL > solver
Definition Mechanics.H:506
void TagCellsForRefinement(int lev, amrex::TagBoxArray &a_tags, Set::Scalar, int) override
Definition Mechanics.H:450
Set::Field< Model::Solid::Finite::NeoHookeanPredeformed > model_mf
Definition Mechanics.H:474
Set::Scalar Tcutoff
Definition Flame.H:108
Model::Propellant::Propellant< Model::Propellant::PowerLaw, Model::Propellant::FullFeedback, Model::Propellant::Homogenize > propellant
Definition Flame.H:141
Set::Scalar volume
Definition Flame.H:129
Set::Scalar Telastic
Definition Flame.H:118
IC::IC< Set::Scalar > * ic_eta
Definition Flame.H:85
Set::Field< Set::Scalar > laser_mf
Definition Flame.H:73
Set::Field< Set::Scalar > alpha_mf
Definition Flame.H:70
amrex::BoxArray prev_finest_ba
Definition Flame.H:89
int prev_finest_level
Definition Flame.H:90
BC::BC< Set::Scalar > * bc_temp
Definition Flame.H:75
void UpdateModel(int a_step, Set::Scalar a_time) override
Definition Flame.cpp:304
Set::Scalar lambda
Definition Flame.H:96
Set::Scalar phi_refinement_criterion_inital
Definition Flame.H:110
Set::Scalar w1
Definition Flame.H:98
Set::Field< Set::Scalar > eta_0_mf
Definition Flame.H:65
Set::Field< Set::Scalar > phi_mf
Definition Flame.H:68
Set::Field< Set::Scalar > temp_old_mf
Definition Flame.H:60
struct Integrator::Flame::@2 pf
Set::Scalar small
Definition Flame.H:84
IC::IC< Set::Scalar > * ic_temp
Definition Flame.H:112
int variable_pressure
Definition Flame.H:87
Set::Scalar area
Definition Flame.H:130
void Regrid(int lev, Set::Scalar time) override
Definition Flame.cpp:681
Set::Scalar pressure
Definition Flame.H:133
model_type model_ap
Definition Flame.H:119
Set::Scalar w0
Definition Flame.H:98
void TimeStepComplete(Set::Scalar a_time, int a_iter) override
Definition Flame.cpp:408
static void Forbids(IO::ParmParse &pp)
Definition Flame.cpp:33
Set::Field< Set::Scalar > eta_mf
Definition Flame.H:63
Set::Scalar phi_refinement_criterion
Definition Flame.H:80
void TimeStepBegin(Set::Scalar a_time, int a_iter) override
Definition Flame.cpp:386
struct Integrator::Flame::@5 chamber
static void Parse(Flame &value, IO::ParmParse &pp)
Definition Flame.cpp:89
void Initialize(int lev) override
Definition Flame.cpp:270
Set::Scalar massflux
Definition Flame.H:131
Set::Scalar kappa
Definition Flame.H:97
Set::Scalar end_initial_refine_time
Definition Flame.H:109
Set::Scalar traction
Definition Flame.H:120
Set::Scalar Tref
Definition Flame.H:106
Set::Field< Set::Scalar > temps_mf
Definition Flame.H:61
void Advance(int lev, Set::Scalar time, Set::Scalar dt) override
Definition Flame.cpp:420
Set::Scalar t_refinement_restriction
Definition Flame.H:83
Set::Scalar t_refinement_criterion
Definition Flame.H:82
struct Integrator::Flame::@3 thermal
Set::Scalar etacutoff
Definition Flame.H:122
Set::Field< Set::Scalar > eta_old_mf
Definition Flame.H:64
Set::Scalar m_refinement_criterion
Definition Flame.H:81
Set::Scalar apply_chamber_pressure
Definition Flame.H:123
Set::Field< Set::Scalar > has_exceeded_Tcutoff
Definition Flame.H:111
Set::Scalar hc
Definition Flame.H:105
void Integrate(int amrlev, Set::Scalar time, int step, const amrex::MFIter &mfi, const amrex::Box &box) override
Definition Flame.cpp:731
model_type model_htpb
Definition Flame.H:119
Set::Scalar Tfluid
Definition Flame.H:107
BC::BC< Set::Scalar > * bc_eta
Definition Flame.H:76
int end_initial_refine
Definition Flame.H:91
IC::IC< Set::Scalar > * ic_laser
Definition Flame.H:78
void TagCellsForRefinement(int lev, amrex::TagBoxArray &tags, amrex::Real, int) override
Definition Flame.cpp:594
struct Integrator::Flame::@4 elastic
Set::Field< Set::Scalar > temp_mf
Definition Flame.H:59
Set::Scalar w12
Definition Flame.H:98
Set::Field< Set::Scalar > heatflux_mf
Definition Flame.H:71
bool plot_field
Definition Flame.H:86
Set::Field< Set::Scalar > mdot_mf
Definition Flame.H:66
IC::IC< Set::Scalar > * ic_phi
Definition Flame.H:77
Set::Scalar eps
Definition Flame.H:95
amrex::Real timestep
Timestep for the base level of refinement.
Definition Integrator.H:393
void RegisterNodalFab(Set::Field< Set::Scalar > &new_fab, int ncomp, int nghost, std::string name, bool writeout, bool evolving=true, std::vector< std::string > suffix={})
Add a new node-based scalar field.
void RegisterNewFab(Set::Field< Set::Scalar > &new_fab, BC::BC< Set::Scalar > *new_bc, int ncomp, int nghost, std::string name, bool writeout, bool evolving=true, std::vector< std::string > suffix={})
Add a new cell-based scalar field.
std::vector< amrex::Box > box
Definition Integrator.H:465
amrex::Vector< amrex::Real > dt
Timesteps for each level of refinement.
Definition Integrator.H:394
void RegisterIntegratedVariable(Set::Scalar *integrated_variable, std::string name, bool extensive=true)
Register a variable to be integrated over the spatial domain using the Integrate function.
AMREX_FORCE_INLINE AMREX_GPU_HOST_DEVICE Set::Scalar get_cp(const Set::Scalar phi)
Definition Propellant.H:78
AMREX_FORCE_INLINE AMREX_GPU_HOST_DEVICE Set::Scalar get_qdot(const Set::Scalar mdot, const Set::Scalar phi)
Definition Propellant.H:91
AMREX_FORCE_INLINE AMREX_GPU_HOST_DEVICE void set_pressure(Set::Scalar P)
Definition Propellant.H:40
AMREX_FORCE_INLINE AMREX_GPU_HOST_DEVICE Set::Scalar get_L(const Set::Scalar phi, const Set::Scalar T)
Definition Propellant.H:104
AMREX_FORCE_INLINE AMREX_GPU_HOST_DEVICE Set::Scalar get_rho(const Set::Scalar phi)
Definition Propellant.H:65
AMREX_FORCE_INLINE AMREX_GPU_HOST_DEVICE Set::Scalar get_K(const Set::Scalar phi)
Definition Propellant.H:52
amrex::Array4< Set::Scalar > Patch(const int lev, const amrex::MFIter &mfi) const &
Definition Set.H:263
amrex::Array4< T > Patch(int lev, amrex::MFIter &mfi) const &
Definition Set.H:76
void setPsi(Set::Field< Set::Scalar > &a_psi)
Definition Newton.H:46
Collection of numerical integrator objects.
Definition AllenCahn.H:43
AMREX_FORCE_INLINE Set::Scalar Laplacian(const amrex::Array4< const Set::Scalar > &f, const int &i, const int &j, const int &k, const int &m, const Set::Scalar dx[AMREX_SPACEDIM], std::array< StencilType, AMREX_SPACEDIM > &stencil=DefaultType)
Definition Stencil.H:555
static AMREX_FORCE_INLINE std::array< StencilType, AMREX_SPACEDIM > GetStencil(const int i, const int j, const int k, const amrex::Box domain)
Definition Stencil.H:45
AMREX_FORCE_INLINE Set::Vector CellGradientOnNode(const amrex::Array4< const Set::Scalar > &f, const int &i, const int &j, const int &k, const int &m, const Set::Scalar dx[AMREX_SPACEDIM])
Definition Stencil.H:699
AMREX_FORCE_INLINE Set::Vector Gradient(const amrex::Array4< const Set::Scalar > &f, const int &i, const int &j, const int &k, const int &m, const Set::Scalar dx[AMREX_SPACEDIM], std::array< StencilType, AMREX_SPACEDIM > stencil=DefaultType)
Definition Stencil.H:681
amrex::Real Scalar
Definition Base.H:18
Eigen::Matrix< amrex::Real, AMREX_SPACEDIM, 1 > Vector
Definition Base.H:19
void Warning(std::string file, std::string func, int line, Args const &... args)
Definition Util.H:202
void Message(std::string file, std::string func, int line, Args const &... args)
Definition Util.H:129
void Exception(std::string file, std::string func, int line, Args const &... args)
Definition Util.H:226
AMREX_FORCE_INLINE void RealFillBoundary(amrex::FabArray< amrex::BaseFab< T > > &a_mf, const amrex::Geometry &, const int nghost=2)
Definition Util.H:339
static AMREX_FORCE_INLINE T NodeToCellAverage(const amrex::Array4< const T > &f, const int &i, const int &j, const int &k, const int &m)
Definition Stencil.H:1428
static AMREX_FORCE_INLINE T CellToNodeAverage(const amrex::Array4< const T > &f, const int &i, const int &j, const int &k, const int &m, std::array< StencilType, AMREX_SPACEDIM > stencil=DefaultType)
Definition Stencil.H:1390
static Unit Area()
Definition Unit.H:209
static Unit Energy()
Definition Unit.H:218
static Unit Time()
Definition Unit.H:199
static Unit Temperature()
Definition Unit.H:201
static Unit Pressure()
Definition Unit.H:217
static Unit Power()
Definition Unit.H:219
static Unit Length()
Definition Unit.H:198
static Unit Less()
Definition Unit.H:197