LCOV - code coverage report
Current view: top level - src/Integrator - TopOp.H (source / functions) Hit Total Coverage
Test: coverage_merged.info Lines: 98 103 95.1 %
Date: 2025-01-16 18:33:59 Functions: 10 10 100.0 %

          Line data    Source code
       1             : #ifndef INTEGRATOR_TOPOP_H
       2             : #define INTEGRATOR_TOPOP_H
       3             : #include <iostream>
       4             : #include <fstream>
       5             : #include <iomanip>
       6             : #include <numeric>
       7             : 
       8             : #include "AMReX.H"
       9             : #include "AMReX_ParallelDescriptor.H"
      10             : #include "AMReX_ParmParse.H"
      11             : 
      12             : #include "IO/ParmParse.H"
      13             : #include "Integrator/Base/Mechanics.H"
      14             : 
      15             : 
      16             : #include "IC/IC.H"
      17             : #include "BC/BC.H"
      18             : #include "BC/Operator/Elastic/Constant.H"
      19             : #include "BC/Operator/Elastic/TensionTest.H"
      20             : #include "BC/Operator/Elastic/Expression.H"
      21             : 
      22             : #include "IC/Ellipse.H"
      23             : #include "IC/Voronoi.H"
      24             : #include "IC/Constant.H"
      25             : #include "IC/BMP.H"
      26             : #include "BC/Constant.H"
      27             : #include "Numeric/Stencil.H"
      28             : 
      29             : #include "Model/Solid/Solid.H"
      30             : #include "Solver/Nonlocal/Linear.H"
      31             : #include "Solver/Nonlocal/Newton.H"
      32             : 
      33             : #include "Operator/Operator.H"
      34             : 
      35             : 
      36             : namespace Integrator
      37             : {
      38             : template<class MODEL>
      39             : class TopOp: virtual public Base::Mechanics<MODEL>
      40             : {
      41             : public:
      42             : 
      43             :     TopOp(): Base::Mechanics<MODEL>() {}
      44           1 :     TopOp(IO::ParmParse& pp): Base::Mechanics<MODEL>()
      45             :     {
      46           1 :         Parse(*this, pp);
      47           1 :     }
      48             : 
      49             :     // The mechanics integrator manages the solution of an elastic 
      50             :     // solve using the MLMG solver. 
      51           1 :     static void Parse(TopOp& value, IO::ParmParse& pp)
      52             :     {
      53           1 :         Base::Mechanics<MODEL>::Parse(value, pp);
      54             : 
      55           1 :         pp_queryclass("model", value.model);
      56             :         // Read in IC for psi
      57           1 :         if (pp.contains("psi.ic.type"))
      58             :         {
      59           1 :             std::string type;
      60           1 :             pp_query_validate("psi.ic.type", type, {"ellipse", "constant"}); // Read IC type for the eta field
      61           1 :             if (type == "ellipse") value.ic_psi = new IC::Ellipse(value.geom, pp, "psi.ic.ellipse");
      62           1 :             else if (type == "constant") value.ic_psi = new IC::Constant(value.geom, pp, "psi.ic.constant");
      63           0 :             else Util::Abort(INFO, "Invalid value for psi.ic.type: ", type);
      64             : 
      65           1 :             value.bc_psi = new BC::Constant(1, pp, "psi.bc");
      66           1 :             value.RegisterNewFab(value.psi_mf, value.bc_psi, 1, 2, "psi", true);
      67           1 :             value.RegisterNewFab(value.psi_old_mf, value.bc_psi, 1, 2, "psiold", false);
      68           1 :             value.psi_on = true;
      69             :         }
      70           1 :         pp_query_default("eta_ref_threshold", value.m_eta_ref_threshold, 0.01); // Refinement threshold based on eta
      71           1 :         pp_query_default("alpha", value.alpha, 1.0); // :math:`\alpha` parameter
      72           1 :         pp_query_default("beta", value.beta, 1.0); // :math:`\beta` parameter
      73           1 :         pp_query_default("gamma", value.gamma, 1.0); // :math:`\gamma` parameter
      74           1 :         pp_queryclass("L", value.L); // Mobility
      75           1 :         if (pp.contains("volume0frac"))
      76             :         {
      77           0 :             if (pp.contains("volume0")) Util::Abort(INFO, "Cannot specify volume0frac and volume0");
      78             :             Set::Scalar volumefrac;
      79           0 :             pp_query_default("volume0frac", volumefrac, 0.5); // Prescribed volume fraction
      80           0 :             value.volume0 = volumefrac *
      81           0 :                 AMREX_D_TERM((value.geom[0].ProbHi()[0] - value.geom[0].ProbLo()[0]),
      82             :                     *(value.geom[0].ProbHi()[1] - value.geom[0].ProbLo()[1]),
      83             :                     *(value.geom[0].ProbHi()[2] - value.geom[0].ProbLo()[2]));
      84             :         }
      85             :         else
      86           1 :             pp_query_default("volume0", value.volume0, 0.5); // Prescribed total vlume
      87             : 
      88           1 :         pp_queryclass("lambda", value.lambda); // Lagrange multiplier (can be interplated)
      89             : 
      90             : 
      91           1 :         value.RegisterIntegratedVariable(&value.volume, "volume");
      92           1 :         value.RegisterIntegratedVariable(&value.w_chem_potential, "chem_potential");
      93           1 :         value.RegisterIntegratedVariable(&value.w_bndry, "bndry");
      94           1 :         value.RegisterIntegratedVariable(&value.w_elastic, "elastic");
      95           1 :     }
      96             : 
      97           1 :     void Initialize(int lev) override
      98             :     {
      99           1 :         Base::Mechanics<MODEL>::Initialize(lev);
     100           1 :         if (psi_on) ic_psi->Initialize(lev, psi_mf);
     101           1 :         if (psi_on) ic_psi->Initialize(lev, psi_old_mf);
     102           1 :     }
     103             : 
     104           2 :     virtual void UpdateModel(int a_step, Set::Scalar /*a_time*/) override
     105             :     {
     106           2 :         if (m_type == Base::Mechanics<MODEL>::Type::Disable) return;
     107             : 
     108           2 :         if (a_step > 0) return;
     109             : 
     110           2 :         for (int lev = 0; lev <= finest_level; ++lev)
     111             :         {
     112           1 :             model_mf[lev]->setVal(model);
     113           1 :             Util::RealFillBoundary(*model_mf[lev], geom[lev]);
     114           1 :             Util::RealFillBoundary(*psi_mf[lev], geom[lev]);
     115           1 :             Util::RealFillBoundary(*psi_old_mf[lev], geom[lev]);
     116             :         }
     117             : 
     118             :     }
     119             : 
     120             : 
     121           6 :     void Advance(int lev, Set::Scalar time, Set::Scalar dt) override
     122             :     {
     123             :         BL_PROFILE("TopOp::Advance");
     124           6 :         Base::Mechanics<MODEL>::Advance(lev, time, dt);
     125           6 :         std::swap(psi_old_mf[lev], psi_mf[lev]);
     126           6 :         const Set::Scalar* DX = geom[lev].CellSize();
     127           6 :         amrex::Box domain = geom[lev].Domain();
     128             : 
     129           6 :         Set::Scalar Lnow = L(time);
     130           6 :         Set::Scalar lambdaT = lambda(time);
     131             : 
     132          22 :         for (amrex::MFIter mfi(*psi_mf[lev], amrex::TilingIfNotGPU()); mfi.isValid(); ++mfi)
     133             :         {
     134          16 :             amrex::Box bx = mfi.tilebox();
     135          16 :             bx.grow(1);
     136          16 :             bx = bx & domain;
     137          16 :             amrex::Array4<const Set::Matrix> const& sig = (*stress_mf[lev]).array(mfi);
     138          16 :             amrex::Array4<const Set::Matrix> const& eps = (*strain_mf[lev]).array(mfi);
     139          16 :             amrex::Array4<const Set::Scalar> const& psi = (*psi_old_mf[lev]).array(mfi);
     140          16 :             amrex::Array4<Set::Scalar> const& psinew = (*psi_mf[lev]).array(mfi);
     141             : 
     142        9932 :             amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
     143             :             {
     144        9916 :                 Set::Scalar driving_force = 0.0;
     145             : 
     146       39664 :                 driving_force += alpha * 2.0 * psi(i, j, k) * (2.0 * psi(i, j, k) * psi(i, j, k) - 3.0 * psi(i, j, k) + 1.0);
     147        9916 :                 driving_force += -beta * Numeric::Laplacian(psi, i, j, k, 0, DX);
     148             : 
     149        9916 :                 Set::Matrix sig_avg = Numeric::Interpolate::NodeToCellAverage(sig, i, j, k, 0);
     150        9916 :                 Set::Matrix eps_avg = Numeric::Interpolate::NodeToCellAverage(eps, i, j, k, 0);
     151             : 
     152        9916 :                 driving_force += -gamma * 0.5 * (sig_avg.transpose() * eps_avg).trace() * psi(i, j, k);
     153             : 
     154        9916 :                 driving_force += lambdaT * (volume - volume0);
     155             : 
     156       19832 :                 psinew(i, j, k) = psi(i, j, k) - Lnow * dt * driving_force;
     157       19832 :                 if (psinew(i, j, k) < 0.0) psinew(i, j, k) = 0.0;
     158       19832 :                 if (psinew(i, j, k) > 1.0) psinew(i, j, k) = 1.0;
     159             :             });
     160             :         }
     161           6 :     }
     162             : 
     163           2 :     void Integrate(int amrlev, Set::Scalar time, int step,
     164             :         const amrex::MFIter& mfi, const amrex::Box& box) override
     165             :     {
     166             :         BL_PROFILE("TopOp::Integrate");
     167           2 :         Base::Mechanics<MODEL>::Integrate(amrlev, time, step, mfi, box);
     168             : 
     169           2 :         const amrex::Real* DX = geom[amrlev].CellSize();
     170           2 :         Set::Scalar dv = AMREX_D_TERM(DX[0], *DX[1], *DX[2]);
     171             : 
     172           2 :         amrex::Array4<amrex::Real> const& psi = (*psi_mf[amrlev]).array(mfi);
     173           2 :         amrex::Array4<const Set::Matrix> const& sig = (*stress_mf[amrlev]).array(mfi);
     174           2 :         amrex::Array4<const Set::Matrix> const& eps = (*strain_mf[amrlev]).array(mfi);
     175           2 :         amrex::ParallelFor(box, [=] AMREX_GPU_DEVICE(int i, int j, int k)
     176             :         {
     177        4096 :             volume += psi(i, j, k, 0) * dv;
     178       16384 :             w_chem_potential += alpha * psi(i, j, k, 0) * psi(i, j, k, 0) * (1. - psi(i, j, k, 0) * psi(i, j, k, 0)) * dv;
     179        8192 :             w_bndry += beta * 0.5 * Numeric::Gradient(psi, i, j, k, 0, DX).squaredNorm() * dv;
     180             : 
     181        4096 :             Set::Matrix sig_avg = Numeric::Interpolate::NodeToCellAverage(sig, i, j, k, 0);
     182        4096 :             Set::Matrix eps_avg = Numeric::Interpolate::NodeToCellAverage(eps, i, j, k, 0);
     183        8192 :             w_elastic += gamma * 0.5 * (sig_avg.transpose() * eps_avg).trace() * psi(i, j, k) * dv;
     184             :         });
     185           2 :     }
     186             : 
     187             : 
     188           4 :     void TagCellsForRefinement(int lev, amrex::TagBoxArray& a_tags, Set::Scalar a_time, int a_ngrow) override
     189             :     {
     190           4 :         if (m_type == Base::Mechanics<MODEL>::Type::Disable) return;
     191           4 :         Base::Mechanics<MODEL>::TagCellsForRefinement(lev, a_tags, a_time, a_ngrow);
     192             : 
     193           4 :         Set::Vector DX(geom[lev].CellSize());
     194           4 :         Set::Scalar DXnorm = DX.lpNorm<2>();
     195          10 :         for (amrex::MFIter mfi(*model_mf[lev], TilingIfNotGPU()); mfi.isValid(); ++mfi)
     196             :         {
     197           6 :             amrex::Box bx = mfi.tilebox();
     198           6 :             amrex::Array4<char> const& tags = a_tags.array(mfi);
     199           6 :             if (psi_on)
     200             :             {
     201           6 :                 amrex::Array4<Set::Scalar> const& psi = psi_mf[lev]->array(mfi);
     202        7416 :                 amrex::ParallelFor(bx, [=] AMREX_GPU_DEVICE(int i, int j, int k)
     203             :                 {
     204        7410 :                     auto sten = Numeric::GetStencil(i, j, k, bx);
     205             :                     {
     206        7410 :                         Set::Vector gradpsi = Numeric::Gradient(psi, i, j, k, 0, DX.data(), sten);
     207        7410 :                         if (gradpsi.lpNorm<2>() * DXnorm > m_eta_ref_threshold)
     208        1230 :                             tags(i, j, k) = amrex::TagBox::SET;
     209             :                     }
     210             :                 });
     211             :             }
     212             :         }
     213             :     }
     214             : 
     215             : 
     216             : protected:
     217             :     MODEL model;
     218             :     IC::IC* ic_psi = nullptr;
     219             :     BC::BC<Set::Scalar>* bc_psi = nullptr;
     220             :     Set::Scalar m_eta_ref_threshold = NAN;
     221             :     Set::Field<Set::Scalar> psi_old_mf;
     222             : 
     223             : 
     224             :     using Base::Mechanics<MODEL>::m_type;
     225             :     using Base::Mechanics<MODEL>::finest_level;
     226             :     using Base::Mechanics<MODEL>::geom;
     227             :     using Base::Mechanics<MODEL>::model_mf;
     228             :     using Base::Mechanics<MODEL>::psi_mf;
     229             :     using Base::Mechanics<MODEL>::psi_on;
     230             :     using Base::Mechanics<MODEL>::stress_mf;
     231             :     using Base::Mechanics<MODEL>::strain_mf;
     232             : 
     233             : 
     234             :     Set::Scalar alpha = NAN;
     235             :     Set::Scalar beta = NAN;
     236             :     Set::Scalar gamma = NAN;
     237             :     //Set::Scalar L = 1.0;
     238             :     Numeric::Interpolator::Linear<Set::Scalar> L;
     239             :     Set::Scalar volume0 = NAN;
     240             :     //Set::Scalar lambda = 1.0;
     241             :     Numeric::Interpolator::Linear<Set::Scalar> lambda;
     242             : 
     243             :     Set::Scalar volume = NAN;
     244             :     Set::Scalar w_chem_potential = NAN;
     245             :     Set::Scalar w_bndry = NAN;
     246             :     Set::Scalar w_elastic = NAN;
     247             : };
     248             : 
     249             : 
     250             : 
     251             : 
     252             : 
     253             : 
     254             : 
     255             : 
     256             : 
     257             : 
     258             : } // namespace Integrator
     259             : #endif

Generated by: LCOV version 1.14