mus_bc_fluid_turbulent_module.f90 Source File


This file depends on

sourcefile~~mus_bc_fluid_turbulent_module.f90~~EfferentGraph sourcefile~mus_bc_fluid_turbulent_module.f90 mus_bc_fluid_turbulent_module.f90 sourcefile~mus_bc_header_module.f90 mus_bc_header_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_dervarpos_module.f90 mus_derVarPos_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_dervarpos_module.f90 sourcefile~mus_field_prop_module.f90 mus_field_prop_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_field_prop_module.f90 sourcefile~mus_mixture_module.f90 mus_mixture_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_mixture_module.f90 sourcefile~mus_param_module.f90 mus_param_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_param_module.f90 sourcefile~mus_physics_module.f90 mus_physics_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_relaxationparam_module.f90 mus_relaxationParam_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_relaxationparam_module.f90 sourcefile~mus_scheme_derived_quantities_type_module.f90 mus_scheme_derived_quantities_type_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_scheme_derived_quantities_type_module.f90 sourcefile~mus_scheme_layout_module.f90 mus_scheme_layout_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_turb_wallfunc_module.f90 mus_turb_wallFunc_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_turb_wallfunc_module.f90 sourcefile~mus_turbulence_module.f90 mus_turbulence_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_turbulence_module.f90 sourcefile~mus_varsys_module.f90 mus_varSys_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_varsys_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_dervarpos_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_field_prop_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_mixture_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_turb_wallfunc_module.f90 sourcefile~mus_pdf_module.f90 mus_pdf_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_pdf_module.f90 sourcefile~mus_dervarpos_module.f90->sourcefile~mus_scheme_derived_quantities_type_module.f90 sourcefile~mus_dervarpos_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_fluid_module.f90 mus_fluid_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_fluid_module.f90 sourcefile~mus_poisson_module.f90 mus_poisson_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_poisson_module.f90 sourcefile~mus_scheme_header_module.f90 mus_scheme_header_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_species_module.f90 mus_species_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_species_module.f90 sourcefile~mus_mixture_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_enrtl_dummy.f90 mus_eNRTL_dummy.f90 sourcefile~mus_mixture_module.f90->sourcefile~mus_enrtl_dummy.f90 sourcefile~mus_mixture_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_param_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_abortcriteria_module.f90 mus_abortCriteria_module.f90 sourcefile~mus_param_module.f90->sourcefile~mus_abortcriteria_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_dervarpos_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_turbulence_module.f90 sourcefile~mus_graddata_module.f90 mus_gradData_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_graddata_module.f90 sourcefile~mus_moments_type_module.f90 mus_moments_type_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_moments_type_module.f90 sourcefile~mus_nonnewtonian_module.f90 mus_nonNewtonian_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_nonnewtonian_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_scheme_layout_module.f90->sourcefile~mus_scheme_derived_quantities_type_module.f90 sourcefile~mus_scheme_layout_module.f90->sourcefile~mus_moments_type_module.f90 sourcefile~mus_wall_function_abstract_module.f90 mus_wall_function_abstract_module.f90 sourcefile~mus_turb_wallfunc_module.f90->sourcefile~mus_wall_function_abstract_module.f90 sourcefile~mus_wall_function_musker_module.f90 mus_wall_function_musker_module.f90 sourcefile~mus_turb_wallfunc_module.f90->sourcefile~mus_wall_function_musker_module.f90 sourcefile~mus_wall_function_reichardt_module.f90 mus_wall_function_reichardt_module.f90 sourcefile~mus_turb_wallfunc_module.f90->sourcefile~mus_wall_function_reichardt_module.f90 sourcefile~mus_wall_function_schmitt_module.f90 mus_wall_function_schmitt_module.f90 sourcefile~mus_turb_wallfunc_module.f90->sourcefile~mus_wall_function_schmitt_module.f90 sourcefile~mus_turbulence_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_turbulence_module.f90->sourcefile~mus_graddata_module.f90 sourcefile~mus_varsys_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_varsys_module.f90->sourcefile~mus_scheme_derived_quantities_type_module.f90 sourcefile~mus_connectivity_module.f90 mus_connectivity_module.f90 sourcefile~mus_varsys_module.f90->sourcefile~mus_connectivity_module.f90 sourcefile~mus_geom_module.f90 mus_geom_module.f90 sourcefile~mus_varsys_module.f90->sourcefile~mus_geom_module.f90 sourcefile~mus_scheme_type_module.f90 mus_scheme_type_module.f90 sourcefile~mus_varsys_module.f90->sourcefile~mus_scheme_type_module.f90

Files dependent on this one

sourcefile~~mus_bc_fluid_turbulent_module.f90~~AfferentGraph sourcefile~mus_bc_fluid_turbulent_module.f90 mus_bc_fluid_turbulent_module.f90 sourcefile~mus_bc_general_module.f90 mus_bc_general_module.f90 sourcefile~mus_bc_general_module.f90->sourcefile~mus_bc_fluid_turbulent_module.f90 sourcefile~mus_control_module.f90 mus_control_module.f90 sourcefile~mus_control_module.f90->sourcefile~mus_bc_general_module.f90 sourcefile~mus_debug_module.f90 mus_debug_module.f90 sourcefile~mus_control_module.f90->sourcefile~mus_debug_module.f90 sourcefile~mus_debug_module.f90->sourcefile~mus_bc_general_module.f90 sourcefile~mus_dynloadbal_module.f90 mus_dynLoadBal_module.f90 sourcefile~mus_dynloadbal_module.f90->sourcefile~mus_bc_general_module.f90 sourcefile~mus_construction_module.f90 mus_construction_module.f90 sourcefile~mus_dynloadbal_module.f90->sourcefile~mus_construction_module.f90 sourcefile~mus_program_module.f90 mus_program_module.f90 sourcefile~mus_program_module.f90->sourcefile~mus_bc_general_module.f90 sourcefile~mus_program_module.f90->sourcefile~mus_control_module.f90 sourcefile~mus_program_module.f90->sourcefile~mus_dynloadbal_module.f90 sourcefile~mus_program_module.f90->sourcefile~mus_construction_module.f90 sourcefile~mus_construction_module.f90->sourcefile~mus_debug_module.f90 sourcefile~musubi.f90 musubi.f90 sourcefile~musubi.f90->sourcefile~mus_control_module.f90 sourcefile~musubi.f90->sourcefile~mus_program_module.f90 sourcefile~mus_harvesting.f90 mus_harvesting.f90 sourcefile~mus_harvesting.f90->sourcefile~mus_construction_module.f90 sourcefile~mus_hvs_construction_module.f90 mus_hvs_construction_module.f90 sourcefile~mus_hvs_construction_module.f90->sourcefile~mus_construction_module.f90

Source Code

