LCOV - code coverage report
Current view: top level - src/Integrator - PhaseFieldMicrostructure.H (source / functions) Coverage Total Hit
Test: coverage_merged.info Lines: 61.9 % 139 86
Test Date: 2025-04-03 04:02:21 Functions: 14.8 % 27 4

            Line data    Source code
       1              : ///
       2              : /// \file PhaseFieldMicrostructure.H
       3              : ///
       4              : #ifndef INTEGRATOR_PHASEFIELDMICROSTRUCTURE_H
       5              : #define INTEGRATOR_PHASEFIELDMICROSTRUCTURE_H
       6              : 
       7              : #include <random>
       8              : 
       9              : #include "AMReX_ParmParse.H"
      10              : #include <AMReX_MLMG.H>
      11              : 
      12              : #include "Integrator/Integrator.H"
      13              : #include "BC/BC.H"
      14              : #include "BC/Constant.H"
      15              : #include "IC/Constant.H"
      16              : #include "IC/PerturbedInterface.H"
      17              : #include "IC/Voronoi.H"
      18              : #include "IC/Sphere.H"
      19              : #include "IC/Expression.H"
      20              : #include "IC/Ellipse.H"
      21              : #include "IC/Random.H"
      22              : #include "Model/Interface/GB/GB.H"
      23              : #include "Model/Interface/GB/Sin.H"
      24              : #include "Model/Interface/GB/AbsSin.H"
      25              : #include "Model/Interface/GB/Read.H"
      26              : #include "Model/Interface/GB/SH.H"
      27              : #include "Model/Defect/Disconnection.H"
      28              : #include "Base/Mechanics.H"
      29              : 
      30              : namespace Integrator
      31              : {
      32              : 
      33              : enum RegularizationType{
      34              :     Wilmore = 1,
      35              :     K12 = 2};
      36              : 
      37              : ///
      38              : /// Solve the Allen-Cahn evolution equation for microstructure with parameters \f$\eta_1\ldots\eta_n\f$,
      39              : /// where n corresponds to the number of grains.
      40              : ///
      41              : template<class model_type>
      42              : class PhaseFieldMicrostructure : public Base::Mechanics<model_type>
      43              : {
      44              : public:
      45              :     const std::string name = "phasefieldmicrostructure." + std::string(model_type::name);
      46              : 
      47            0 :     PhaseFieldMicrostructure() : Integrator() {};
      48            9 :     PhaseFieldMicrostructure(IO::ParmParse &pp) : Integrator() 
      49           12 :     {pp_queryclass(*this);}
      50            6 :     virtual ~PhaseFieldMicrostructure()
      51              :     {
      52            2 :         delete boundary;
      53            3 :         delete ic;
      54            3 :         delete mybc;
      55            9 :     }
      56            3 :     static void Parse(PhaseFieldMicrostructure &value, IO::ParmParse &pp)
      57              :     {
      58              :         BL_PROFILE("PhaseFieldMicrostructure::Parse");
      59              :         
      60              :         // Number of grain fields (may be more if using different IC)
      61           18 :         pp_query_default("pf.number_of_grains", value.number_of_grains,2);     
      62           18 :         pp_query_required("pf.M", value.pf.M);                         // Mobility
      63           18 :         pp_query_required("pf.gamma", value.pf.gamma);                 // Phase field :math:`\gamma`
      64           18 :         pp_query_required("pf.sigma0", value.pf.sigma0);               // Initial GB energy if not using  anisotropy
      65           18 :         pp_query_required("pf.l_gb", value.pf.l_gb);                   // Mobility
      66           18 :         pp_query_default("pf.elastic_df",value.pf.elastic_df,false);   // Determine whether to use elastic driving force
      67           18 :         pp_query_default("pf.elastic_mult",value.pf.elastic_mult,1.0); // Multiplier of elastic energy
      68              : 
      69           18 :         pp_query_default("pf.threshold.value",value.pf.threshold.value,0.0);          // Value used for thresholding kinetic relation
      70           18 :         pp_query_default("pf.threshold.chempot",value.pf.threshold.chempot,false);    // Whether to include chemical potential in threshold
      71           18 :         pp_query_default("pf.threshold.boundary",value.pf.threshold.boundary,false);  // Whether to include boundary energy in threshold
      72           18 :         pp_query_default("pf.threshold.corner",value.pf.threshold.corner,false);      // Whether to include corner regularization in threshold
      73           18 :         pp_query_default("pf.threshold.lagrange",value.pf.threshold.lagrange,false);  // Whether to include lagrange multiplier in threshold
      74           15 :         pp_query_default("pf.threshold.mechanics",value.pf.threshold.mechanics,false);// Whether to include mechanical driving force in threshold
      75              :         {
      76            3 :             value.pf.threshold.on =
      77            3 :                 value.pf.threshold.chempot || value.pf.threshold.boundary ||
      78            9 :                 value.pf.threshold.corner || value.pf.threshold.lagrange ||
      79            3 :                 value.pf.threshold.mechanics;
      80              : 
      81            6 :             std::string type = "continuous";
      82           21 :             pp_query_validate("pf.threshold.type",type,{"continuous","chop"}); // Type of thresholding to use
      83            3 :             if (type == "continuous") value.pf.threshold.type = ThresholdType::Continuous;
      84            0 :             else if (type == "chop") value.pf.threshold.type = ThresholdType::Chop;
      85            3 :         }
      86              :         
      87            3 :         value.pf.L = (4./3.)*value.pf.M / value.pf.l_gb;
      88              :     
      89           18 :         pp_query_required("amr.max_level", value.max_level);            // Maximum AMR level
      90           15 :         pp_query_default("amr.ref_threshold", value.ref_threshold, 0.1);    // Phase field refinement threshold
      91              : 
      92              : 
      93            3 :         std::string type_str;
      94              :         // Reading this is redundant but necessary because of the way the code was originally structured
      95              :         // (need to fix eventually)
      96           21 :         pp.query_validate("mechanics.type", type_str, {"disable","static","dynamic"}); 
      97            3 :         value.m_type = Base::Mechanics<model_type>::Disable; // Turn mechanics off by default
      98              : 
      99            3 :         if (type_str != "disable") // do this only if mechanics is activated
     100              :         {
     101              :             // Elasticity
     102            0 :             pp_query_default("mechanics.tstart",value.mechanics.tstart, 0.0);
     103              : 
     104              :             // Read in models - either one model for all grains, or
     105              :             // individual models, one for each grain.
     106            0 :             pp.queryclass_enumerate("mechanics.model",value.mechanics.model, value.number_of_grains);
     107              :         
     108              :             // Mixing order 
     109            0 :             pp_query_validate("mechanics.mix_order",value.mechanics.model_mix_order,{1,2});
     110              :             // Force Neumann BCs on the model
     111            0 :             pp_query_default("mechanics.model_neuman_boundary",value.mechanics.model_neumann_boundary,false);
     112              : 
     113            0 :             pp.queryclass<Base::Mechanics<model_type>>("mechanics",value);
     114            0 :             if (value.m_type == Base::Mechanics<model_type>::Type::Static)
     115            0 :                 value.number_of_ghost_cells = std::max(value.number_of_ghost_cells, 3);
     116              :         }
     117              : 
     118              : 
     119              :         // Lagrange multiplier method for enforcing volumes
     120           15 :         pp_query_default("lagrange.on", value.lagrange.on,false);
     121            3 :         if (value.lagrange.on)
     122              :         {
     123            6 :             pp_query_required("lagrange.lambda", value.lagrange.lambda);      // Lagrange multiplier value
     124            6 :             pp_query_required("lagrange.vol0", value.lagrange.vol0);          // Prescribed volume
     125            5 :             pp_query_default("lagrange.tstart", value.lagrange.tstart,0.0);   // Lagrange multipler start time
     126            1 :             value.SetThermoInt(1);
     127              :         }
     128              : 
     129           15 :         pp_query_default("sdf.on",value.sdf.on,false); // synthetic driving force (SDF)
     130            3 :         if(value.sdf.on)
     131              :         {
     132            0 :             std::vector<std::string> vals;
     133            0 :             pp_queryarr("sdf.val",vals);  // value of SDF for each grain
     134            0 :             int nvals = static_cast<int>(vals.size());
     135            0 :             if (nvals == 1)
     136            0 :                 for (int i = 0; i < value.number_of_grains; i++)
     137            0 :                     value.sdf.val.push_back(Numeric::Interpolator::Linear<Set::Scalar>(vals[0]));
     138            0 :             else if (nvals == value.number_of_grains)
     139            0 :                 for (int i = 0; i < value.number_of_grains; i++)
     140            0 :                     value.sdf.val.push_back(Numeric::Interpolator::Linear<Set::Scalar>(vals[i]));
     141              :             else
     142            0 :                 Util::Abort(INFO,"sdf.val received ", vals.size(), " but requires 1 or ", value.number_of_grains);
     143              : 
     144            0 :             pp_query_default("sdf.tstart",value.sdf.tstart,0.0); // time to begin applying SDF
     145            0 :         }
     146              : 
     147              :         // Anisotropic grain boundary energy parameters
     148              : 
     149           15 :         pp_query_default("anisotropy.on", value.anisotropy.on,false);              // Turn on
     150            3 :         if (value.anisotropy.on)
     151              :         {
     152              :             // Regularization para m
     153           12 :             pp_query_required("anisotropy.beta", value.anisotropy.beta);
     154              :             // Time to turn on anisotropy
     155           10 :             pp_query_required("anisotropy.tstart", value.anisotropy.tstart);
     156            2 :             value.anisotropy.timestep = value.timestep;
     157              :             // Modify timestep when turned on
     158           10 :             pp_query_required("anisotropy.timestep", value.anisotropy.timestep);
     159            2 :             value.anisotropy.plot_int = value.plot_int;
     160              :             // Modify plot_int when turned on
     161           10 :             pp_query_default("anisotropy.plot_int", value.anisotropy.plot_int, -1);
     162            2 :             value.anisotropy.plot_dt = value.plot_dt;
     163              :             // Modify plot_dt when turned on
     164           12 :             pp_query_default("anisotropy.plot_dt", value.anisotropy.plot_dt, -1.0);
     165              :             // Modify thermo int when turned on
     166           12 :             pp_query_default("anisotropy.thermo_int", value.anisotropy.thermo_int, -1);
     167              :             // Modify thermo plot int when turned on
     168           12 :             pp_query_default("anisotropy.thermo_plot_int", value.anisotropy.thermo_plot_int, -1);
     169              :             // Frequency of elastic calculation
     170           10 :             pp_query_default("anisotropy.elastic_int",value.anisotropy.elastic_int, -1);           
     171            2 :             if (value.anisotropy.on) 
     172            2 :                 value.number_of_ghost_cells = std::max(value.number_of_ghost_cells,2);
     173              : 
     174              :             // Determine the kind of regularization to use
     175            2 :             std::map<std::string, RegularizationType> regularization_type;
     176            4 :             regularization_type["wilmore"] = RegularizationType::Wilmore;
     177            2 :             regularization_type["k12"] = RegularizationType::K12;
     178              : 
     179            2 :             std::string regularization_type_input;
     180              :             // Type of regularization to use  
     181           14 :             pp_query_validate("anisotropy.regularization", regularization_type_input,{"k12","wilmore"});    
     182            2 :             value.regularization = regularization_type[regularization_type_input];
     183              : 
     184           14 :             pp_forbid("anisotropy.gb_type"," --> anisotropy.type");
     185              :             // Type of GB to use
     186            4 :             pp.select<Model::Interface::GB::AbsSin, Model::Interface::GB::Sin, Model::Interface::GB::Read, Model::Interface::GB::SH>("anisotropy",value.boundary); 
     187            2 :         }
     188              :         // Thermal fluctuations
     189            3 :         pp.query("fluctuation.on",value.fluctuation.on);
     190            3 :         if (value.fluctuation.on)
     191              :         {
     192            0 :             pp.query("fluctuation.amp",value.fluctuation.amp); // fluctuation amplitude
     193            0 :             pp.query("fluctuation.sd",value.fluctuation.sd); // fluctuation stadard deviation
     194            0 :             pp.query("fluctuation.tstart", value.fluctuation.tstart); // time to start applying fluctuation
     195            0 :             value.fluctuation.norm_dist = std::normal_distribution<double>(0.0,value.fluctuation.sd);
     196              :         }
     197              : 
     198              :         // Disconnection generation
     199            3 :         pp.query("disconnection.on",value.disconnection.on);
     200            3 :         if (value.disconnection.on)
     201              :         {
     202              :             // Read in nucleation parameters from disconnection class
     203            0 :             pp.queryclass<Model::Defect::Disconnection>("disconnection",value.disconnection.model);
     204              :         }
     205              : 
     206              :         // Shear coupling matrices
     207           15 :         pp_query_default("shearcouple.on",value.shearcouple.on,false);
     208            3 :         if (value.shearcouple.on)
     209              :         {
     210            0 :             Util::AssertException(INFO,TEST(value.m_time_evolving == true), " mechanics.time_evolving must be true when using shearcouple");
     211            0 :             Util::AssertException(INFO,TEST(value.mechanics.model_mix_order == 2), " mechanics.model.mix_order must be 2 when using shearcouple");
     212              :             
     213            0 :             value.shearcouple.Fgb.resize(value.number_of_grains * value.number_of_grains, Set::Matrix::Zero());
     214            0 :             for (int i = 0 ; i < value.number_of_grains ; i++)
     215            0 :                 for (int j = 0 ; j < value.number_of_grains ; j++)
     216              :                 {
     217            0 :                     std::string name    = "shearcouple.Fgb."+std::to_string(i)+"."+std::to_string(j);
     218            0 :                     std::string namerev = "shearcouple.Fgb."+std::to_string(j)+"."+std::to_string(i);
     219            0 :                     if ( i==j && pp.contains(name.data())) 
     220            0 :                         Util::Abort(INFO,"Cannot specify self FGB ", name);
     221            0 :                     if (pp.contains(name.data()) && pp.contains(namerev.data())) 
     222            0 :                         Util::Abort(INFO,"Cannot specify both ",name," and ",namerev);
     223            0 :                     if (pp.contains(name.data())) 
     224              :                     {
     225            0 :                         pp.queryarr(name.data(),value.shearcouple.Fgb[i*value.number_of_grains + j]);
     226            0 :                         value.shearcouple.Fgb[j*value.number_of_grains + i] = - value.shearcouple.Fgb[i*value.number_of_grains + j];
     227              :                     }
     228              :                 }
     229              :         }
     230              : 
     231              :         // Boundary condition for eta
     232            6 :         pp.select<BC::Constant>("bc.eta",value.mybc,value.number_of_grains);
     233              : 
     234              :         // Initial condition for the order parameter eta
     235            9 :         pp.select<IC::Constant,IC::PerturbedInterface,IC::Voronoi,IC::Expression,IC::Sphere,IC::Ellipse,IC::Random>("ic",value.ic,value.geom);
     236              : 
     237              :         // Anisotropic mobility
     238           15 :         pp_query_default("anisotropic_kinetics.on",value.anisotropic_kinetics.on, 0);
     239            3 :         if (value.anisotropic_kinetics.on)
     240              :         {
     241              :             // simulation time when anisotropic kinetics is activated
     242            0 :             pp_query_default("anisotropic_kinetics.tstart",value.anisotropic_kinetics.tstart, 0.0);
     243            0 :             std::string mobility_filename, threshold_filename;
     244              :             // file containing anisotropic mobility data
     245            0 :             pp_query_file("anisotropic_kinetics.mobility",mobility_filename); 
     246            0 :             value.anisotropic_kinetics.mobility = Numeric::Interpolator::Linear<Set::Scalar>::Read(mobility_filename);
     247              :             // file containing anisotropic mobility data
     248            0 :             pp_query_file("anisotropic_kinetics.threshold",threshold_filename); 
     249            0 :             value.anisotropic_kinetics.threshold = Numeric::Interpolator::Linear<Set::Scalar>::Read(threshold_filename);
     250            0 :             value.RegisterNewFab(value.anisotropic_kinetics.L_mf, value.mybc, value.number_of_grains, 0, "mobility",true);
     251            0 :             value.RegisterNewFab(value.anisotropic_kinetics.threshold_mf, value.mybc, value.number_of_grains, 0, "theshold",true);
     252            0 :         }
     253              : 
     254              : 
     255              : 
     256              : 
     257            9 :         value.RegisterNewFab(value.eta_mf, value.mybc, value.number_of_grains, value.number_of_ghost_cells, "Eta",true);
     258            9 :         value.RegisterNewFab(value.eta_old_mf, value.mybc, value.number_of_grains, value.number_of_ghost_cells, "EtaOld",false);
     259              :         //value.RegisterNewFab(value.driving_force_mf, value.mybc, value.number_of_grains, value.number_of_ghost_cells, "DrivingForce",false);
     260              :         // if (value.pf.threshold.on)
     261              :         //     value.RegisterNewFab(value.driving_force_threshold_mf, value.mybc, value.number_of_grains, value.number_of_ghost_cells, "DrivingForceThreshold",false);
     262              :         // if (value.disconnection.on)
     263              :         //     value.RegisterNewFab(value.disc_mf, new BC::Nothing(), 1, value.number_of_ghost_cells, "disc",true);  // see box
     264              : 
     265            6 :         value.RegisterIntegratedVariable(&value.volume, "volume");
     266            6 :         value.RegisterIntegratedVariable(&value.area, "area");
     267            6 :         value.RegisterIntegratedVariable(&value.gbenergy, "gbenergy");
     268            6 :         value.RegisterIntegratedVariable(&value.realgbenergy, "realgbenergy");
     269            6 :         value.RegisterIntegratedVariable(&value.regenergy, "regenergy");
     270              : 
     271            3 :     }
     272              : 
     273              :     
     274              : protected:
     275              : 
     276              :     /// \fn    Advance
     277              :     /// \brief Evolve phase field in time
     278              :     void Advance (int lev, Real time, Real dt) override;
     279              :     void Initialize (int lev) override;
     280              : 
     281              :     void TagCellsForRefinement (int lev, amrex::TagBoxArray& tags, amrex::Real time, int ngrow) override;
     282              : 
     283              :     virtual void TimeStepBegin(amrex::Real time, int iter) override;
     284              :     virtual void TimeStepComplete(amrex::Real time, int iter) override;
     285              :     void Integrate(int amrlev, Set::Scalar time, int step,
     286              :                     const amrex::MFIter &mfi, const amrex::Box &box) override;
     287              : 
     288              :     virtual void UpdateEigenstrain(int lev);
     289            0 :     virtual void UpdateEigenstrain()
     290              :     {
     291            0 :         for (int lev = 0; lev <= this->max_level; lev++)
     292            0 :             UpdateEigenstrain(lev);
     293            0 :     }
     294              : 
     295              :     virtual void UpdateModel(int /*a_step*/, Set::Scalar /*a_time*/) override;
     296              : 
     297              : private:
     298              : 
     299              :     int number_of_grains = -1;
     300              :     int number_of_ghost_cells = 1;
     301              :     Set::Scalar ref_threshold = 0.1;
     302              : 
     303              :     // Cell fab
     304              :     Set::Field<Set::Scalar> eta_mf; // Multicomponent field variable storing \t$\eta_i\t$ for the __current__ timestep
     305              :     Set::Field<Set::Scalar> eta_old_mf; 
     306              :     Set::Field<Set::Scalar> driving_force_mf;
     307              :     Set::Field<Set::Scalar> driving_force_threshold_mf;
     308              :     Set::Field<Set::Scalar> fluct_mf;
     309              :     //Set::Field<Set::Scalar> disc_mf; //see box
     310              :     // Node fab
     311              :     //Set::Field<Set::Scalar> elasticdf_mf;
     312              :     Set::Field<Set::Scalar> totaldf_mf;
     313              : 
     314              :     BC::BC<Set::Scalar> *mybc = nullptr;
     315              : 
     316              :     //amrex::Real M, mu, gamma, sigma0, l_gb, beta;
     317              :     RegularizationType regularization = RegularizationType::K12;
     318              :     enum ThresholdType {
     319              :         Continuous = 0, Chop = 1
     320              :     };
     321              :     struct {
     322              :         Set::Scalar M = NAN;
     323              :         Set::Scalar L = NAN;
     324              :         Set::Scalar mu = NAN;
     325              :         Set::Scalar gamma = NAN;
     326              :         Set::Scalar sigma0 = NAN;
     327              :         Set::Scalar l_gb = NAN;
     328              :         bool elastic_df = false;
     329              :         Set::Scalar elastic_mult = NAN;
     330              :         struct {
     331              :             bool on = false;
     332              :             bool chempot = false;
     333              :             bool boundary = false;
     334              :             bool corner = false;
     335              :             bool lagrange = false;
     336              :             bool mechanics = false;
     337              :             bool sdf = false;
     338              :             Set::Scalar value = NAN;
     339              :             ThresholdType type = ThresholdType::Continuous;
     340              :         } threshold;
     341              :     } pf;
     342              : 
     343              :     struct {
     344              :         int on = 0;
     345              :         Set::Scalar tstart = 0.0;
     346              :         Numeric::Interpolator::Linear<Set::Scalar> mobility;
     347              :         Numeric::Interpolator::Linear<Set::Scalar> threshold;
     348              :         Set::Field<Set::Scalar> L_mf;
     349              :         Set::Field<Set::Scalar> threshold_mf;
     350              :     } anisotropic_kinetics;
     351              : 
     352              :     struct {
     353              :         int on = 0;
     354              :         Set::Scalar beta;
     355              :         Set::Scalar timestep;
     356              :         Set::Scalar tstart;
     357              :         int plot_int = -1;
     358              :         Set::Scalar plot_dt = -1.0;
     359              :         int thermo_int = -1, thermo_plot_int = -1;
     360              :         Set::Scalar theta0,sigma0,sigma1;
     361              :         Set::Scalar phi0 = 0.0;
     362              :         int elastic_int = -1;
     363              :     } anisotropy;
     364              :     
     365              :     struct { 
     366              :         bool on = 0;
     367              :         Set::Scalar tstart = NAN;
     368              :         Set::Scalar vol0 = NAN;
     369              :         Set::Scalar lambda = NAN;
     370              :     } lagrange;
     371              : 
     372              :     struct {
     373              :         int on = 0;
     374              :         std::vector<Numeric::Interpolator::Linear<Set::Scalar>> val;
     375              :         Set::Scalar tstart = 0.0;
     376              :     } sdf;
     377              : 
     378              :     struct {
     379              :         int on = 0;
     380              :         Set::Scalar amp = 0.0;
     381              :         Set::Scalar sd = 0.0;
     382              :         Set::Scalar tstart = 0.0;
     383              :         std::normal_distribution<double> norm_dist;
     384              :         std::default_random_engine rand_num_gen;
     385              :     } fluctuation;
     386              : 
     387              :     struct {
     388              :         int on = 0;
     389              :         Model::Defect::Disconnection model;
     390              :     } disconnection;
     391              : 
     392              :     struct {
     393              :         int on = 0;
     394              :         std::vector<Set::Matrix> Fgb;
     395              :     } shearcouple;
     396              : 
     397              :     std::string gb_type, filename;
     398              : 
     399              :     Model::Interface::GB::GB *boundary = nullptr;
     400              : 
     401              :     IC::IC<Set::Scalar> *ic = nullptr;
     402              : 
     403              :     Set::Scalar volume = 5;
     404              :     Set::Scalar area = 0.0;
     405              :     Set::Scalar gbenergy = 0.0;
     406              :     Set::Scalar realgbenergy = 0.0;
     407              :     Set::Scalar regenergy = 0.0;
     408              : 
     409              :     struct
     410              :     {
     411              :         Set::Scalar tstart = 0.0;
     412              :         std::vector<model_type> model;
     413              :         int model_mix_order;
     414              :         bool model_neumann_boundary = false;
     415              :     } mechanics;
     416              :     
     417              :     using Base::Mechanics<model_type>::model_mf;
     418              :     using Base::Mechanics<model_type>::stress_mf;
     419              : 
     420              : };
     421              : }
     422              : #endif
        

Generated by: LCOV version 2.0-1