|           Line data    Source code 
       1             : //* This file is part of the MOOSE framework
       2             : //* https://mooseframework.inl.gov
       3             : //*
       4             : //* All rights reserved, see COPYRIGHT for full restrictions
       5             : //* https://github.com/idaholab/moose/blob/master/COPYRIGHT
       6             : //*
       7             : //* Licensed under LGPL 2.1, please see LICENSE for details
       8             : //* https://www.gnu.org/licenses/lgpl-2.1.html
       9             : 
      10             : #include "INSFVTurbulentTemperatureWallFunction.h"
      11             : #include "NavierStokesMethods.h"
      12             : #include "NS.h"
      13             : 
      14             : registerADMooseObject("NavierStokesApp", INSFVTurbulentTemperatureWallFunction);
      15             : 
      16             : InputParameters
      17         366 : INSFVTurbulentTemperatureWallFunction::validParams()
      18             : {
      19         366 :   InputParameters params = FVFluxBC::validParams();
      20             : 
      21         366 :   params.addClassDescription("Adds turbulent temperature wall function.");
      22         732 :   params.addRequiredParam<MooseFunctorName>("T_w", "The wall temperature.");
      23         732 :   params.addRequiredParam<MooseFunctorName>("u", "The velocity in the x direction.");
      24         732 :   params.addParam<MooseFunctorName>("v", "The velocity in the y direction.");
      25         732 :   params.addParam<MooseFunctorName>("w", "The velocity in the z direction.");
      26         366 :   params.addRequiredParam<MooseFunctorName>(NS::density, "Density");
      27         366 :   params.addRequiredParam<MooseFunctorName>(NS::mu, "Dynamic viscosity.");
      28         366 :   params.addRequiredParam<MooseFunctorName>(NS::cp, "The specific heat at constant pressure.");
      29         366 :   params.addParam<MooseFunctorName>(NS::kappa, "The thermal conductivity.");
      30         732 :   params.addParam<MooseFunctorName>("Pr_t", 0.58, "The turbulent Prandtl number.");
      31         366 :   params.addParam<MooseFunctorName>(NS::TKE, "Turbulent kinetic energy functor.");
      32         732 :   params.addParam<Real>("C_mu", 0.09, "Coupled turbulent kinetic energy closure.");
      33         732 :   MooseEnum wall_treatment("eq_newton eq_incremental eq_linearized neq", "neq");
      34         732 :   params.addParam<MooseEnum>(
      35             :       "wall_treatment", wall_treatment, "The method used for computing the wall functions");
      36             : 
      37         732 :   params.addParam<bool>("newton_solve", false, "Whether a Newton nonlinear solve is being used");
      38         732 :   params.addParamNamesToGroup("newton_solve", "Advanced");
      39         366 :   return params;
      40         366 : }
      41             : 
      42         198 : INSFVTurbulentTemperatureWallFunction::INSFVTurbulentTemperatureWallFunction(
      43         198 :     const InputParameters & parameters)
      44             :   : FVFluxBC(parameters),
      45         198 :     _dim(_subproblem.mesh().dimension()),
      46         396 :     _T_w(getFunctor<ADReal>("T_w")),
      47         396 :     _u_var(getFunctor<ADReal>("u")),
      48         594 :     _v_var(parameters.isParamValid("v") ? &(getFunctor<ADReal>("v")) : nullptr),
      49         198 :     _w_var(parameters.isParamValid("w") ? &(getFunctor<ADReal>("w")) : nullptr),
      50         198 :     _rho(getFunctor<ADReal>(NS::density)),
      51         198 :     _mu(getFunctor<ADReal>(NS::mu)),
      52         198 :     _cp(getFunctor<ADReal>(NS::cp)),
      53         198 :     _kappa(getFunctor<ADReal>(NS::kappa)),
      54         396 :     _Pr_t(getFunctor<ADReal>("Pr_t")),
      55         198 :     _k(getFunctor<ADReal>(NS::TKE)),
      56         396 :     _C_mu(getParam<Real>("C_mu")),
      57         396 :     _wall_treatment(getParam<MooseEnum>("wall_treatment")),
      58         594 :     _newton_solve(getParam<bool>("newton_solve"))
      59             : {
      60         198 : }
      61             : 
      62             : ADReal
      63       16984 : INSFVTurbulentTemperatureWallFunction::computeQpResidual()
      64             : {
      65             :   // Useful parameters
      66       16984 :   const FaceInfo & fi = *_face_info;
      67             :   // if on an internal face (internal to the mesh, but an external boundary of the flow area),
      68             :   // we have to select the element on which the flow variables / temperature are defined
      69       16984 :   const bool use_elem = (_face_type == FaceInfo::VarFaceNeighbors::ELEM);
      70             :   const Real wall_dist = std::abs(
      71       16984 :       ((use_elem ? fi.elemCentroid() : fi.neighborCentroid()) - fi.faceCentroid()) * fi.normal());
      72       16984 :   const Elem * const elem_ptr = use_elem ? fi.elemPtr() : fi.neighborPtr();
      73       16984 :   const auto elem_arg = makeElemArg(elem_ptr);
      74       16984 :   const auto state = determineState();
      75             :   const auto old_state = Moose::StateArg(1, Moose::SolutionIterationType::Nonlinear);
      76       16984 :   const auto mu = _mu(elem_arg, state);
      77       16984 :   const auto rho = _rho(elem_arg, state);
      78       16984 :   const auto cp = _cp(elem_arg, state);
      79       16984 :   const auto kappa = _kappa(elem_arg, state);
      80             : 
      81             :   // Get the velocity vector
      82       33968 :   ADRealVectorValue velocity(_u_var(elem_arg, state));
      83       16984 :   if (_v_var)
      84       16984 :     velocity(1) = (*_v_var)(elem_arg, state);
      85       16984 :   if (_w_var)
      86           0 :     velocity(2) = (*_w_var)(elem_arg, state);
      87             : 
      88             :   // Compute the velocity and direction of the velocity component that is parallel to the wall
      89             :   const ADReal parallel_speed =
      90       50952 :       NS::computeSpeed<ADReal>(velocity - velocity * (fi.normal()) * (fi.normal()));
      91             : 
      92             :   // Computing friction velocity and y+
      93             :   ADReal u_tau, y_plus;
      94             : 
      95       16984 :   if (_wall_treatment == "eq_newton")
      96             :   {
      97             :     // Full Newton-Raphson solve to find the wall quantities from the law of the wall
      98           0 :     u_tau = NS::findUStar<ADReal>(mu, rho, parallel_speed, wall_dist);
      99           0 :     y_plus = wall_dist * u_tau * rho / mu;
     100             :   }
     101       16984 :   else if (_wall_treatment == "eq_incremental")
     102             :   {
     103             :     // Incremental solve on y_plus to get the near-wall quantities
     104           0 :     y_plus = NS::findyPlus<ADReal>(mu, rho, std::max(parallel_speed, 1e-10), wall_dist);
     105           0 :     u_tau = parallel_speed / (std::log(std::max(NS::E_turb_constant * y_plus, 1.0 + 1e-4)) /
     106           0 :                               NS::von_karman_constant);
     107             :   }
     108       16984 :   else if (_wall_treatment == "eq_linearized")
     109             :   {
     110             :     // Linearized approximation to the wall function to find the near-wall quantities faster
     111       16984 :     const ADReal a_c = 1 / NS::von_karman_constant;
     112       33968 :     const ADReal b_c = 1.0 / NS::von_karman_constant *
     113       50952 :                        (std::log(NS::E_turb_constant * std::max(wall_dist, 1.0) / mu) + 1.0);
     114       16984 :     const ADReal c_c = parallel_speed;
     115             : 
     116       84920 :     u_tau = (-b_c + std::sqrt(std::pow(b_c, 2) + 4.0 * a_c * c_c)) / (2.0 * a_c);
     117       16984 :     y_plus = wall_dist * u_tau * rho / mu;
     118             :   }
     119           0 :   else if (_wall_treatment == "neq")
     120             :   {
     121             :     // Assign non-equilibrium wall function value
     122           0 :     y_plus = wall_dist * std::sqrt(std::sqrt(_C_mu) * _k(elem_arg, old_state)) * rho / mu;
     123           0 :     u_tau = parallel_speed / (std::log(std::max(NS::E_turb_constant * y_plus, 1.0 + 1e-4)) /
     124           0 :                               NS::von_karman_constant);
     125             :   }
     126             : 
     127             :   ADReal alpha;
     128       16984 :   if (y_plus <= 5.0) // sub-laminar layer
     129             :   {
     130       16723 :     alpha = kappa / (rho * cp);
     131             :   }
     132         261 :   else if (y_plus >= 30.0) // log-layer
     133             :   {
     134           0 :     const auto Pr = cp * mu / kappa;
     135           0 :     const auto Pr_ratio = Pr / _Pr_t(elem_arg, state);
     136             :     const auto jayatilleke_P =
     137           0 :         9.24 * (std::pow(Pr_ratio, 0.75) - 1.0) * (1.0 + 0.28 * std::exp(-0.007 * Pr_ratio));
     138             :     const auto wall_scaling =
     139           0 :         1.0 / NS::von_karman_constant * std::log(NS::E_turb_constant * y_plus) + jayatilleke_P;
     140           0 :     alpha = u_tau * wall_dist / (_Pr_t(elem_arg, state) * wall_scaling);
     141             :   }
     142             :   else // buffer layer
     143             :   {
     144         261 :     const auto alpha_lam = kappa / (rho * cp);
     145         261 :     const auto Pr = cp * mu / kappa;
     146         261 :     const auto Pr_t = _Pr_t(elem_arg, state);
     147             :     const auto Pr_ratio = Pr / Pr_t;
     148             :     const auto jayatilleke_P =
     149        1566 :         9.24 * (std::pow(Pr_ratio, 0.75) - 1.0) * (1.0 + 0.28 * std::exp(-0.007 * Pr_ratio));
     150             :     const auto wall_scaling =
     151         522 :         1.0 / NS::von_karman_constant * std::log(NS::E_turb_constant * y_plus) + jayatilleke_P;
     152         261 :     const auto alpha_turb = u_tau * wall_dist / (Pr_t * wall_scaling);
     153         261 :     const auto blending_function = (y_plus - 5.0) / 25.0;
     154         522 :     alpha = blending_function * alpha_turb + (1.0 - blending_function) * alpha_lam;
     155             :   }
     156             : 
     157             :   // To make sure new derivatives are not introduced as the solve progresses
     158       16984 :   if (_newton_solve)
     159       33672 :     alpha += 0 * kappa * (rho * cp) + 0 * u_tau * _Pr_t(elem_arg, state) * y_plus;
     160             : 
     161       16984 :   const auto face_arg = singleSidedFaceArg();
     162       50952 :   return -rho * cp * alpha * (_T_w(face_arg, state) - _var(elem_arg, state)) / wall_dist;
     163             : }
 |