! Copyright (c) 2021-2022 Kannan Masilamani <kannan.masilamani@dlr.de>
! Copyright (c) 2021 Gregorio Gerardo Spinelli <gregoriogerardo.spinelli@dlr.de>
!
! Redistribution and use in source and binary forms, with or without
! modification, are permitted provided that the following conditions are met:
!
! 1. Redistributions of source code must retain the above copyright notice,
! this list of conditions and the following disclaimer.
!
! 2. Redistributions in binary form must reproduce the above copyright notice,
! this list of conditions and the following disclaimer in the documentation
! and/or other materials provided with the distribution.
!
! THIS SOFTWARE IS PROVIDED BY THE UNIVERSITY OF SIEGEN “AS IS” AND ANY EXPRESS
! OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
! OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
! IN NO EVENT SHALL UNIVERSITY OF SIEGEN OR CONTRIBUTORS BE LIABLE FOR ANY
! DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
! (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
! LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
! ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
! (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
! SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
! Copyright (c) 2011-2013 Manuel Hasert <m.hasert@grs-sim.de>
! Copyright (c) 2011 Harald Klimach <harald.klimach@uni-siegen.de>
! Copyright (c) 2011 Konstantin Kleinheinz <k.kleinheinz@grs-sim.de>
! Copyright (c) 2011-2012 Simon Zimny <s.zimny@grs-sim.de>
! Copyright (c) 2012, 2014-2016 Jiaxing Qi <jiaxing.qi@uni-siegen.de>
! Copyright (c) 2012 Kartik Jain <kartik.jain@uni-siegen.de>
! Copyright (c) 2013-2015, 2019 Kannan Masilamani <kannan.masilamani@uni-siegen.de>
! Copyright (c) 2016 Tobias Schneider <tobias1.schneider@student.uni-siegen.de>
!
! Redistribution and use in source and binary forms, with or without
! modification, are permitted provided that the following conditions are met:
!
! 1. Redistributions of source code must retain the above copyright notice,
! this list of conditions and the following disclaimer.
!
! 2. Redistributions in binary form must reproduce the above copyright notice,
! this list of conditions and the following disclaimer in the documentation
! and/or other materials provided with the distribution.
!
! THIS SOFTWARE IS PROVIDED BY THE UNIVERSITY OF SIEGEN “AS IS” AND ANY EXPRESS
! OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
! OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
! IN NO EVENT SHALL UNIVERSITY OF SIEGEN OR CONTRIBUTORS BE LIABLE FOR ANY
! DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
! (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
! LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
! ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
! (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
! SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
! **************************************************************************** !
!> Boundary condition wall treatment routines
!!
!! This module contains higher order wall treatments
!! A detailed description on the implementation details are given
!! in [[tem_bc_module]].
!!
module mus_bc_fluid_turbulent_module
  use iso_c_binding, only: c_f_pointer

  ! include treelm modules
  use env_module,               only: rk, long_k
  use tem_param_module,         only: cs2inv, cs2, rho0, rho0Inv, cs4inv, &
    &                                 div1_3
  use tem_isNaN_module,         only: tem_isNaN
  use tem_time_module,          only: tem_time_type
  use treelmesh_module,         only: treelmesh_type
  use tem_varSys_module,        only: tem_varSys_type
  use tem_debug_module,         only: dbgUnit
  use tem_geometry_module,      only: tem_ElemSizeLevel
  use tem_property_module,      only: prp_solid
  use tem_construction_module,  only: tem_levelDesc_type
  use tem_float_module,         only: operator(.feq.), operator(.fne.)
  use tem_stencil_module,       only: tem_stencilHeader_type
  use tem_math_module,          only: cross_product3D
  use tem_aux_module,           only: tem_abort
  use tem_logging_module,       only: logUnit

  ! include musubi modules
  use mus_bc_header_module,       only: boundary_type, glob_boundary_type
  use mus_scheme_layout_module,   only: mus_scheme_layout_type
  use mus_field_prop_module,      only: mus_field_prop_type
  use mus_derVarPos_module,       only: mus_derVarPos_type
  use mus_param_module,           only: mus_param_type
  use mus_physics_module,         only: mus_physics_type
  use mus_mixture_module,         only: mus_mixture_type
  use mus_varSys_module,          only: mus_varSys_data_type
  use mus_relaxationParam_module, only: mus_viscosity_type
  use mus_turb_wallFunc_module,   only: mus_turb_wallFunc_type
  use mus_turbulence_module,      only: mus_turbulence_type
  use mus_scheme_derived_quantities_module, only: mus_scheme_derived_quantities_type

  implicit none

  private

  public :: turbulent_wall
  public :: turbulent_wall_libb
  public :: turbulent_wall_eq
  public :: turbulent_wall_eq_curved
  public :: turbulent_wall_noneq_expol
  public :: turbulent_wall_noneq_expol_curved
  !! Constant parameters for van-driest damping function
  real(kind=rk), parameter :: vd_Aplus = 26.0_rk

contains

  ! ************************************************************************** !
  !> BC routine for turbulent wall.
  !! It uses wall model to compute velocity on the boundary node.
  !! The implementation is based on the following paper:
  !! Haussmann, Marc; Ries, Florian; Jeppener-Haltenhoff, Jonathan B.; Li,
  !! Yongxiang; Schmidt, Marius; Welch, Cooper et al. (2020): Evaluation of a
  !! Near-Wall-Modeled Large Eddy Lattice Boltzmann Method for the Analysis of
  !! Complex Flows Relevant to IC Engines. In Computation 8 (2), p. 43.
  !! DOI: 10.3390/computation8020043.
  !!
  !! Usage
  !! -----
  !!```lua
  !!boundary_condition = {
  !!  {
  !!    label = 'wall',
  !!    kind = 'turbulent_wall',
  !!    wall_model = 'musker',
  !!    nonlinear_solver = 'fixed_point'
  !!  }
  !!}
  !!
  !! This subroutine's interface must match the abstract interface definition
  !! [[boundaryRoutine]] in bc/[[mus_bc_header_module]].f90 in order to be
  !! callable via [[boundary_type:fnct]] function pointer.
  subroutine turbulent_wall( me, state, bcBuffer, globBC,                &
    &          levelDesc, tree, nSize, iLevel, sim_time, neigh, layout,  &
    &          fieldProp, varPos, nScalars, varSys, derVarPos, physics,  &
    &          iField, mixture )
    ! ------------------------------------------------------------------------ !
    !> global boundary type
    class(boundary_type) :: me
    !> Current state vector of iLevel
    real(kind=rk), intent(inout) :: state(:)
    !> size of state array ( in terms of elements )
    integer, intent(in) :: nSize
    !> state values of boundary elements of all fields of iLevel
    real(kind=rk), intent(in) :: bcBuffer(:)
    !> iLevel descriptor
    type(tem_levelDesc_type), intent(in) :: levelDesc
    !> Treelm Mesh
    type(treelmesh_type), intent(in) :: tree
    !> fluid parameters and properties
    type(mus_field_prop_type), intent(in) :: fieldProp
    !> stencil layout information
    type(mus_scheme_layout_type), intent(in) :: layout
    !> the level On which this boundary was invoked
    integer, intent(in) :: iLevel
    !> connectivity array corresponding to state vector
    integer, intent(in) :: neigh(:)
    !> global time information
    type(tem_time_type), intent(in)  :: sim_time
    !> pointer to field variable in the state vector
    integer, intent(in) :: varPos(:)
    !> number of Scalars in the scheme var system
    integer, intent(in) :: nScalars
    !> scheme variable system
    type(tem_varSys_type), intent(in) :: varSys
    !> position of derived quantities in varsys
    type(mus_derVarPos_type), intent(in) :: derVarPos
    !> scheme global boundary type
    type(glob_boundary_type), intent(in) :: globBC
    !> contains physics conversion factors
    type(mus_physics_type), intent(in) :: physics
    !> current field
    integer, intent(in) :: iField
    !> mixture info
    type(mus_mixture_type), intent(in) :: mixture
    ! ------------------------------------------------------------------------ !
    integer :: iElem, iDir, QQ, elemPos
    real(kind=rk) :: f_pre(layout%fStencil%QQ)
    real(kind=rk) :: f_neigh(layout%fStencil%QQ)
    real(kind=rk) :: dens, velSW(3)
    real(kind=rk) :: dens_neigh, vel_neigh(3)
    real(kind=rk) :: fEq_bnd(layout%fStencil%QQ), fEq_neigh(layout%fStencil%QQ)
    real(kind=rk) :: unitSW(3, globBC%nElems(iLevel))
    real(kind=rk) :: velSW_bnd(globBC%nElems(iLevel))
    ! ------------------------------------------------------------------------ !
    ! Load stencil for velocity space (DdQq) with q=QQ
    QQ = layout%fStencil%QQ

    ! Calculate stream-wise velocity component from wall function
    call calcVelSW_unitSW_velTau_tVisc(                           &
      & velSW          = velSW_bnd,                               &
      & unitSW         = unitSW,                                  &
      & turbwallFunc   = me%turbwallFunc,                         &
      & nElems         = globBC%nElems(iLevel),                   &
      & elemPos        = globBC%elemLvl(iLevel)%elem%val(:),      &
      & neighBufferPre = me%neigh(iLevel)%neighBufferPre_nNext,   &
      & viscKine       = fieldProp%fluid%viscKine,                &
      & turbulence     = fieldProp%fluid%turbulence,              &
      & stencil        = layout%fStencil,                         &
      & iLevel         = iLevel,                                  &
      & quantities     = layout%quantities                        )

    ! Velocity correction on boundary element incoming direction
    do iElem = 1, globBC%nElems(iLevel)
      ! Current element position in state array. (Used for state in step 10.)
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)
      ! Compute density and velocity from pre-collision state.
      ! computeNeighBuf uses FETCH which does implicit bounce back which
      ! is valid for qVal=0.5
      f_pre = me%neigh(iLevel)%computeNeighBuf( (iElem-1)*QQ+1: iElem*QQ )
      ! Compute density and velocity from pre-collision state on neighbor
      f_neigh = me%neigh(iLevel)%neighBufferPre_nNext( 1,         &
        &                          (iElem-1)*QQ+1:(iElem-1)*QQ+QQ )
      dens = sum(f_pre)
      dens_neigh = sum(f_neigh)
      ! velocity
      vel_neigh = layout%quantities%vel_from_pdf_ptr(pdf = f_neigh, dens = dens_neigh)

      ! stream-wise velocity on boundary element from wall model
      velSW = velSW_bnd(iElem) * unitSW(:, iElem)

      ! Equilibrium on precollision density and velocity
      fEq_neigh(:) = layout%quantities%pdfEq_ptr( rho = dens_neigh, &
        &                                         vel = vel_neigh,  &
        &                                         QQ = QQ           )

      ! Equilibrium on wall model velocity
      fEq_bnd(:) = layout%quantities%pdfEq_ptr( rho = dens,  &
        &                                       vel = velSW, &
        &                                       QQ = QQ      )

      do iDir = 1, layout%fStencil%QQ
        state(  neigh(( idir-1)*nsize+ elempos)+( ifield-1)* qq+ nscalars*0) &
          & = fEq_bnd(iDir) + f_neigh(iDir) - fEq_neigh(iDir)
      end do
    end do

    ! Calculated bndForce on boundary elements using momentum exchange method
    call calcTurbWallBndForceAndMoment(                            &
      & bndForce    = me%turbWallFunc%dataOnLvl(iLevel)%bndForce,  &
      & bndMoment   = me%turbWallFunc%dataOnLvl(iLevel)%bndMoment, &
      & momRefPnt   = me%bndMomRefPnt,                             &
      & globBC      = globBC,                                      &
      & baryOfTotal = levelDesc%baryOfTotal,                       &
      & bcBuffer    = bcBuffer,                                    &
      & state       = state,                                       &
      & nSize       = nSize,                                       &
      & neigh       = neigh,                                       &
      & layout      = layout,                                      &
      & physics     = physics,                                     &
      & nScalars    = nScalars,                                    &
      & varPos      = varPos,                                      &
      & iLevel      = iLevel,                                      &
      & iField      = iField                                       )

  end subroutine turbulent_wall
  ! ************************************************************************** !


  ! ************************************************************************** !
  !> BC routine for turbulent wall.
  !! It uses wall model to compute velocity on the boundary node.
  !! The implementation is based on the following paper:
  !! Haussmann, Marc; Ries, Florian; Jeppener-Haltenhoff, Jonathan B.; Li,
  !! Yongxiang; Schmidt, Marius; Welch, Cooper et al. (2020): Evaluation of a
  !! Near-Wall-Modeled Large Eddy Lattice Boltzmann Method for the Analysis of
  !! Complex Flows Relevant to IC Engines. In Computation 8 (2), p. 43.
  !! DOI: 10.3390/computation8020043.
  !!
  !! Usage
  !! -----
  !!```lua
  !!boundary_condition = {
  !!  {
  !!    label = 'wall',
  !!    kind = 'turbulent_wall',
  !!    curved = true,
  !!    wall_model = 'musker',
  !!    nonlinear_solver = 'fixed_point'
  !!  }
  !!}
  !!
  !! This subroutine's interface must match the abstract interface definition
  !! [[boundaryRoutine]] in bc/[[mus_bc_header_module]].f90 in order to be
  !! callable via [[boundary_type:fnct]] function pointer.
  subroutine turbulent_wall_libb( me, state, bcBuffer, globBC,           &
    &          levelDesc, tree, nSize, iLevel, sim_time, neigh, layout,  &
    &          fieldProp, varPos, nScalars, varSys, derVarPos, physics,  &
    &          iField, mixture )
    ! ------------------------------------------------------------------------ !
    !> global boundary type
    class(boundary_type) :: me
    !> Current state vector of iLevel
    real(kind=rk), intent(inout) :: state(:)
    !> size of state array ( in terms of elements )
    integer, intent(in) :: nSize
    !> state values of boundary elements of all fields of iLevel
    real(kind=rk), intent(in) :: bcBuffer(:)
    !> iLevel descriptor
    type(tem_levelDesc_type), intent(in) :: levelDesc
    !> Treelm Mesh
    type(treelmesh_type), intent(in) :: tree
    !> fluid parameters and properties
    type(mus_field_prop_type), intent(in) :: fieldProp
    !> stencil layout information
    type(mus_scheme_layout_type), intent(in) :: layout
    !> the level On which this boundary was invoked
    integer, intent(in) :: iLevel
    !> connectivity array corresponding to state vector
    integer, intent(in) :: neigh(:)
    !> global time information
    type(tem_time_type), intent(in)  :: sim_time
    !> pointer to field variable in the state vector
    integer, intent(in) :: varPos(:)
    !> number of Scalars in the scheme var system
    integer, intent(in) :: nScalars
    !> scheme variable system
    type(tem_varSys_type), intent(in) :: varSys
    !> position of derived quantities in varsys
    type(mus_derVarPos_type), intent(in) :: derVarPos
    !> scheme global boundary type
    type(glob_boundary_type), intent(in) :: globBC
    !> contains physics conversion factors
    type(mus_physics_type), intent(in) :: physics
    !> current field
    integer, intent(in) :: iField
    !> mixture info
    type(mus_mixture_type), intent(in) :: mixture
    ! ------------------------------------------------------------------------ !
    real(kind=rk) :: fIn, fOut, fNgh
    real(kind=rk) :: cIn, cOut, cNgh
    integer :: iLink
    integer :: iElem, iDir, QQ, elemPos
    real(kind=rk) :: f_pre(layout%fStencil%QQ)
    real(kind=rk) :: f_preBuffer(layout%fStencil%QQ * globBC%nElems(iLevel))
    real(kind=rk) :: dens, velSW(3), vel(3)
    real(kind=rk) :: fEq_bnd, fEq
    real(kind=rk) :: unitSW(3, globBC%nElems(iLevel))
    real(kind=rk) :: velSW_bnd(globBC%nElems(iLevel))
    ! ------------------------------------------------------------------------ !
    ! Load stencil for velocity space (DdQq) with q=QQ
    QQ = layout%fStencil%QQ

    ! 1. First perform libb step i.e. curved boundary step
    !NEC$ ivdep
    !DIR$ ivdep
    !IBM* independent
    do iLink = 1, me%links(iLevel)%nVals

      cIn  = me%bouzidi(iLevel)% cIn( iLink )
      cOut = me%bouzidi(iLevel)%cOut( iLink )
      cNgh = me%bouzidi(iLevel)%cNgh( iLink )

      fIn  = bcBuffer( me%bouzidi(iLevel)% inPos(iLink) )
      fOut = bcBuffer( me%bouzidi(iLevel)%outPos(iLink) )
      fNgh = me%neigh(iLevel)%computeNeighBuf(me%bouzidi(iLevel)%nghPos(iLink))

      state( me%links(iLevel)%val(iLink) ) = cIn*fIn + cOut*fOut + cNgh*fNgh

    end do ! iLink

    ! 2. Calculate stream-wise velocity component from wall function
    call calcVelSW_unitSW_velTau_tVisc(                           &
      & velSW          = velSW_bnd,                               &
      & unitSW         = unitSW,                                  &
      & turbwallFunc   = me%turbwallFunc,                         &
      & nElems         = globBC%nElems(iLevel),                   &
      & elemPos        = globBC%elemLvl(iLevel)%elem%val(:),      &
      & neighBufferPre = me%neigh(iLevel)%neighBufferPre_nNext,   &
      & viscKine       = fieldProp%fluid%viscKine,                &
      & turbulence     = fieldProp%fluid%turbulence,              &
      & stencil        = layout%fStencil,                         &
      & iLevel         = iLevel,                                  &
      & quantities     = layout%quantities                        )

    ! 3. Velocity correction step
    ! Get pre-collision of boundary elementy in a buffer since it gets
    ! overwritten in the next element loop
    do iElem = 1, globBC%nElems(iLevel)
      ! Current element position in state array. (Used for state in step 10.)
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)
      do iDir = 1, layout%fStencil%QQ
        f_preBuffer((iElem-1)*QQ + iDir) = state(                        &
          &  neigh (( idir-1)* nsize+ elempos)+( ifield-1)* qq+ nscalars*0 )
      end do
    end do

    do iElem = 1, globBC%nElems(iLevel)
      ! Current element position in state array. (Used for state in step 10.)
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)
      ! Compute density and velocity from pre-collision state after
      ! curved boundary step
      f_pre = f_preBuffer( (iElem-1)*QQ+1: iElem*QQ )
      dens = sum(f_pre)
      ! velocity
      vel = layout%quantities%vel_from_pdf_ptr(pdf = f_pre, dens = dens)

      ! stream-wise velocity on boundary element from wall model
      velSW = velSW_bnd(iElem) * unitSW(:, iElem)

      do iDir = 1, layout%fStencil%QQN
        if ( globBC%elemLvl(iLevel)%bitmask%val( iDir, iElem )) then
          ! Equilibrium on wall model velocity
          fEq_bnd = layout%quantities%pdfEq_iDir_ptr( rho = dens,                                &
            &                                         vel = velSW,                               &
            &                                         iDir = iDir,                               &
            &                                         cxDirRK = layout%fStencil%cxDirRK(:,iDir), &
            &                                         weight = layout%weight(iDir)               )

          fEq = layout%quantities%pdfEq_iDir_ptr( rho = dens,                                &
            &                                     vel = vel,                                 &
            &                                     iDir = iDir,                               &
            &                                     cxDirRK = layout%fStencil%cxDirRK(:,iDir), &
            &                                     weight = layout%weight(iDir)               )

          state(  neigh(( idir-1)*nsize+ elempos)+( ifield-1)* qq+ nscalars*0) &
            & = fEq_bnd + f_pre(iDir) - fEq
        end if
      end do
    end do

    ! Calculated bndForce on boundary elements using momentum exchange method
    call calcTurbWallBndForceAndMoment(                            &
      & bndForce    = me%turbWallFunc%dataOnLvl(iLevel)%bndForce,  &
      & bndMoment   = me%turbWallFunc%dataOnLvl(iLevel)%bndMoment, &
      & momRefPnt   = me%bndMomRefPnt,                             &
      & globBC      = globBC,                                      &
      & baryOfTotal = levelDesc%baryOfTotal,                       &
      & bcBuffer    = bcBuffer,                                    &
      & state       = state,                                       &
      & nSize       = nSize,                                       &
      & neigh       = neigh,                                       &
      & layout      = layout,                                      &
      & physics     = physics,                                     &
      & nScalars    = nScalars,                                    &
      & varPos      = varPos,                                      &
      & iLevel      = iLevel,                                      &
      & iField      = iField                                       )

  end subroutine turbulent_wall_libb
  ! ************************************************************************** !


  ! ************************************************************************** !
  !> BC routine for turbulent wall based on Guo's nonequilibrium extrapolation.
  !! The implementation is based on the following paper:
  !! Haussmann, M. et al. (2019) ‘Large-eddy simulation coupled with wall models
  !! for turbulent channel flows at high Reynolds numbers with a lattice
  !! Boltzmann method — Application to Coriolis mass flowmeter’, Computers &
  !! Mathematics with Applications. Elsevier Ltd, 78(10), pp. 3285–3302.
  !!
  !! It uses wall model to compute velocity on the boundary node.
  !! All directions of PDF in the boundary elements are updated with
  !! Equilibrium plus non-equilibrium.
  !! Density is computed using Zho-He approach for straight walls.
  !! "On pressure and velocity boundary conditions for the lattice Boltzmann
  !!  BGK model", Physics of Fluids 9, 1591-1598 (1997)
  !! https://doi.org/10.1063/1.869307
  !!
  !! non-equilibrium are computed from PDF on neighbor and extrapolated to
  !! boundary. This routine is used for straight wall boundaries.
  !!
  !! Usage
  !! -----
  !!```lua
  !!boundary_condition = {
  !!  {
  !!    label = 'wall',
  !!    kind = 'turbulent_wall_noneq_expol',
  !!    wall_model = 'musker',
  !!    nonlinear_solver = 'fixed_point'
  !!  }
  !!}
  !!
  !! This subroutine's interface must match the abstract interface definition
  !! [[boundaryRoutine]] in bc/[[mus_bc_header_module]].f90 in order to be
  !! callable via [[boundary_type:fnct]] function pointer.
  subroutine turbulent_wall_noneq_expol( me, state, bcBuffer, globBC,    &
    &          levelDesc, tree, nSize, iLevel, sim_time, neigh, layout,  &
    &          fieldProp, varPos, nScalars, varSys, derVarPos, physics,  &
    &          iField, mixture )
    ! ------------------------------------------------------------------------ !
    !> global boundary type
    class(boundary_type) :: me
    !> Current state vector of iLevel
    real(kind=rk), intent(inout) :: state(:)
    !> size of state array ( in terms of elements )
    integer, intent(in) :: nSize
    !> state values of boundary elements of all fields of iLevel
    real(kind=rk), intent(in) :: bcBuffer(:)
    !> iLevel descriptor
    type(tem_levelDesc_type), intent(in) :: levelDesc
    !> Treelm Mesh
    type(treelmesh_type), intent(in) :: tree
    !> fluid parameters and properties
    type(mus_field_prop_type), intent(in) :: fieldProp
    !> stencil layout information
    type(mus_scheme_layout_type), intent(in) :: layout
    !> the level On which this boundary was invoked
    integer, intent(in) :: iLevel
    !> connectivity array corresponding to state vector
    integer, intent(in) :: neigh(:)
    !> global time information
    type(tem_time_type), intent(in)  :: sim_time
    !> pointer to field variable in the state vector
    integer, intent(in) :: varPos(:)
    !> number of Scalars in the scheme var system
    integer, intent(in) :: nScalars
    !> scheme variable system
    type(tem_varSys_type), intent(in) :: varSys
    !> position of derived quantities in varsys
    type(mus_derVarPos_type), intent(in) :: derVarPos
    !> scheme global boundary type
    type(glob_boundary_type), intent(in) :: globBC
    !> contains physics conversion factors
    type(mus_physics_type), intent(in) :: physics
    !> current field
    integer, intent(in) :: iField
    !> mixture info
    type(mus_mixture_type), intent(in) :: mixture
    ! ------------------------------------------------------------------------ !
    integer :: iElem, iDir, QQ, elemPos, normalInd_inv
    real(kind=rk) :: f_pre(layout%fStencil%QQ)
    real(kind=rk) :: f_neigh(layout%fStencil%QQ)
    real(kind=rk) :: dens_neigh, velSW(3), vel_neigh(3)
    real(kind=rk) :: dens_bnd, vel_normal, normal(3)
    real(kind=rk) :: fEq(layout%fStencil%QQ), fEq_bnd(layout%fStencil%QQ)
    real(kind=rk) :: unitSW(3, globBC%nElems(iLevel))
    real(kind=rk) :: velSW_bnd(globBC%nElems(iLevel))
    ! ------------------------------------------------------------------------ !
    ! Load stencil for velocity space (DdQq) with q=QQ
    QQ = layout%fStencil%QQ

    ! Calculate stream-wise velocity component from wall function
    call calcVelSW_unitSW_velTau_tVisc(                           &
      & velSW          = velSW_bnd,                               &
      & unitSW         = unitSW,                                  &
      & turbwallFunc   = me%turbwallFunc,                         &
      & nElems         = globBC%nElems(iLevel),                   &
      & elemPos        = globBC%elemLvl(iLevel)%elem%val(:),      &
      & neighBufferPre = me%neigh(iLevel)%neighBufferPre_nNext,   &
      & viscKine       = fieldProp%fluid%viscKine,                &
      & turbulence     = fieldProp%fluid%turbulence,              &
      & stencil        = layout%fStencil,                         &
      & iLevel         = iLevel,                                  &
      & quantities     = layout%quantities                        )

    ! Velocity correction on boundary element incoming direction
    do iElem = 1, globBC%nElems(iLevel)
      ! Current element position in state array. (Used for state in step 10.)
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)
      ! Compute density and velocity from pre-collision state on neighbor
      f_neigh = me%neigh(iLevel)%neighBufferPre_nNext( 1,         &
        &                          (iElem-1)*QQ+1:(iElem-1)*QQ+QQ )
      dens_neigh = sum(f_neigh)
      ! velocity
      vel_neigh = layout%quantities%vel_from_pdf_ptr(pdf = f_neigh, dens = dens_neigh)

      ! stream-wise velocity on boundary element from wall model
      velSW = velSW_bnd(iElem) * unitSW(:, iElem)

      ! calculate density at boundary for straight wall using Zho-He approach
      normalInd_inv = layout%fStencil%cxDirInv( globBC%elemLvl(iLevel) &
        &                   %normalInd%val(iElem) )
      normal = layout%fStencil%cxDirRK(:, normalInd_inv)
      vel_normal = dot_product(velSW, normal)
      dens_bnd = 0.0_rk
      f_pre = me%neigh(iLevel)%computeNeighBuf( (iElem-1)*QQ+1: iElem*QQ )
      do iDir = 1, layout%fStencil%QQ
        if ( globBC%elemLvl(iLevel)%bitmask%val( iDir, iElem )) then
          dens_bnd = dens_bnd + f_pre(layout%fStencil%cxDirInv(iDir))
        else
          dens_bnd = dens_bnd + f_pre(iDir)
        end if
      end do
      dens_bnd = dens_bnd / (1.0_rk + vel_normal)

      ! Equilibrium on precollision density and velocity
      fEq(:) = layout%quantities%pdfEq_ptr( rho = dens_neigh, &
        &                                   vel = vel_neigh,  &
        &                                   QQ = QQ           )

      ! Equilibrium on wall model velocity
      fEq_bnd(:) = layout%quantities%pdfEq_ptr( rho = dens_bnd,  &
        &                                       vel = velSW,     &
        &                                       QQ = QQ          )

      do iDir = 1, layout%fStencil%QQ
        state(  neigh(( idir-1)*nsize+ elempos)+( ifield-1)* qq+ nscalars*0) &
          & = fEq_bnd(iDir) + ( f_neigh(iDir) - fEq(iDir) )
      end do
    end do

    ! Calculated bndForce on boundary elements using momentum exchange method
    call calcTurbWallBndForceAndMoment(                            &
      & bndForce    = me%turbWallFunc%dataOnLvl(iLevel)%bndForce,  &
      & bndMoment   = me%turbWallFunc%dataOnLvl(iLevel)%bndMoment, &
      & momRefPnt   = me%bndMomRefPnt,                             &
      & globBC      = globBC,                                      &
      & baryOfTotal = levelDesc%baryOfTotal,                       &
      & bcBuffer    = bcBuffer,                                    &
      & state       = state,                                       &
      & nSize       = nSize,                                       &
      & neigh       = neigh,                                       &
      & layout      = layout,                                      &
      & physics     = physics,                                     &
      & nScalars    = nScalars,                                    &
      & varPos      = varPos,                                      &
      & iLevel      = iLevel,                                      &
      & iField      = iField                                       )

  end subroutine turbulent_wall_noneq_expol
  ! ************************************************************************** !

  ! ************************************************************************** !
  !> BC routine for turbulent wall based on Guo's nonequilibrium extrapolation.
  !! The implementation is based on the following paper:
  !! Haussmann, M. et al. (2019) ‘Large-eddy simulation coupled with wall models
  !! for turbulent channel flows at high Reynolds numbers with a lattice
  !! Boltzmann method — Application to Coriolis mass flowmeter’, Computers &
  !! Mathematics with Applications. Elsevier Ltd, 78(10), pp. 3285–3302.
  !!
  !! It uses wall model to compute velocity on the boundary node.
  !! All directions of PDF in the boundary elements are updated with
  !! Equilibrium plus non-equilibrium. Density and non-equilibrium are commputed
  !! from PDF on neighbor and extrapolated to boundary.
  !! This routine is used for curved boundaries.
  !!
  !! Usage
  !! -----
  !!```lua
  !!boundary_condition = {
  !!  {
  !!    label = 'wall',
  !!    kind = 'turbulent_wall_noneq_expol',
  !!    curved = true,
  !!    wall_model = 'musker',
  !!    nonlinear_solver = 'fixed_point'
  !!  }
  !!}
  !!
  !! This subroutine's interface must match the abstract interface definition
  !! [[boundaryRoutine]] in bc/[[mus_bc_header_module]].f90 in order to be
  !! callable via [[boundary_type:fnct]] function pointer.
  subroutine turbulent_wall_noneq_expol_curved( me, state, bcBuffer, globBC, &
    &          levelDesc, tree, nSize, iLevel, sim_time, neigh, layout,      &
    &          fieldProp, varPos, nScalars, varSys, derVarPos, physics,      &
    &          iField, mixture )
    ! ------------------------------------------------------------------------ !
    !> global boundary type
    class(boundary_type) :: me
    !> Current state vector of iLevel
    real(kind=rk), intent(inout) :: state(:)
    !> size of state array ( in terms of elements )
    integer, intent(in) :: nSize
    !> state values of boundary elements of all fields of iLevel
    real(kind=rk), intent(in) :: bcBuffer(:)
    !> iLevel descriptor
    type(tem_levelDesc_type), intent(in) :: levelDesc
    !> Treelm Mesh
    type(treelmesh_type), intent(in) :: tree
    !> fluid parameters and properties
    type(mus_field_prop_type), intent(in) :: fieldProp
    !> stencil layout information
    type(mus_scheme_layout_type), intent(in) :: layout
    !> the level On which this boundary was invoked
    integer, intent(in) :: iLevel
    !> connectivity array corresponding to state vector
    integer, intent(in) :: neigh(:)
    !> global time information
    type(tem_time_type), intent(in)  :: sim_time
    !> pointer to field variable in the state vector
    integer, intent(in) :: varPos(:)
    !> number of Scalars in the scheme var system
    integer, intent(in) :: nScalars
    !> scheme variable system
    type(tem_varSys_type), intent(in) :: varSys
    !> position of derived quantities in varsys
    type(mus_derVarPos_type), intent(in) :: derVarPos
    !> scheme global boundary type
    type(glob_boundary_type), intent(in) :: globBC
    !> contains physics conversion factors
    type(mus_physics_type), intent(in) :: physics
    !> current field
    integer, intent(in) :: iField
    !> mixture info
    type(mus_mixture_type), intent(in) :: mixture
    ! ------------------------------------------------------------------------ !
    integer :: iElem, iDir, QQ, elemPos
    real(kind=rk) :: f_neigh(layout%fStencil%QQ)
    real(kind=rk) :: dens_neigh, velSW(3), vel_neigh(3)
    real(kind=rk) :: fEq(layout%fStencil%QQ), fEq_bnd(layout%fStencil%QQ)
    real(kind=rk) :: unitSW(3, globBC%nElems(iLevel))
    real(kind=rk) :: velSW_bnd(globBC%nElems(iLevel))
    ! ------------------------------------------------------------------------ !
    ! Load stencil for velocity space (DdQq) with q=QQ
    QQ = layout%fStencil%QQ

    ! Calculate stream-wise velocity component from wall function
    call calcVelSW_unitSW_velTau_tVisc(                           &
      & velSW          = velSW_bnd,                               &
      & unitSW         = unitSW,                                  &
      & turbwallFunc   = me%turbwallFunc,                         &
      & nElems         = globBC%nElems(iLevel),                   &
      & elemPos        = globBC%elemLvl(iLevel)%elem%val(:),      &
      & neighBufferPre = me%neigh(iLevel)%neighBufferPre_nNext,   &
      & viscKine       = fieldProp%fluid%viscKine,                &
      & turbulence     = fieldProp%fluid%turbulence,              &
      & stencil        = layout%fStencil,                         &
      & iLevel         = iLevel,                                  &
      & quantities     = layout%quantities                        )

    ! Velocity correction on boundary element incoming direction
    do iElem = 1, globBC%nElems(iLevel)
      ! Current element position in state array. (Used for state in step 10.)
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)
      ! Compute density and velocity from pre-collision state on neighbor
      f_neigh = me%neigh(iLevel)%neighBufferPre_nNext( 1,         &
        &                          (iElem-1)*QQ+1:(iElem-1)*QQ+QQ )
      dens_neigh = sum(f_neigh)
      ! velocity
      vel_neigh = layout%quantities%vel_from_pdf_ptr(pdf = f_neigh, dens = dens_neigh)

      ! stream-wise velocity on boundary element from wall model
      velSW = velSW_bnd(iElem) * unitSW(:, iElem)

      ! Equilibrium on precollision density and velocity
      fEq(:) = layout%quantities%pdfEq_ptr( rho = dens_neigh, &
        &                                   vel = vel_neigh,  &
        &                                   QQ = QQ           )

      ! Equilibrium on wall model velocity
      fEq_bnd(:) = layout%quantities%pdfEq_ptr( rho = dens_neigh, &
        &                                       vel = velSW,      &
        &                                       QQ = QQ           )

      do iDir = 1, layout%fStencil%QQ
        state(  neigh(( idir-1)*nsize+ elempos)+( ifield-1)* qq+ nscalars*0) &
          & = fEq_bnd(iDir) + ( f_neigh(iDir) - fEq(iDir) )
      end do
    end do

    ! Calculated bndForce on boundary elements using momentum exchange method
    call calcTurbWallBndForceAndMoment(                            &
      & bndForce    = me%turbWallFunc%dataOnLvl(iLevel)%bndForce,  &
      & bndMoment   = me%turbWallFunc%dataOnLvl(iLevel)%bndMoment, &
      & momRefPnt   = me%bndMomRefPnt,                             &
      & globBC      = globBC,                                      &
      & baryOfTotal = levelDesc%baryOfTotal,                       &
      & bcBuffer    = bcBuffer,                                    &
      & state       = state,                                       &
      & nSize       = nSize,                                       &
      & neigh       = neigh,                                       &
      & layout      = layout,                                      &
      & physics     = physics,                                     &
      & nScalars    = nScalars,                                    &
      & varPos      = varPos,                                      &
      & iLevel      = iLevel,                                      &
      & iField      = iField                                       )

  end subroutine turbulent_wall_noneq_expol_curved
  ! ************************************************************************** !


  ! ************************************************************************** !
  !> BC routine for turbulent wall based on equilibrium BC.
  !! The implementation is based on the following paper:
  !! Haussmann, M. et al. (2019) ‘Large-eddy simulation coupled with wall models
  !! for turbulent channel flows at high Reynolds numbers with a lattice
  !! Boltzmann method — Application to Coriolis mass flowmeter’, Computers &
  !! Mathematics with Applications. Elsevier Ltd, 78(10), pp. 3285–3302.
  !!
  !! It uses wall model to compute velocity on the boundary node.
  !! All directions of PDF in the boundary elements are updated with
  !! Equilibrium.
  !! Density is computed using Zho-He approach for straight walls.
  !! "On pressure and velocity boundary conditions for the lattice Boltzmann
  !!  BGK model", Physics of Fluids 9, 1591-1598 (1997)
  !! https://doi.org/10.1063/1.869307
  !!
  !! Usage
  !! -----
  !!```lua
  !!boundary_condition = {
  !!  {
  !!    label = 'wall',
  !!    kind = 'turbulent_wall_eq',
  !!    wall_model = 'musker',
  !!    nonlinear_solver = 'fixed_point'
  !!  }
  !!}
  !!
  !! This subroutine's interface must match the abstract interface definition
  !! [[boundaryRoutine]] in bc/[[mus_bc_header_module]].f90 in order to be
  !! callable via [[boundary_type:fnct]] function pointer.
  subroutine turbulent_wall_eq( me, state, bcBuffer, globBC, levelDesc, tree, &
    &          nSize, iLevel, sim_time, neigh, layout, fieldProp, varPos,     &
    &          nScalars, varSys, derVarPos, physics, iField, mixture )
    ! ------------------------------------------------------------------------ !
    !> global boundary type
    class(boundary_type) :: me
    !> Current state vector of iLevel
    real(kind=rk), intent(inout) :: state(:)
    !> size of state array ( in terms of elements )
    integer, intent(in) :: nSize
    !> state values of boundary elements of all fields of iLevel
    real(kind=rk), intent(in) :: bcBuffer(:)
    !> iLevel descriptor
    type(tem_levelDesc_type), intent(in) :: levelDesc
    !> Treelm Mesh
    type(treelmesh_type), intent(in) :: tree
    !> fluid parameters and properties
    type(mus_field_prop_type), intent(in) :: fieldProp
    !> stencil layout information
    type(mus_scheme_layout_type), intent(in) :: layout
    !> the level On which this boundary was invoked
    integer, intent(in) :: iLevel
    !> connectivity array corresponding to state vector
    integer, intent(in) :: neigh(:)
    !> global time information
    type(tem_time_type), intent(in)  :: sim_time
    !> pointer to field variable in the state vector
    integer, intent(in) :: varPos(:)
    !> number of Scalars in the scheme var system
    integer, intent(in) :: nScalars
    !> scheme variable system
    type(tem_varSys_type), intent(in) :: varSys
    !> position of derived quantities in varsys
    type(mus_derVarPos_type), intent(in) :: derVarPos
    !> scheme global boundary type
    type(glob_boundary_type), intent(in) :: globBC
    !> contains physics conversion factors
    type(mus_physics_type), intent(in) :: physics
    !> current field
    integer, intent(in) :: iField
    !> mixture info
    type(mus_mixture_type), intent(in) :: mixture
    ! ------------------------------------------------------------------------ !
    integer :: iElem, iDir, QQ, elemPos, normalInd_inv
    real(kind=rk) :: f_pre(layout%fStencil%QQ)
    real(kind=rk) :: velSW(3)
    real(kind=rk) :: dens_bnd, vel_normal, normal(3)
    real(kind=rk) :: fEq_bnd(layout%fStencil%QQ)
    real(kind=rk) :: unitSW(3, globBC%nElems(iLevel))
    real(kind=rk) :: velSW_bnd(globBC%nElems(iLevel))
    ! ------------------------------------------------------------------------ !
    ! Load stencil for velocity space (DdQq) with q=QQ
    QQ = layout%fStencil%QQ

    ! Calculate stream-wise velocity component from wall function
    call calcVelSW_unitSW_velTau_tVisc(                           &
      & velSW          = velSW_bnd,                               &
      & unitSW         = unitSW,                                  &
      & turbwallFunc   = me%turbwallFunc,                         &
      & nElems         = globBC%nElems(iLevel),                   &
      & elemPos        = globBC%elemLvl(iLevel)%elem%val(:),      &
      & neighBufferPre = me%neigh(iLevel)%neighBufferPre_nNext,   &
      & viscKine       = fieldProp%fluid%viscKine,                &
      & turbulence     = fieldProp%fluid%turbulence,              &
      & stencil        = layout%fStencil,                         &
      & iLevel         = iLevel,                                  &
      & quantities     = layout%quantities                        )

    ! Velocity correction on boundary element incoming direction
    do iElem = 1, globBC%nElems(iLevel)
      ! Current element position in state array. (Used for state in step 10.)
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)

      ! stream-wise velocity on boundary element from wall model
      velSW = velSW_bnd(iElem) * unitSW(:, iElem)

      ! calculate density at boundary for straight wall using Zho-He approach
      normalInd_inv = layout%fStencil%cxDirInv( globBC%elemLvl(iLevel) &
        &                   %normalInd%val(iElem) )
      normal = layout%fStencil%cxDirRK(:, normalInd_inv)
      vel_normal = dot_product(velSW, normal)
      dens_bnd = 0.0_rk
      f_pre = me%neigh(iLevel)%computeNeighBuf( (iElem-1)*QQ+1: iElem*QQ )
      do iDir = 1, layout%fStencil%QQ
        if ( globBC%elemLvl(iLevel)%bitmask%val( iDir, iElem )) then
          dens_bnd = dens_bnd + f_pre(layout%fStencil%cxDirInv(iDir))
        else
          dens_bnd = dens_bnd + f_pre(iDir)
        end if
      end do
      dens_bnd = dens_bnd / (1.0_rk + vel_normal)

      ! Equilibrium on wall model velocity
      fEq_bnd(:) = layout%quantities%pdfEq_ptr( rho = dens_bnd,    &
        &                                       vel = velSW,       &
        &                                       QQ = QQ            )

      do iDir = 1, layout%fStencil%QQ
        state(  neigh(( idir-1)*nsize+ elempos)+( ifield-1)* qq+ nscalars*0) &
          & = fEq_bnd(iDir)
      end do
    end do

    ! Calculated bndForce on boundary elements using momentum exchange method
    call calcTurbWallBndForceAndMoment(                            &
      & bndForce    = me%turbWallFunc%dataOnLvl(iLevel)%bndForce,  &
      & bndMoment   = me%turbWallFunc%dataOnLvl(iLevel)%bndMoment, &
      & momRefPnt   = me%bndMomRefPnt,                             &
      & globBC      = globBC,                                      &
      & baryOfTotal = levelDesc%baryOfTotal,                       &
      & bcBuffer    = bcBuffer,                                    &
      & state       = state,                                       &
      & nSize       = nSize,                                       &
      & neigh       = neigh,                                       &
      & layout      = layout,                                      &
      & physics     = physics,                                     &
      & nScalars    = nScalars,                                    &
      & varPos      = varPos,                                      &
      & iLevel      = iLevel,                                      &
      & iField      = iField                                       )

  end subroutine turbulent_wall_eq
  ! ************************************************************************** !

  ! ************************************************************************** !
  !> BC routine for turbulent wall based on equilibrium BC.
  !! The implementation is based on the following paper:
  !! Haussmann, M. et al. (2019) ‘Large-eddy simulation coupled with wall models
  !! for turbulent channel flows at high Reynolds numbers with a lattice
  !! Boltzmann method — Application to Coriolis mass flowmeter’, Computers &
  !! Mathematics with Applications. Elsevier Ltd, 78(10), pp. 3285–3302.
  !!
  !! It uses wall model to compute velocity on the boundary node.
  !! All directions of PDF in the boundary elements are updated with
  !! Equilibrium plus non-equilibrium. Density is commputed
  !! from PDF on neighbor and extrapolated to boundary.
  !! This routine is used for curved boundaries.
  !!
  !! Usage
  !! -----
  !!```lua
  !!boundary_condition = {
  !!  {
  !!    label = 'wall',
  !!    kind = 'turbulent_wall_eq',
  !!    curved = true,
  !!    wall_model = 'musker',
  !!    nonlinear_solver = 'fixed_point'
  !!  }
  !!}
  !!
  !! This subroutine's interface must match the abstract interface definition
  !! [[boundaryRoutine]] in bc/[[mus_bc_header_module]].f90 in order to be
  !! callable via [[boundary_type:fnct]] function pointer.
  subroutine turbulent_wall_eq_curved( me, state, bcBuffer, globBC,     &
    &          levelDesc, tree, nSize, iLevel, sim_time, neigh, layout, &
    &          fieldProp, varPos, nScalars, varSys, derVarPos, physics, &
    &          iField, mixture )
    ! ------------------------------------------------------------------------ !
    !> global boundary type
    class(boundary_type) :: me
    !> Current state vector of iLevel
    real(kind=rk), intent(inout) :: state(:)
    !> size of state array ( in terms of elements )
    integer, intent(in) :: nSize
    !> state values of boundary elements of all fields of iLevel
    real(kind=rk), intent(in) :: bcBuffer(:)
    !> iLevel descriptor
    type(tem_levelDesc_type), intent(in) :: levelDesc
    !> Treelm Mesh
    type(treelmesh_type), intent(in) :: tree
    !> fluid parameters and properties
    type(mus_field_prop_type), intent(in) :: fieldProp
    !> stencil layout information
    type(mus_scheme_layout_type), intent(in) :: layout
    !> the level On which this boundary was invoked
    integer, intent(in) :: iLevel
    !> connectivity array corresponding to state vector
    integer, intent(in) :: neigh(:)
    !> global time information
    type(tem_time_type), intent(in)  :: sim_time
    !> pointer to field variable in the state vector
    integer, intent(in) :: varPos(:)
    !> number of Scalars in the scheme var system
    integer, intent(in) :: nScalars
    !> scheme variable system
    type(tem_varSys_type), intent(in) :: varSys
    !> position of derived quantities in varsys
    type(mus_derVarPos_type), intent(in) :: derVarPos
    !> scheme global boundary type
    type(glob_boundary_type), intent(in) :: globBC
    !> contains physics conversion factors
    type(mus_physics_type), intent(in) :: physics
    !> current field
    integer, intent(in) :: iField
    !> mixture info
    type(mus_mixture_type), intent(in) :: mixture
    ! ------------------------------------------------------------------------ !
    integer :: iElem, iDir, QQ, elemPos
    real(kind=rk) :: f_neigh(layout%fStencil%QQ)
    real(kind=rk) :: dens_neigh, velSW(3)
    real(kind=rk) :: fEq_bnd(layout%fStencil%QQ)
    real(kind=rk) :: unitSW(3, globBC%nElems(iLevel))
    real(kind=rk) :: velSW_bnd(globBC%nElems(iLevel))
    ! ------------------------------------------------------------------------ !
    ! Load stencil for velocity space (DdQq) with q=QQ
    QQ = layout%fStencil%QQ

    ! Calculate stream-wise velocity component from wall function
    call calcVelSW_unitSW_velTau_tVisc(                           &
      & velSW          = velSW_bnd,                               &
      & unitSW         = unitSW,                                  &
      & turbwallFunc   = me%turbwallFunc,                         &
      & nElems         = globBC%nElems(iLevel),                   &
      & elemPos        = globBC%elemLvl(iLevel)%elem%val(:),      &
      & neighBufferPre = me%neigh(iLevel)%neighBufferPre_nNext,   &
      & viscKine       = fieldProp%fluid%viscKine,                &
      & turbulence     = fieldProp%fluid%turbulence,              &
      & stencil        = layout%fStencil,                         &
      & iLevel         = iLevel,                                  &
      & quantities     = layout%quantities                        )

    ! Velocity correction on boundary element incoming direction
    do iElem = 1, globBC%nElems(iLevel)
      ! Current element position in state array. (Used for state in step 10.)
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)
      ! Compute density and velocity from pre-collision state on neighbor
      f_neigh = me%neigh(iLevel)%neighBufferPre_nNext( 1,         &
        &                          (iElem-1)*QQ+1:(iElem-1)*QQ+QQ )
      dens_neigh = sum(f_neigh)

      ! stream-wise velocity on boundary element from wall model
      velSW = velSW_bnd(iElem) * unitSW(:, iElem)

      ! Equilibrium on wall model velocity
      fEq_bnd(:) = layout%quantities%pdfEq_ptr( rho = dens_neigh, &
        &                                       vel = velSW,      &
        &                                       QQ = QQ           )

      do iDir = 1, layout%fStencil%QQ
        state(  neigh(( idir-1)*nsize+ elempos)+( ifield-1)* qq+ nscalars*0) &
          & = fEq_bnd(iDir)
      end do
    end do

    ! Calculated bndForce on boundary elements using momentum exchange method
    call calcTurbWallBndForceAndMoment(                            &
      & bndForce    = me%turbWallFunc%dataOnLvl(iLevel)%bndForce,  &
      & bndMoment   = me%turbWallFunc%dataOnLvl(iLevel)%bndMoment, &
      & momRefPnt   = me%bndMomRefPnt,                             &
      & globBC      = globBC,                                      &
      & baryOfTotal = levelDesc%baryOfTotal,                       &
      & bcBuffer    = bcBuffer,                                    &
      & state       = state,                                       &
      & nSize       = nSize,                                       &
      & neigh       = neigh,                                       &
      & layout      = layout,                                      &
      & physics     = physics,                                     &
      & nScalars    = nScalars,                                    &
      & varPos      = varPos,                                      &
      & iLevel      = iLevel,                                      &
      & iField      = iField                                       )


  end subroutine turbulent_wall_eq_curved
  ! ************************************************************************** !


  ! ************************************************************************** !
  !> Calculation stream-wise velocity compononent from wall function and
  !! friction velocity, stream-wise unit vector and turbulent viscosity with
  !! mixing length formulation.
  subroutine calcVelSW_unitSW_velTau_tVisc(velSW, unitSW, turbwallFunc, &
    &          nElems, elemPos, neighBufferPre, viscKine, turbulence,   &
    &          stencil, iLevel, quantities)
    ! --------------------------------------------------------------------------
    !> Stream-wise velocity component from wall function
    real(kind=rk), intent(out) :: velSW(:)
    !> Stream-wise unit vector
    real(kind=rk), intent(out) :: unitSW(:,:)
    !> Turbulent wall model type contains viscosity, velTau, distToBnd and
    !! function pointers to compute velTau and velSW
    type(mus_turb_wallFunc_type), intent(inout) :: turbwallFunc
    !> Number of elements in current boundary
    integer, intent(in) :: nElems
    !> Current element position in state array. (Used to access viscosity)
    integer, intent(in) :: elemPos(:)
    !> Pre-collision from 1st and 2nd fluid neighbor
    real(kind=rk), intent(in) :: neighBufferPre(:,:)
    !> Kinematic viscosity
    type(mus_viscosity_type) :: viscKine
    !> turbulence model type
    type(mus_turbulence_type), intent(in) :: turbulence
    !> Fluid stencil
    type(tem_stencilHeader_type), intent(in) :: stencil
    !> Current level
    integer, intent(in) :: iLevel
    !> Class that contains pointers to the proper derived quantities functions
    type(mus_scheme_derived_quantities_type), intent(in) :: quantities
    ! --------------------------------------------------------------------------
    integer :: iElem, QQ
    real(kind=rk) :: f_neigh(stencil%QQ)
    real(kind=rk) :: vel_neigh(3), dens_neigh, vec(3), vec_mag
    real(kind=rk) :: velSW_neigh(nElems)
    real(kind=rk) :: yPlus, gradU, velSW_neigh_2, unitNormal(3)
    ! --------------------------------------------------------------------------
    QQ = stencil%QQ
    ! Initialize unit streamwise to zero. It is set to non-zero value only
    ! if vec_mag is not zero
    unitSW = 0.0_rk
    ! Calculate stream-wise velocity component on first neighbor
    do iElem = 1, nElems
      ! Get density and velocity on second fluid element along the normal
      ! direction i.e. 1st neighbor.
      f_neigh = neighBufferPre(1, (iElem-1)*QQ+1:(iElem-1)*QQ+QQ)
      dens_neigh = sum(f_neigh)

      ! Get velocity from pdf, inverse of density and stencil as input.
      vel_neigh = quantities%vel_from_pdf_ptr( pdf = f_neigh, dens = dens_neigh )

      ! Calculate local stream-wise unit vector
      ! e' = (u - (u . n) . n) / |(u - (u . n) . n|
      !
      unitNormal = turbwallFunc%dataOnLvl(iLevel)%unitNormal(:, iElem)
      vec = vel_neigh - dot_product( vel_neigh, unitNormal ) * unitNormal
      vec_mag = sqrt(dot_product(vec, vec))

      if ( tem_isnan(vec_mag) ) then
        call tem_abort("Error: In calcVelSW_unitSW_velTau_tVisc. " &
          &          //"Stream-wise velocity mag is NaN.")
      end if
      ! Unit vector
      if (vec_mag .fne. 0.0_rk) then
        unitSW(:, iElem) = vec / vec_mag
      end if
      ! stream-wise velocity component
      velSW_neigh(iElem) = dot_product(vel_neigh, unitSW(:, iElem))
      if (velSW_neigh(iElem) < 0.0_rk) then
        call tem_abort("Error: In calcVelSW_unitSW_velTau_tVisc. " &
          &          //"Stream-wise velocity component is negative.")
      end if
    end do

    ! calculate friction velocity on neighbor element
    call turbwallFunc%calcFricVel(                                 &
      & velTau    = turbwallFunc%dataOnLvl(iLevel)%velTau,         &
      & velSW     = velSW_neigh,                                   &
      & distToBnd = turbwallFunc%dataOnLvl(iLevel)%neighDistToBnd, &
      & viscKine  = viscKine%dataOnLvl(iLevel)%val( elemPos(:) ),  &
      & nElems    = nElems                                         )

    ! calculate stream-wise velocity on boundary element
    call turbwallFunc%calcStreamWiseVel(                          &
      & velSW     = velSW,                                        &
      & velTau    = turbwallFunc%dataOnLvl(iLevel)%velTau,        &
      & distToBnd = turbwallFunc%dataOnLvl(iLevel)%distToBnd,     &
      & viscKine  = viscKine%dataOnLvl(iLevel)%val( elemPos(:) ), &
      & nElems    = nElems,                                       &
      & wall_function = turbwallFunc%wall_function                )

    ! Calculate Turbulent viscosity according to mixing length formulation
    ! with von karman constant.
    ! only if LES turbulence model is smagorinsky because turb viscosity
    ! from Vreman and WALE reduces towards the wall
    ! nu_t = (k*y)**2 * |dudy|
    if (turbulence%active .and.                       &
      & trim(turbulence%config%model) == 'smagorinsky') then
      do iElem = 1, nElems
        ! Get density and velocity on third fluid element along the normal
        ! direction i.e. 2nd neighbor.
        f_neigh = neighBufferPre(2, (iElem-1)*QQ+1:(iElem-1)*QQ+QQ)
        dens_neigh = sum(f_neigh)

        ! Get velocity from pdf, inverse of density and stencil as input.
        vel_neigh = quantities%vel_from_pdf_ptr( pdf = f_neigh, dens = dens_neigh )
        velSW_neigh_2 = dot_product(vel_neigh, unitSW(:, iElem))

        ! computed velocity gradient from second order forward difference
        gradU = abs( - 3.0_rk * velSW(iElem) + 4.0_rk * velSW_neigh(iElem) &
          & - velSW_neigh_2 ) / 2.0_rk
        turbwallFunc%dataOnLvl(iLevel)%tVisc(iElem) = ( turbWallFunc%vonKarman &
          & * turbwallFunc%dataOnLvl(iLevel)%distToBnd(iElem) )**2.0_rk        &
          & * gradU
      end do

      ! Apply van Driest damping
      ! nu_t = nu_t * (1-exp(-yplus/Aplus))**2
      if (turbWallFunc%useVanDriest) then
        do iElem = 1, nElems
          yPlus = turbwallFunc%dataOnLvl(iLevel)%distToBnd(iElem) &
            & * turbwallFunc%dataOnLvl(iLevel)%velTau(iElem)      &
            & / viscKine%dataOnLvl(iLevel)%val(elemPos(iElem))
          turbwallFunc%dataOnLvl(iLevel)%tVisc(iElem)       &
            & = turbwallFunc%dataOnLvl(iLevel)%tVisc(iElem) &
            & * (1.0_rk-exp(-yPlus/vd_Aplus))**2.0_rk
        end do
      end if
    end if
  end subroutine calcVelSW_unitSW_velTau_tVisc
  ! ************************************************************************** !


  ! ************************************************************************** !
  !> This routine computes bndForce on boundary elements using momentum
  !! exchange method.
  subroutine calcTurbWallBndForceAndMoment(bndForce, bndMoment, momRefPnt,     &
    &                                      globBC, baryOfTotal, bcBuffer,      &
    &                                      state, nSize, neigh, layout,        &
    &                                      physics, nScalars, varPos, iLevel,  &
    &                                      iField)
    ! ------------------------------------------------------------------------ !
    !> Boundary force
    real(kind=rk), intent(out) :: bndForce(:,:)
    !> Boundary moment
    real(kind=rk), intent(out) :: bndMoment(:,:)
    !> Reference point for moment calculation
    real(kind=rk), intent(in) :: momRefPnt(:)
    !> scheme global boundary type
    type(glob_boundary_type), intent(in) :: globBC
    !> barycenter of elements in total list
    !! Size: nElemsTotal, 3
    real(kind=rk), intent(in) :: baryOfTotal(:,:)
    !> state values of boundary elements of all fields of iLevel
    real(kind=rk), intent(in) :: bcBuffer(:)
    !> Current state vector of iLevel
    real(kind=rk), intent(inout) :: state(:)
    !> size of state array ( in terms of elements )
    integer, intent(in) :: nSize
    !> connectivity array corresponding to state vector
    integer, intent(in) :: neigh(:)
    !> stencil layout information
    type(mus_scheme_layout_type), intent(in) :: layout
    !> contains physics conversion factors
    type(mus_physics_type), intent(in) :: physics
    !> number of Scalars in the scheme var system
    integer, intent(in) :: nScalars
    !> pointer to field variable in the state vector
    integer, intent(in) :: varPos(:)
    !> the level On which this boundary was invoked
    integer, intent(in) :: iLevel
    !> current field
    integer, intent(in) :: iField
    ! ------------------------------------------------------------------------ !
    integer :: iElem, iDir, QQ, invDir, elemPos, posInBuffer
    real(kind=rk) :: fIn, fOut, conv2LatLen
    real(kind=rk) :: force(3), delta_x(3)
    ! ------------------------------------------------------------------------ !
    conv2LatLen = 1.0_rk / physics%fac(iLevel)%length
    ! Load stencil for velocity space (DdQq) with q=QQ
    QQ = layout%fStencil%QQ
    do iElem = 1, globBC%nElems(iLevel)
      force = 0.0_rk
      ! Current element position in state array
      elemPos = globBC%elemLvl(iLevel)%elem%val(iElem)
      ! position in bcBuffer
      posInBuffer = globBC%elemLvl(iLevel)%posInBcElemBuf%val( iElem )

      ! arm of the momentum
      delta_x = (baryOfTotal(elemPos, :) - momRefPnt(:)) * conv2LatLen

      do iDir = 1, layout%fStencil%QQN
        if ( globBC%elemLvl(iLevel)%bitmask%val( iDir, iElem )) then
          ! direction towards boundary
          invDir  = layout%fStencil%cxDirInv(iDir)
          ! Incoming state before collision
          fIn = state(                                                   &
            &  neigh((idir-1)* nsize+ elempos)+( ifield-1)* qq+ nscalars*0 )
          ! outgoing state after collision
          fOut = bcBuffer( (posInBuffer-1)*nScalars+varPos(invDir) )
          ! momentum exchange
          force = force + layout%fStencil%cxDirRK(:, invDir) * (fOut + fIn)
        end if
      end do
      bndForce(:, iElem) = force
      ! moment = arm x force
      bndMoment(:, iElem) = cross_product3D(delta_x, force)
    end do
  end subroutine calcTurbWallBndForceAndMoment
  ! ************************************************************************** !

end module mus_bc_fluid_turbulent_module
! *****************************************************************************!