mus_bc_header_module.f90 Source File


This file depends on

sourcefile~~mus_bc_header_module.f90~~EfferentGraph sourcefile~mus_bc_header_module.f90 mus_bc_header_module.f90 sourcefile~mus_scheme_layout_module.f90 mus_scheme_layout_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_dervarpos_module.f90 mus_derVarPos_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_dervarpos_module.f90 sourcefile~mus_turb_wallfunc_module.f90 mus_turb_wallFunc_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_turb_wallfunc_module.f90 sourcefile~mus_physics_module.f90 mus_physics_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_mixture_module.f90 mus_mixture_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_mixture_module.f90 sourcefile~mus_field_prop_module.f90 mus_field_prop_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_field_prop_module.f90 sourcefile~mus_pdf_module.f90 mus_pdf_module.f90 sourcefile~mus_bc_header_module.f90->sourcefile~mus_pdf_module.f90 sourcefile~mus_moments_type_module.f90 mus_moments_type_module.f90 sourcefile~mus_scheme_layout_module.f90->sourcefile~mus_moments_type_module.f90 sourcefile~mus_dervarpos_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_mixture_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_scheme_header_module.f90 mus_scheme_header_module.f90 sourcefile~mus_mixture_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_enrtl_module.f90 mus_eNRTL_module.f90 sourcefile~mus_mixture_module.f90->sourcefile~mus_enrtl_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_poisson_module.f90 mus_poisson_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_poisson_module.f90 sourcefile~mus_fluid_module.f90 mus_fluid_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_fluid_module.f90 sourcefile~mus_species_module.f90 mus_species_module.f90 sourcefile~mus_field_prop_module.f90->sourcefile~mus_species_module.f90 sourcefile~mus_poisson_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_pdf_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_nonnewtonian_module.f90 mus_nonNewtonian_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_nonnewtonian_module.f90 sourcefile~mus_relaxationparam_module.f90 mus_relaxationParam_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_relaxationparam_module.f90 sourcefile~mus_turb_viscosity_module.f90 mus_turb_viscosity_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_turb_viscosity_module.f90 sourcefile~mus_turbulence_module.f90 mus_turbulence_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_turbulence_module.f90 sourcefile~mus_cumulantinit_module.f90 mus_cumulantInit_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_cumulantinit_module.f90 sourcefile~mus_mrtrelaxation_module.f90 mus_mrtRelaxation_module.f90 sourcefile~mus_fluid_module.f90->sourcefile~mus_mrtrelaxation_module.f90 sourcefile~mus_species_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_nonnewtonian_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_nonnewtonian_module.f90->sourcefile~mus_physics_module.f90 sourcefile~mus_nonnewtonian_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_scheme_layout_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_header_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_moments_type_module.f90 sourcefile~mus_relaxationparam_module.f90->sourcefile~mus_nonnewtonian_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_turb_viscosity_module.f90->sourcefile~mus_scheme_layout_module.f90 sourcefile~mus_turb_viscosity_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_turb_viscosity_module.f90->sourcefile~mus_turbulence_module.f90 sourcefile~mus_wale_module.f90 mus_WALE_module.f90 sourcefile~mus_turb_viscosity_module.f90->sourcefile~mus_wale_module.f90 sourcefile~mus_vreman_module.f90 mus_Vreman_module.f90 sourcefile~mus_turb_viscosity_module.f90->sourcefile~mus_vreman_module.f90 sourcefile~mus_smagorinsky_module.f90 mus_Smagorinsky_module.f90 sourcefile~mus_turb_viscosity_module.f90->sourcefile~mus_smagorinsky_module.f90 sourcefile~mus_turb_viscosity_module.f90->sourcefile~mus_graddata_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_mrtrelaxation_module.f90->sourcefile~mus_scheme_header_module.f90 sourcefile~mus_mrtrelaxation_module.f90->sourcefile~mus_moments_type_module.f90

Files dependent on this one

sourcefile~~mus_bc_header_module.f90~~AfferentGraph sourcefile~mus_bc_header_module.f90 mus_bc_header_module.f90 sourcefile~mus_bc_passivescalar_module.f90 mus_bc_passiveScalar_module.f90 sourcefile~mus_bc_passivescalar_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_weights_module.f90 mus_weights_module.f90 sourcefile~mus_weights_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bndforce_module.f90 mus_bndForce_module.f90 sourcefile~mus_bndforce_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_fluid_noneqexpol_module.f90 mus_bc_fluid_nonEqExpol_module.f90 sourcefile~mus_bc_fluid_noneqexpol_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_dynloadbal_module.f90 mus_dynLoadBal_module.f90 sourcefile~mus_dynloadbal_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_species_module.f90 mus_bc_species_module.f90 sourcefile~mus_bc_species_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_general_module.f90 mus_bc_general_module.f90 sourcefile~mus_bc_general_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_construction_module.f90 mus_construction_module.f90 sourcefile~mus_construction_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_var_module.f90 mus_bc_var_module.f90 sourcefile~mus_bc_var_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_fluid_wall_module.f90 mus_bc_fluid_wall_module.f90 sourcefile~mus_bc_fluid_wall_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_poisson_module.f90 mus_bc_poisson_module.f90 sourcefile~mus_bc_poisson_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_fluid_experimental_module.f90 mus_bc_fluid_experimental_module.f90 sourcefile~mus_bc_fluid_experimental_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_field_module.f90 mus_field_module.f90 sourcefile~mus_field_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90 mus_bc_fluid_turbulent_module.f90 sourcefile~mus_bc_fluid_turbulent_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_nernstplanck_module.f90 mus_bc_nernstPlanck_module.f90 sourcefile~mus_bc_nernstplanck_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_bc_fluid_module.f90 mus_bc_fluid_module.f90 sourcefile~mus_bc_fluid_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_connectivity_module.f90 mus_connectivity_module.f90 sourcefile~mus_connectivity_module.f90->sourcefile~mus_bc_header_module.f90 sourcefile~mus_scheme_type_module.f90 mus_scheme_type_module.f90 sourcefile~mus_scheme_type_module.f90->sourcefile~mus_bc_header_module.f90

Contents


Source Code

! Copyright (c) 2012-2021 Kannan Masilamani <kannan.masilamani@uni-siegen.de>
! Copyright (c) 2022 Kannan Masilamani <kannan.masilamani@dlr.de>
! Copyright (c) 2012-2013 Manuel Hasert <m.hasert@grs-sim.de>
! Copyright (c) 2012-2013 Simon Zimny <s.zimny@grs-sim.de>
! Copyright (c) 2012, 2015, 2021 Harald Klimach <harald.klimach@uni-siegen.de>
! Copyright (c) 2012, 2014-2016 Jiaxing Qi <jiaxing.qi@uni-siegen.de>
! Copyright (c) 2014 Kartik Jain <kartik.jain@uni-siegen.de>
! Copyright (c) 2015-2016 Tobias Schneider <tobias1.schneider@student.uni-siegen.de>
! Copyright (c) 2017 Sindhuja Budaraju <nagasai.budaraju@student.uni-siegen.de>
! Copyright (c) 2017 Raphael Haupt <raphael.haupt@uni-siegen.de>
! Copyright (c) 2018,2020 Jana Gericke <jana.gericke@uni-siegen.de>
! Copyright (c) 2019 Seyfettin Bilgi <seyfettin.bilgi@student.uni-siegen.de>
! Copyright (c) 2019-2020 Peter Vitt <peter.vitt2@uni-siegen.de>
! Copyright (c) 2021-2022 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.
! **************************************************************************** !
!> This module contains boundary type definitions, different boundary treatment
!! types, abtract interface and routine to load boundary table from config file
!!
!! The boundary conditions are stored in boundary type definitions for each
!! boundary which is defined in the main lua file.
!!```lua
!! boundary_condition = {  }
!!```
!!
!! The definitions there must equal the boundary definition within the mesh on
!! disk, in terms of the amount of boundary conditions and the labels.
!! A detailed description on the implementation details are given in
!! [[tem_bc_module]] "boundary condition implementation"
!!
!! This module will be used as a header to use boundary definitions in other
!! modules.
!!
! 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.
module mus_bc_header_module

  ! include treelm modules
  use env_module,               only: rk, LabelLen
  use tem_aux_module,           only: tem_abort
  use tem_float_module,         only: operator(.feq.)
  use tem_property_module,      only: prp_solid
  use tem_bc_module,            only: tem_bc_state_type, tem_load_bc_state
  use tem_bc_header_module,     only: tem_load_bc_header, tem_bc_header_type
  use tem_bc_prop_module,       only: tem_bc_prop_type
  use tem_dyn_array_module,     only: init, destroy, dyn_intArray_type, &
    &                                 dyn_longArray_type
  use tem_grow_array_module,    only: init, destroy, grw_int2darray_type, &
    &                                 grw_logical2darray_type,            &
    &                                 grw_intarray_type,                  &
    &                                 grw_real2darray_type
  use tem_tools_module,         only: tem_horizontalSpacer
  use tem_time_module,          only: tem_time_type
  use treelmesh_module,         only: treelmesh_type
  use tem_varSys_module,        only: tem_varSys_type
  use tem_logging_module,       only: logUnit
  use tem_debug_module,         only: dbgUnit
  use tem_construction_module,  only: tem_levelDesc_type
  use tem_geometry_module,      only: tem_BaryOfId,tem_ElemSizeLevel
  use tem_stencil_module,       only: tem_stencilHeader_type
  use tem_stringKeyValuePair_module, only: init, truncate,                &
    &                                      grw_stringKeyValuePairArray_type

  ! include aotus modules
  use aotus_module,     only: flu_State, aoterr_Fatal, aoterr_NonExistent, &
    &                         aoterr_WrongType, aot_top_get_val, aot_get_val
  use aot_table_module, only: aot_table_open, aot_table_close

  ! include musubi modules
  use mus_field_prop_module,     only: mus_field_prop_type
  use mus_scheme_layout_module,  only: mus_scheme_layout_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_pdf_module,            only: pdf_data_type
  use mus_turb_wallFunc_module,  only: mus_turb_wallFunc_type, &
    &                                  mus_load_turb_wallFunc

  implicit none

  private

  public :: boundary_type
  public :: glob_boundary_type
  public :: mus_load_bc
  public :: mus_init_bc_elems
  public :: check_solid_in_bc
  public :: rearrange_bc_elems
  public :: debug_glob_boundary_type
  public :: mus_set_posInNghElems
  public :: mus_set_bcLinks
  public :: mus_set_bouzidi
  public :: mus_alloc_bouzidi
  public :: mus_alloc_fieldBC
  public :: mus_set_inletUbb
  public :: mus_set_inletBfl
  public :: mus_set_nonEqExpol
  public :: mus_set_outletExpol
  public :: mus_fieldBC_cleanup

  !> information needed for moments based boundary condition
  type bc_moments_type
    !> Number of unknown state links to update
    integer :: nUnKnownPdfs
    !> known moments position in moments vector
    integer, allocatable :: knownMom_pos(:)
    !> inverse matrix of unknown pdf matrix
    real(kind=rk), allocatable :: unKnownPdfs_MatInv(:,:)

  end type bc_moments_type


  !> Level wise boundary elements information
  type bc_elems_type

    !> Positions in levelDesc total list
    !! Its purpose is to get the treeID of BC elements
    !! size: globBC%nElems
    !! to use: levelDesc(iLevel)%total( globBC%elemLvl(iLevel)%elem%val(iElem) )
    type( dyn_intArray_type ) :: elem

    !> Position of this boundary in the bc_elemBuffer
    !! bc_elemBuffer is growing array
    !! initial size: globBC%nElems
    !! It is initiated in mus_init_bc_elems of mus_bc_header_module
    !! Only non-wall BC needs elemBuffer
    type ( grw_intarray_type ) :: posInBcElemBuf

    !> Normal vector for each boundary element pointing into the domain.
    !! normal is a growing array with size: normal%val(3, nElems)
    type ( grw_int2darray_type ) :: normal

    !> which is the index in the stencil corresponding to the normal vector?
    type ( grw_intarray_type ) :: normalInd

    !> bit mask for each node holding directions which have to be updated.
    !! The bitmask points into the incoming direction into the flow domain,
    !!  which actually we want to update
    !! * For PUSH, we write to the incoming position,
    !!   as the kernel reads it from there without propagation.
    !! * For PULL, we need to write to the inverse direction, as the kernel
    !!   performs a bounce back before reading it.
    !!   However, this bounced back direction actually comes from the
    !!   non-existent boundary element and would point into the incoming
    !!   direction, so the value has to be treated and set as if it points
    !!   really into the incoming direction.
    !! bitmask is a growing array, the values are in bitmask%val(:,:)
    !! 1st index size is QQN since center is not treated for bcElems
    type ( grw_logical2darray_type ) :: bitmask

    !> The q-Values for the exact wall positions
    !! Its size: QQN, nElemsBC
    !! It is allocated in routine: allocateBCList
    !! assigned in routine: assignBCList
    type( grw_real2darray_type ) ::  qVal

  end type bc_elems_type


  !> contains information needed to treat corner nodes or nodes intersected
  !! by more than one boundary type
  type mus_bc_corner_type
    !> number of local boundary elements per level
    integer, allocatable :: nElems(:)
    !> boundary neighbor in different refinement level
    type( bc_elems_type ), allocatable :: elemLvl(:)
    !> boundary id in each direction for each corner node
    !! size layout%stencil%QQ, nElems
    integer, allocatable :: bcid(:,:)
  end type mus_bc_corner_type

  !> Wall bouzidi data for one level
  type bc_wall_bouzidi_type

    !> size: links(iLevel)%nVals
    real(kind=rk), allocatable :: qVal(:)
    real(kind=rk), allocatable ::  cIn(:)
    real(kind=rk), allocatable :: cOut(:)
    real(kind=rk), allocatable :: cNgh(:)

    !> size: links(iLevel)%nVals
    !! set in routine: mus_set_bouzidi
    integer, allocatable ::  inPos(:) ! position in bcBuffer
    integer, allocatable :: outPos(:) ! position in bcBuffer
    integer, allocatable :: nghPos(:) ! position in computeNeighBuf

  end type bc_wall_bouzidi_type


  !> Provides bouzidi coefficients and q-values needed for link-wise
  !! implementation of certain inlet boundary conditions.
  type bc_inlet_bouzidi_type

    !> size: links(iLevel)%nVals
    real(kind=rk), allocatable :: qVal(:) ! q-Values
    real(kind=rk), allocatable ::  cIn(:) ! coefficient for incoming directions
    real(kind=rk), allocatable :: cOut(:) ! coefficient for outgoing directions
    real(kind=rk), allocatable :: cNgh(:) ! coefficient for neighbours

    !> size: links(iLevel)%nVals
    !! set in routine: mus_set_bouzidi
    integer, allocatable ::  inPos(:) ! position in bcBuffer
    integer, allocatable :: outPos(:) ! position in bcBuffer
    integer, allocatable :: nghPos(:) ! position in computeNeighBuf

    !> size: links(iLevel)%nVals
    real(kind=rk), allocatable ::    cVel(:) ! coefficient for eqPlus

    !> size: links(iLevel)%nVals
    !! set in routine: mus_set_bouzidi
    integer, allocatable :: iDir(:) ! incoming direction from boundary

    integer, allocatable ::   posInBuffer(:) !< used for position in bcBuffer
  end type bc_inlet_bouzidi_type


  !> Provides coefficients needed for link-wise implementation of
  !! nonEquilibrium extrapolation scheme for boundary conditions.
  type bc_nonEqExpol_type

    !> size: links(iLevel)%nVals
    real(kind=rk), allocatable ::      c_w(:) ! coefficient for surface
    real(kind=rk), allocatable ::      c_f(:) ! coefficient for fluid
    real(kind=rk), allocatable ::     c_ff(:) ! coefficient for overnext fluid
    real(kind=rk), allocatable ::  c_neq_f(:) ! nonEq coefficient for fluid
    real(kind=rk), allocatable :: c_neq_ff(:) ! nonEq coeff. for overnext fluid

    !> size: links(iLevel)%nVals
    !! set in routine: mus_set_bouzidi
    integer, allocatable ::          iDir(:) ! incoming direction
    integer, allocatable ::   posInBuffer(:) ! used for position in bcBuffer
    ! Position of fluid neigh in incoming direction in neighBuffer
    integer, allocatable :: posInNeighBuf(:)
    integer, allocatable :: posInBCelems(:)

  end type bc_nonEqExpol_type


  !> Provides needed positions for link-wise implementation of outlet boundary
  !> condition outlet_expol
  type bc_outlet_type

    !> size: links(iLevel)%nVals
    integer, allocatable ::  statePos(:) ! position of state
    integer, allocatable ::  iElem(:) ! position of state

  end type bc_outlet_type


  !> variable definition for non-reflective type of boundary conditions\n
  !! These boundary condition is taken from the paper:
  !!```lua
  !!   S. Izquierdo and N. Fueyo,
  !!   "Characteristic nonreflecting boundary
  !!   conditions for open boundaries in lattice Boltzmann methods,"
  !!   Physical Review E, vol. 78, no. 46707, 2008.
  !!```
  type bc_nrbc_type

    !> specific heat ratio
    !! appears in eq 3: cs = sqrt( kappa * R * T )
    real(kind=rk) :: kappa

    !> from eq 17: k = sigma * ( 1 - Ma^2 ) * cs / L
    real(kind=rk) :: K_mod

    !> used in eq 17
    real(kind=rk) :: cs_mod

    !> used in eq 17
    real(kind=rk) :: sigma

    !> length between inlet and outlet, represented by L in the paper
    real(kind=rk) :: lodi_length

    !> Lattice Mach number characterised by lattice flow velocity
    real(kind=rk) :: Ma_L

  end type bc_nrbc_type


  !> Contains variables for black-box membrane boundary condition
  type bc_blackbox_membrane_type
    !> type of membrance (AEM or CEM)
    character(len=labelLen) :: label
    !> transference number defining transfer property
    !! of ionic species on the membrance
    !! KM @todo check transNr constrain
    !!          sumOfIonicSpecies(transNr) = 1
    real(kind=rk) :: transNr
    !> osmotic solvent transport coefficient
    real(kind=rk) :: solvTrans_coeff
  end type bc_blackbox_membrane_type


  !> contains all possible bc state variables needed for boundary
  type bc_states_type

    !> Velocity at boundary
    type(tem_bc_state_type) :: velocity

    !> mass flow rate at boundary node for mfr (mass flow rate)
    !! boundary condition
    type(tem_bc_state_type) :: massFlowRate

    !> Pressure at boundary
    type(tem_bc_state_type) :: pressure

    !> mole fraction for species boundary
    type(tem_bc_state_type) :: moleFrac

    !> molar flux for species (c_i*u_i)
    type(tem_bc_state_type) :: moleFlux

    !> mole density for species (c_i)
    type(tem_bc_state_type) :: moleDens

    !> molar diffusion flux for species
    type(tem_bc_state_type) :: moleDiff_flux

    !> probability density function for boundary
    type(tem_bc_state_type) :: pdf

    !> potential for Poisson boundary
    type(tem_bc_state_type) :: potential

    !> surface charge density for Poisson Neumann boundary
    type(tem_bc_state_type) :: surChargeDens

  end type bc_states_type


  !> Simple array type which include array and its size
  type array_type

    !> array
    integer, allocatable :: val(:)

    !> size
    integer :: nVals

  end type array_type


  !> Information about boundary elements
  !!
  !! This derived type encapsulates the definition of a boundary
  !! with all necessary parameters and data fields
  type glob_boundary_type

    !> name of this boundary
    character(len=LabelLen) :: label

    !> elements list of this boundary in different refinement level
    !! size: minLevel -> maxLevel
    type( bc_elems_type ), allocatable :: elemLvl(:)

    !> Collect corner BC i.e elements which are interesected by multiple
    !! boundaries only for moments based BC
    logical :: treat_corner

    !> list corner boundary elements
    type( mus_bc_corner_type ) :: cornerBC

    !> number of local (this process) boundary elements per level
    !! including ghostFromCoarser.
    !! GhostFromFiner are interpolated but ghostFromCoarser needs
    !! correct boundary value in fine level sub-iteration.
    integer, allocatable :: nElems(:)

    !> number of local (this process) boundary elements per level
    !! wihtout Ghost elements
    integer, allocatable :: nElems_Fluid(:)

    !> number of total (total processes) boundary elements per level
    integer, allocatable :: nElems_totalLevel(:)

    !> number of local boundary elements
    integer :: nElems_local = 0

    !> number of total boundary elements
    integer :: nElems_total = 0

    !> whether this boundary requires qVal
    logical :: hasQVal = .false.

    !> has qVal initialized with default qVal = 0.5
    logical :: qValInitialized = .false.

    !> if this is a wall or symmetry BC (implicit treated during streaming)
    logical :: isWall = .false.

    !> Average normal direction of this boundary along layout%prevailDir
    !! Normal in elemLvl is normal direction per element and this
    !! is normal per boundary
    integer :: normal(3)

    !> which is the index in the stencil corresponding to the normal vector?
    integer :: normalInd
  end type glob_boundary_type


  !> Level wise information about neighbor of boundary elements
  !! for a single field
  type bc_neigh_type

    !> Neighbor position in level-wise state array
    !! size: nNeighs, nElems
    !! allocated in routine: setFieldBCNeigh, mus_construction_module
    integer, allocatable :: posInState(:,:)

    !> Pre-collision state values of neighbors on normal direction on next
    !! time step
    !! size: nNeighs, nElems*stencil%QQ
    !! It is allocated in routine update_BClists
    real(kind=rk), allocatable :: neighBufferPre_nNext(:,:)

    !> Pre-collision state values of neighbors on normal direction on previous
    !! time step
    !! size: nNeighs, nElems*stencil%QQ
    !! It is allocated in routine update_BClists
    real(kind=rk), allocatable :: neighBufferPre(:,:)

    !> Post-collision state values of neighbors on normal direction
    !! size: nNeighs, nElems*stencil%QQ
    !! It is allocated in routine update_BClists
    !! It is filled in fill_neighBuffer
    real(kind=rk), allocatable :: neighBufferPost(:,:)

    !> Post-collision state values of neighbors on all directions at
    !! current time step.
    !! Its also a Pre-collision state values of an element next to boundary
    !! at next time step
    !! size: nElems * computeStencil%QQ
    !! It is allocated in routine update_BClists (mus_construction_module),
    !! filled in routine: fill_computeNeighBuf (mus_bc_general_module).
    !! it use AOS layout always!
    real(kind=rk), allocatable :: computeNeighBuf(:)

  end type bc_neigh_type


  !> Contains field specific element information for each level
  type bc_field_elems_type
    !> Position of stencil of each element in array of stencils
    !! in layout.
    !! Unique stencil label for boundary stencils are created with boundary label
    !! and stencil%cxDir therefore each stencil is limited to one boundary type
    !! Dimension: globBC%nElems(iLevel)
    !! allocated in mus_build_BCStencils (mus_construction_module)
    integer, allocatable :: stencilPos(:)

    !> LODI array for the NRBC
    !! Dimension: 4, 3, globBC%nElems(iLevel)
    !! allocated in init_nrbc routine
    real(kind=rk), allocatable :: lodi(:,:,:)

    !> defined moments position in moments matrix and
    !! inverse of unknown pdf matrix
    !! Dimension: bc%nElems(iLevel)
    type( bc_moments_type ), allocatable :: moments(:)

    !> Position in LevelDesc%neigh%NghElems
    !! size: globBC%nElems(iLevel)
    !! allocated in mus_build_BCStencils routine, mus_construction_module
    integer, allocatable :: posInNghElems(:)

  end type bc_field_elems_type


  !> Boundary treatment data type.
  !! It include boundary treatment for only ONE field.
  !! For the same boundary elements, each field may have a different treatment.
  !! It includes function pointers and neighbor elements required by this
  !! boundary treatment
  type boundary_type

    !> Boundary ID
    integer :: bc_id

    !> kind of this boundary
    character(len=LabelLen) :: BC_kind

    !> name of this boundary
    character(len=LabelLen) :: label

    !> number of neighbors fluid element needed for extrapolation to
    !! get data at boundary element
    integer :: nNeighs =  0

    !> Curved boundaries are treated with qValues and extrapolation of
    !! nonEq from 2nd neighbor
    logical :: curved = .false.

    !> Level wise neighbor information of boundary elements
    !! Each field may have a different handlement on the same boundary
    !! Thus requires a different neighborhood
    !! allocated in mus_alloc_fieldBC
    type(bc_neigh_type), allocatable :: neigh(:)

    !> Level wise element information of boundary elements
    !! like stencil position in array of stencils,
    !! moments type for momentBC and LODI array for NRBC
    !! allocated in mus_alloc_fieldBC
    type(bc_field_elems_type), allocatable :: elemLvl(:)

    !> function pointer to boundary routine to call
    procedure(boundaryRoutine), pointer :: fnct => null()

    !> function pointer to compute bndForce on wall
    procedure(mus_proc_calcBndForce), pointer :: calcBndForce => null()

    !> musubi bc state variables
    type(bc_states_type) :: BC_states

    !> Dictionary of boundary variables with
    !! varDict%val()%key is the name of boundary variable and
    !! varDict%val()%value is the name of spacetime function variable
    type(grw_stringKeyValuePairArray_type) :: varDict

    !> order of boundary which determines number of neighbor need for this boundary
    integer :: order

    !> standard q-value for higher order boundaries without any definition
    !! in seeder
    real(kind=rk) :: qVal

    !> Non-reflective boundary type
    type(bc_nrbc_type) :: nrbc

    !> slip wall factor
    real(kind=rk) :: slip_fac

    !> black box membrane boundary
    type(bc_blackbox_membrane_type) :: blackbox_mem

    !> whether this boundary needs neighbors on compute stencil
    logical :: useComputeNeigh = .false.

    !> Is true if this boundary requires boundary variables
    !! on boundary surface linkwise.
    !! Required to compute real coordinates points on boundary surface
    logical :: evalBcVar_link = .false.

    !> Is true only if boundary requires neighbuf Pre-collision values from
    !! next time step
    logical :: requireNeighBufPre_nNext = .false.

    !> Is true only if boundary requires neighbuf Pre-collision values from
    !! current time step
    logical :: requireNeighBufPre = .false.

    !> Is true only if boundary requires neighbuf Post-collision values
    logical :: requireNeighBufPost = .false.

    !> Collect corner BC i.e elements which are interesected by multiple
    !! boundaries only for moments based BC
    logical :: treat_corner = .false.

    !> Level-wise links that are to be updated
    !! size: minLevel:maxLevel
    !! allocated in mus_alloc_fieldBC
    type(array_type), allocatable :: links(:)

    !> Level wise wall bouzidi data
    !! size: minLevel:maxLevel
    type(bc_wall_bouzidi_type), allocatable :: bouzidi(:)

    !> Level wise inletUbbQVal data
    !! size: minLevel:maxLevel
    type(bc_inlet_bouzidi_type), allocatable :: inletUbbQVal(:)

    !> Level wise inletBfl data
    !! size: minLevel:maxLevel
    type(bc_inlet_bouzidi_type), allocatable :: inletBFL(:)

    !> Level wise nonEquilibrium extrapolation data for link-wise implementation
    !! size: minLevel:maxLevel
    type(bc_nonEqExpol_type), allocatable :: nonEqExpol(:)

    !> Level wise outletExpol data
    !! size: minLevel:maxLevel
    type(bc_outlet_type), allocatable :: outletExpol(:)

    !> Turbulent wall model type contains function pointers to
    !! compute friction and stream-wise velocity. Also stores friction
    !! velocity and turbulent viscosity on boundary elements
    type(mus_turb_wallFunc_type) :: turbwallFunc
  end type boundary_type


  !> Interface definition for the boundary condition subroutines
  abstract interface
    !>  This abstract interface defines the interface for boundary routines. All
    !! boundary routines that need to be called via [[boundary_type:fnct]]
    !! function pointer need to implement this interface.
    !! In case of new boundary routines, mark them with a comment reffering to
    !! [[boundary_type:fnct]] in order to find the routine in case the interface
    !! needs to be changed.
    subroutine boundaryRoutine( me, state, bcBuffer, globBC, levelDesc, tree, &
      &                         nSize, iLevel, sim_time, neigh, layout,       &
      &                         fieldProp, varPos, nScalars, varSys,          &
      &                         derVarPos, physics, iField, mixture           )
      import :: boundary_type, rk, mus_field_prop_type,                    &
        &       mus_scheme_layout_type, tem_time_type, tem_leveldesc_type, &
        &       treelmesh_type, mus_derVarPos_type, tem_varSys_type,       &
        &       glob_boundary_type, mus_physics_type, mus_mixture_type
      !> field boundary type
      class( boundary_type ) :: me
      !> State array
      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(:)
      !> global treelm mesh
      type( treelmesh_type ), intent(in) ::tree
      !> iLevel descriptor
      type(tem_levelDesc_type), intent(in) :: levelDesc
      !> level which invokes boundary
      integer,intent(in) :: iLevel
      !> global time information
      type( tem_time_type ), intent(in)  :: sim_time
      !> global parameters
      ! type(mus_param_type),intent(in) :: params
      integer,intent(in)  :: neigh(:)  !< connectivity array
      !> scheme layout
      type( mus_scheme_layout_type ),intent(in) :: layout
      !> Fluid property
      type( mus_field_prop_type ), intent(in)  :: fieldProp
      !> 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
      !> scheme global boundary type
      type( mus_physics_type ), intent(in) :: physics
      !> current field
      integer, intent(in) :: iField
      !> mixture info
      type(mus_mixture_type), intent(in) :: mixture
    end subroutine boundaryRoutine

    !>  This abstract interface defines the interface for bndForce calculation.
    !! [[boundary_type:calcBndForce]] in order to find the routine in case the
    !! interface needs to be changed.
    subroutine mus_proc_calcBndForce( me, bndForce, posInBndID, globBC,    &
      &                               currState, levelDesc, nSize, iLevel, &
      &                               neigh, layout, nScalars )
      import :: rk, mus_scheme_layout_type, tem_leveldesc_type, &
        &       glob_boundary_type, boundary_type
      !> field boundary type
      class( boundary_type ), intent(in) :: me
      !> bndForce to fill
      real(kind=rk), intent(inout) :: bndForce(:,:)
      ! position of boundary element in boundary%bcID
      integer, intent(in) :: posInBndID(:)
      !> scheme global boundary type
      type( glob_boundary_type ), intent(in) :: globBC
      !> current state array to access post-collision values
      real(kind=rk), intent(in) :: currState(:)
      !> size of state array ( in terms of elements )
      integer, intent(in) :: nSize
      !> iLevel descriptor
      type(tem_levelDesc_type), intent(in) :: levelDesc
      !> level which invokes boundary
      integer,intent(in) :: iLevel
      !> global parameters
      ! type(mus_param_type),intent(in) :: params
      integer,intent(in)  :: neigh(:)  !< connectivity array
      !> scheme layout
      type( mus_scheme_layout_type ),intent(in) :: layout
      !> number of Scalars in the scheme var system
      integer, intent(in) :: nScalars
    end subroutine mus_proc_calcBndForce
  end interface


contains


  ! ------------------------------------------------------------------------ !
  !> Read in the boundary conditions from the LUA parameter file.\n
  !!
  subroutine mus_load_bc( me, bc_prop, conf, parent, varSys, stencil )
    ! -------------------------------------------------------------------- !
    !> field boundary type
    type( boundary_type ), allocatable   :: me(:)
    !> boundary data from mesh
    type( tem_bc_prop_type ), intent(in) :: bc_prop
    !> lua flu state
    type( flu_state )               :: conf
    !> bc parent handle
    integer, intent(in), optional   :: parent
    !> Global variable system required to append annoymous boundary variables
    type(tem_varSys_type), intent(inout) :: varSys
    !> stencil information
    type(tem_stencilHeader_type) :: stencil
    ! -------------------------------------------------------------------- !
    integer :: bc_handle
    integer :: sub_handle
    integer :: iError
    integer :: errFatal(3)
    ! loop variable
    integer :: iBC, myBCID
    ! boundary header type
    type( tem_bc_header_type ) :: bc_header
    ! -------------------------------------------------------------------- !

    errFatal = aoterr_Fatal

    ! if fluid informations in scheme table parentHandle /= 0
    ! load bounary labels  in lua config, and match them with those of
    ! loaded from mesh (information in tem_bc_prop_type)
    ! If a BC in bc_prop can not be match by lua, then code abort
    ! If a BC in lua can not be match by bc_prop, it gives a warning.
    call tem_load_bc_header( me           = bc_header, &
      &                      conf         = conf,      &
      &                      parentHandle = parent,    &
      &                      BC_prop      = bc_prop    )
    call aot_table_open( L       = conf,                &
      &                  parent  = parent,              &
      &                  thandle = bc_handle,           &
      &                  key     = 'boundary_condition' )

    call tem_horizontalSpacer(fUnit = logUnit(1))
    write(logUnit(1),'(A,I0)') ' Loading boundary definition for nBCs: ', &
      &                        bc_header%nBCs

    ! load tables inside the boundary table for bc kind and their state
    ! variables boundaries defined in lua are match with bc label read from
    ! mesh using bc_header
    allocate( me( bc_header%nBCs ))

    do iBC = 1, bc_header%nBCs
      myBCID = bc_header%BC_ID(iBC)
      call aot_table_open( L       = conf,       &
        &                  parent  = bc_handle,  &
        &                  thandle = sub_handle, &
        &                  pos     = iBC         )
      ! Skip BC definitions that are not present in the mesh.
      if ( (sub_handle /= 0) .and. (myBCID > 0) ) then
        ! matching the boundary label from mesh to boundary label read for lua
        ! file bc_header%BC_ID(iBC) returns the position of boundary label iBC
        ! in the mesh.
        me( myBCID )%bc_id = myBCID
        me( myBCID )%label   = bc_header%label(iBC)
        me( myBCID )%BC_kind = bc_header%BC_kind(iBC)

        ! Initialize varDict for current boundary
        call init( me = me(myBCID)%varDict )

        ! Load qVal for the boundary, if certain boundary kind requires
        ! default qVal then they are set in the select case
        call aot_get_val( L       = conf,              &
          &               thandle = sub_handle,        &
          &               val     = me( myBCID )%qVal, &
          &               ErrCode = iError,            &
          &               key     = 'qVal',            &
          &               default = 0.5_rk             )

        ! Flag to treat curved boundaries. Especially nonEq expolation
        ! boundary kind. For curved boundaries, neighbors are taken along
        ! the link direction
        call aot_get_val( L       = conf,                &
          &               thandle = sub_handle,          &
          &               val     = me( myBCID )%curved, &
          &               ErrCode = iError,              &
          &               key     = 'curved',            &
          &               default = .false.              )
        write(logUnit(1),*)'   Curved: ', me( myBCID )%curved


        write(logUnit(1),"(A,I0)") '   Boundary: ',iBC
        select case( trim( bc_header%BC_kind(iBC) ))
        case( 'wall' )
          write(logUnit(1),*) '    Simple bounce back treatment for '// &
            &  'no-slip wall '// trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 0

        case( 'symmetry' )
          write(logUnit(1),*) '    Symmetry treatment for slip wall ' &
            & // trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 0

        case( 'spc_bb_wall' )
          write(logUnit(1),*) '    Species bounce back treatment for '// &
            &  'no-slip wall '// trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 1

        case( 'moments_wall', 'spc_moments_wall' )
          write(logUnit(1),*) '    Moments wall treatment for '// &
            &  'no-slip wall '// trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 0

        case( 'spc_outflow', 'spc_solvent_outflow' )
          write(logUnit(1),*) '    '//trim(bc_header%BC_kind(iBC))        &
            &                 //' treatment for '//trim(me( myBCID )%label)
          if (.not. me( myBCID )%curved) me( myBCID )%qVal = 0.0_rk
          me( myBCID )%nNeighs = 2

        case( 'spc_moments_outflow' )
          write(logUnit(1),*) '    Spc moments outflow treatment for '// &
            &  trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 1

        case( 'wall_multiReflection','wall_mr')
          write(logUnit(1),*) '    Wall multi reflection BC does NOT work!'
          call tem_abort()

          bc_header%BC_kind(iBC) = 'wall_multiReflection'
          me( myBCID )%BC_kind = bc_header%BC_kind(iBC)
          write(logUnit(1),*) '    Wall multi reflection for no-slip wall ' &
            &             //trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 2

        case( 'wall_linearInterpolation',    &
          &   'wall_libb', 'wall_libb_link', &
          &   'wall_bouzidi'                 )
          bc_header%BC_kind(iBC) = 'wall_libb'
          me( myBCID )%BC_kind = bc_header%BC_kind(iBC)
          write(logUnit(1),*) '    Wall LIBB treatment: '// &
            &                 trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 0
          me( myBCID )%useComputeNeigh = .true.

        case( 'slip_wall', 'spc_slip_wall' )
          write(logUnit(1),*) '   '//trim( bc_header%BC_kind(iBC) ) // &
            &    ' treatment for '//trim(me( myBCID )%label)
          call aot_get_val( L       = conf,                  &
            &               thandle = sub_handle,            &
            &               val     = me( myBCID )%slip_fac, &
            &               ErrCode = iError,                &
            &               key     = 'fac',                 &
            &               default = 1.0_rk                 )
          me( myBCID )%nNeighs = 0

        case( 'turbulent_wall', 'turbulent_wall_noneq_expol', &
          & 'turbulent_wall_eq')
          write(logUnit(1),*) '    Wall function ' //                   &
            &                 trim( bc_header%BC_kind(iBC) ) //         &
            &                 ' treatment for '//trim( me(myBCID)%label )
          me( myBCID )%qVal = 0.5_rk
          ! Use upto 2 neighs because turbulent wall requires precollision
          ! from 1st neigh which uses fetch.
          ! Also, this BC works only if 1st neigh is local fluid not halo.
          me( myBCID )%nNeighs = 2
          me( myBCID )%requireNeighBufPre_nNext = .true.
          select case( trim( bc_header%BC_kind(iBC) ))
          case ('turbulent_wall')
            me( myBCID )%useComputeNeigh = .true.
          case ('turbulent_wall_noneq_expol', 'turbulent_wall_eq')
            ! Use compute neigh buffer to compute density locally for
            ! straight boundary
            if(.not. me( myBCID )%curved) me( myBCID )%useComputeNeigh = .true.
          end select
          ! load wall model type
          call mus_load_turb_wallFunc(me     = me(myBCID)%turbwallFunc, &
            &                         conf   = conf,                    &
            &                         parent = sub_handle               )

        case( 'bc_pdf' )
          write(logUnit(1),*) '   '//trim( bc_header%BC_kind(iBC) ) //     &
            &    ' treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states%pdf, &
            &                     state_name = 'pdf',                    &
            &                     nComp      = stencil%QQ,               &
            &                     conf       = conf,                     &
            &                     bc_handle  = sub_handle,               &
            &                     varDict    = me(myBCID)%varDict,       &
            &                     varSys     = varSys                    )

        case( 'velocity_bounceback', 'velocity_eq', 'spc_inlet_eq',     &
          &   'spc_outlet_vel', 'spc_vel_bb', 'velocity_momentsbased',  &
          &   'spc_inlet', 'spc_moments_vel', 'spc_bb_vel_test',        &
          &   'velocity_bfl', 'vel_neq', 'velocity_noneq_expol',        &
          &   'spc_inflow', 'spc_velocity_noneq_expol',                 &
          &   'spc_solvent_inflow', 'inlet_nrbc'                        )
          write(logUnit(1),*) '   '//trim( bc_header%BC_kind(iBC) ) &
            &                 //' treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %velocity, &
            &                     state_name = 'velocity',          &
            &                     nComp      = 3,                   &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )

          call aot_get_val( L       = conf,               &
            &               thandle = sub_handle,         &
            &               val     = me( myBCID )%order, &
            &               ErrCode = iError,             &
            &               key     = 'order',            &
            &               default = 1                   )

          me(myBCID)%nNeighs = me(myBCID)%order

          ! load molefraction in addition for spc_inlet_eq
          select case( trim( bc_header%BC_kind(iBC) ))
          case( 'spc_velocity_noneq_expol', 'spc_solvent_inflow' )
            if (.not. me( myBCID )%curved) me( myBCID )%qVal = 1.0_rk
            me( myBCID )%nNeighs = 1
          case( 'spc_inlet_eq', 'spc_inlet', 'spc_inflow' )
            call tem_load_bc_state( bc         = me(myBCID)%BC_states &
              &                                            %moleFrac, &
              &                     state_name = 'mole_fraction',     &
              &                     nComp      = 1,                   &
              &                     conf       = conf,                &
              &                     bc_handle  = sub_handle,          &
              &                     varDict    = me(myBCID)%varDict,  &
              &                     varSys     = varSys               )

            if ( trim( bc_header%BC_kind(iBC) ) == 'spc_inflow') then
              if (.not. me( myBCID )%curved) me( myBCID )%qVal = 1.0_rk
              me( myBCID )%requireNeighBufPre_nNext = .true.
              me( myBCID )%nNeighs = 1
            else
              if (.not. me( myBCID )%curved) me( myBCID )%qVal = 0.5_rk
              me( myBCID )%requireNeighBufPost = .true.
              me( myBCID )%nNeighs = 2
            end if

          case( 'moments_inflow' )
            call tem_load_bc_state( bc         = me(myBCID)%BC_states &
              &                                            %pressure, &
              &                     state_name = 'pressure',          &
              &                     nComp      = 1,                   &
              &                     conf       = conf,                &
              &                     bc_handle  = sub_handle,          &
              &                     varDict    = me(myBCID)%varDict,  &
              &                     varSys     = varSys               )

          !case ('inlet_bouzidi', 'velocity_bounceback_qval', 'velocity_bfl')
          case ('velocity_bounceback', 'velocity_bfl')
            write(logUnit(1),*) '    Inlet higher order with linear'   //  &
              &             ' interpolation treatment for velocity bc '//  &
              &            trim(me( myBCID )%label)
            if (.not. me( myBCID )%curved) me( myBCID )%qVal = 0.5_rk
            me( myBCID )%nNeighs = 0
            me( myBCID )%useComputeNeigh = .true.

          ! Load additionals for velocity_noneq_expol
          case( 'velocity_noneq_expol' )
            write(logUnit(1),*) '    Non-equilibrium extrapolation ' &
              &               //' treatment for '// trim(me( myBCID )%label)

            if (.not. me( myBCID )%curved) then
               me( myBCID )%qVal = 0.0_rk
               me( myBCID )%requireNeighBufPre_nNext = .true.
            else
               me( myBCID )%requireNeighBufPost = .true.
            end if
            me( myBCID )%nNeighs = 1
          case( 'inlet_nrbc' )
            write(logUnit(1),*) '    Characteristic based non-reflective ' &
              &                 //'inflow velocity boundary condition '// &
              &                 trim(me( myBCID )%label)
            me( myBCID )%requireNeighBufPre_nNext = .true.
            me( myBCID )%nNeighs = 2
          end select

        ! inlet mass flow rate boundary condition
        case( 'mfr_bounceback', 'mfr_eq' )
          write(logUnit(1),*) '   '//trim( bc_header%BC_kind(iBC) ) //  &
            &                 ' treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states     &
            &                                            %massFlowRate, &
            &                     state_name = 'mass_flowrate',         &
            &                     nComp      = 1,                       &
            &                     conf       = conf,                    &
            &                     bc_handle  = sub_handle,              &
            &                     varDict    = me(myBCID)%varDict,      &
            &                     varSys     = varSys                   )

          call aot_get_val( L       = conf,               &
            &               thandle = sub_handle,         &
            &               val     = me( myBCID )%order, &
            &               ErrCode = iError,             &
            &               key     = 'order',            &
            &               default = 1                   )

          me(myBCID)%nNeighs = me(myBCID)%order

        case( 'pressure_eq', 'pressure_antibounceback', 'outlet_bouzidi',      &
          &   'pressure_expol', 'pressure_expol_slow', 'pressure_noneq_expol', &
          &   'pressure_momentsbased', 'press_neq'                             )
          select case( trim( bc_header%BC_kind(iBC) ))
          case( 'pressure_eq' )
            write(logUnit(1),*) '    Outlet pressure equilibrium type for ' &
              &                 // trim(me( myBCID )%label)
            me( myBCID )%requireNeighBufPre_nNext = .true.
            me( myBCID )%nNeighs = 2

          case( 'pressure_antibounceback' )
            write(logUnit(1),*) '    Outlet pressure anti bounce back for ' &
              &                 // trim(me( myBCID )%label)
            me( myBCID )%requireNeighBufPost = .true.
            me( myBCID )%nNeighs = 1

          case( 'outlet_bouzidi' )
            write(logUnit(1),*) '    Outlet bouzidi boundary condition for' &
              &                 // ' ' // trim(me( myBCID )%label)
            me( myBCID )%nNeighs = 1

          case( 'press_neq' )
            write(logUnit(1),*) '    Pressure non-equilibrium boundary ' &
              &            //'condition for '//trim(me( myBCID )%label)
            me( myBCID )%nNeighs = 1

          case( 'pressure_noneq_expol' )
            write(logUnit(1),*) '    Pressure Non-equilibrium extrapolation' &
              &               //' treatment for '//trim(me( myBCID )%label)

            me( myBCID )%qVal = 0.0_rk
            me( myBCID )%requireNeighBufPre_nNext = .true.
            me( myBCID )%nNeighs = 1

          case( 'pressure_expol', 'pressure_expol_slow' )
            write(logUnit(1),*) '  Outlet extrapolation type for '//  &
              &                      trim(me( myBCID )%label)
            me( myBCID )%qVal = 0.0_rk
            me( myBCID )%requireNeighBufPre_nNext = .true.
            me( myBCID )%nNeighs = 2

          case( 'pressure_momentsbased' )
            write(logUnit(1),*) '    Moments based pressure boundary '// &
              &                 'type for '//                            &
              &                 trim(me( myBCID )%label)
            me( myBCID )%nNeighs = 0
          end select

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %pressure, &
            &                     state_name = 'pressure',          &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )
        case( 'spc_outlet_eq', 'spc_outlet_expol' )
          select case( trim( bc_header%BC_kind(iBC) ))
          case( 'spc_outlet_eq' )
            write(logUnit(1),*) '    Species outlet pressure '// &
              &                 'equilibrium type for '//        &
              &                  trim(me( myBCID )%label)
            me( myBCID )%qVal = 0.5_rk
            me( myBCID )%nNeighs = 1

          case( 'spc_outlet_expol' )
            write(logUnit(1),*) '    Species outlet extrapolation type for ' &
              &                 // trim(me( myBCID )%label)
            me( myBCID )%qVal = 0.5_rk
            me( myBCID )%requireNeighBufPre_nNext = .true.
            me( myBCID )%nNeighs = 2
          end select

        case( 'outlet_nrbc', 'outlet_nrbc_eq' )
          me( myBCID )%requireNeighBufPre_nNext = .true.
          me( myBCID )%nNeighs = 2
          select case( trim( bc_header%BC_kind(iBC) ))
          case( 'outlet_nrbc' )
            write(logUnit(1),*) '    Characteristic based non-reflective ' &
              &                 //'outflow pressure boundary condition '// &
              &                 trim(me( myBCID )%label)
            call tem_load_bc_state( bc         = me(myBCID)%BC_states &
              &                                            %pressure, &
              &                     state_name = 'pressure',          &
              &                     conf       = conf,                &
              &                     bc_handle  = sub_handle,          &
              &                     varDict    = me(myBCID)%varDict,  &
              &                     varSys     = varSys               )

          case( 'outlet_nrbc_eq' )
            write(logUnit(1),*) '    Eq-type Characteristic based ' // &
              &             'non-reflective boundary condition '    // &
              &             trim(me( myBCID )%label)
            call tem_load_bc_state( bc         = me(myBCID)%BC_states &
              &                                            %pressure, &
              &                     state_name = 'pressure',          &
              &                     conf       = conf,                &
              &                     bc_handle  = sub_handle,          &
              &                     varDict    = me(myBCID)%varDict,  &
              &                     varSys     = varSys               )
          end select

          call aot_get_val( L       = conf,                    &
            &               thandle = sub_handle,              &
            &               val     = me( myBCID )%nrbc%kappa, &
            &               ErrCode = iError,                  &
            &               key     = 'kappa',                 &
            &               default = 1.0_rk                   )
          call aot_get_val( L       = conf,                    &
            &               thandle = sub_handle,              &
            &               val     = me( myBCID )%nrbc%sigma, &
            &               ErrCode = iError,                  &
            &               key     = 'sigma',                 &
            &               default = 0.58_rk                  )
          call aot_get_val(                                    &
            &         L       = conf,                          &
            &         thandle = sub_handle,                    &
            &         val     = me( myBCID )%nrbc%lodi_length, &
            &         ErrCode = iError,                        &
            &         key     = 'length'                       )

          if ( btest(iError, aoterr_Fatal) ) then
            write(logUnit(1),*)'FATAL Error occured in loading LODI length.'
            write(logUnit(1),*)'HINT:It is nElements between inlet and outlet'
            write(logUnit(1),*)'STOPPING'
            call tem_abort()
          end if

          if ( me(myBCID)%nrbc%lodi_length .feq. 0._rk ) then
            write(logUnit(1),*)'LODI length is zero.'
            write(logUnit(1),*)'HINT:It is nElements between inlet and outlet'
            write(logUnit(1),*)'STOPPING'
            call tem_abort()
          end if

          call aot_get_val(                             &
            &         L       = conf,                   &
            &         thandle = sub_handle,             &
            &         val     = me( myBCID )%nrbc%Ma_L, &
            &         ErrCode = iError,                 &
            &         key     = 'mach_lat'              )
          if ( btest(iError, aoterr_Fatal) ) then
            write(logUnit(1),*) 'FATAL Error occured in loading mach_lat.'
            write(logUnit(1),*) 'HINT: Ma_L = u_L/cs_L'
            write(logUnit(1),*) 'STOPPING'
            call tem_abort()
          end if

          write(logUnit(1),"(A,F9.2)") ' kappa: ', me( myBCID )%nrbc%kappa
          write(logUnit(1),"(A,F9.5)") ' sigma: ', me( myBCID )%nrbc%sigma
          write(logUnit(1),"(A,F9.2)") ' lodi L:', me( myBCID )%nrbc%lodi_length
          write(logUnit(1),"(A,F9.2)") ' mach_lat:', me( myBCID )%nrbc%Ma_L

        case( 'outlet_zero_prsgrd', 'spc_outlet_zero_prsgrd' )
          write(logUnit(1),*) '    Outlet zero pressure gradient for '  // &
            &                      trim(me( myBCID )%label)
          me( myBCID )%nNeighs = 1

        case( 'outlet_dnt' )
          write(logUnit(1),*) '    Outlet do-nothing type for '// &
            &                      trim(me( myBCID )%label)
          me( myBCID )%requireNeighBufPost = .true.
          me( myBCID )%nNeighs = 1

        case( 'moments_outflow' )
          write(logUnit(1),*) '    Outlet moments type for '// &
            &                      trim(me( myBCID )%label)
          me( myBCID )%requireNeighBufPre_nNext = .true.
          me( myBCID )%nNeighs = 2

        case( 'spc_molefrac', 'spc_molefrac_wtdf', 'spc_moments_molefrac', &
          &   'spc_molefrac_eq' )
          write(logUnit(1),*) '    Species '//trim(bc_header%BC_kind(iBC)) &
            &                 //'  treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %moleFrac, &
            &                     state_name = 'mole_fraction',     &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )

          me( myBCID )%nNeighs = 0

        case( 'spc_mole_fraction_noneq_expol' )
          write(logUnit(1),*) '    Species '//trim(bc_header%BC_kind(iBC)) &
            &                 //'  treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %moleFrac, &
            &                     state_name = 'mole_fraction',     &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )
          if (.not. me( myBCID )%curved) me( myBCID )%qVal = 0.0_rk
          me( myBCID )%nNeighs = 1

        case( 'spc_moledens_eq' )
          write(logUnit(1),*) '    Species '//trim(bc_header%BC_kind(iBC)) &
            &                 //'  treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %moleDens, &
            &                     state_name = 'mole_density',      &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )

          me( myBCID )%nNeighs = 0


        case( 'spc_moleflux', 'spc_moleflux_eq', 'spc_moments_moleflux' )
          write(logUnit(1),*) '    Species molar flux treatment for ' // &
            &                 trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %moleFlux, &
            &                     state_name = 'mole_flux',         &
            &                     nComp      = 3,                   &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )

          me( myBCID )%nNeighs = 0

        case( 'spc_blackbox_mem_ion', 'spc_blackbox_mem_solvent' )
          write(logUnit(1),*) '    Species black box membrance ' //    &
            &                 'treatment for '//trim(me( myBCID )%label)

          call aot_get_val( L       = conf,                                &
            &               thandle = sub_handle,                          &
            &               val     = me( myBCID )%blackbox_mem%transNr,   &
            &               ErrCode = iError,                              &
            &               key     = 'transference_number'                )
          if( btest( iError, aoterr_Fatal ))then
            write(logUnit(1),*)'FATAL Error occured in definition of '// &
              &            'black box membrance boundary condition'
            write(logUnit(1),*)'while retrieving transference number'
            write(logUnit(1),*)'STOPPING'
            call tem_abort()
          end if

          select case( trim( bc_header%BC_kind(iBC) ))
            case( 'spc_blackbox_mem_solvent' )
              call aot_get_val( L       = conf,                            &
                &               thandle = sub_handle,                      &
                &               val     = me( myBCID )                     &
                &                         %blackbox_mem%solvTrans_coeff,   &
                &               ErrCode = iError,                          &
                &               key     = 'transference_number'            )
              if( btest( iError, aoterr_Fatal ))then
                write(logUnit(1),*)'FATAL Error occured in definition of ' &
                  &            //'black box '//                            &
                  &            'membrance solvent boundary condition'
                write(logUnit(1),*)'while retrieving solv_trans_coeff'
                write(logUnit(1),*)'STOPPING'
                call tem_abort()
              end if
          end select
        case( 'spc_molediff_flux' )
          write(logUnit(1),*) '    Species molar diffusion flux '//    &
            &                 'treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %moleFrac, &
            &                     state_name = 'mole_fraction',     &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )

          call tem_load_bc_state( bc         = me(myBCID)%BC_states      &
            &                                            %moleDiff_flux, &
            &                     state_name = 'mole_diff_flux',         &
            &                     nComp      = 3,                        &
            &                     conf       = conf,                     &
            &                     bc_handle  = sub_handle,               &
            &                     varDict    = me(myBCID)%varDict,       &
            &                     varSys     = varSys                    )

          me( myBCID )%nNeighs = 0

        case( 'potential_noneq_expol' )
          write(logUnit(1),*) '    Non-equilibrium extrapolation '//   &
            &                 'treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states  &
            &                                            %potential, &
            &                     state_name = 'potential',          &
            &                     conf       = conf,                 &
            &                     bc_handle  = sub_handle,           &
            &                     varDict    = me(myBCID)%varDict,   &
            &                     varSys     = varSys                )

          if (.not. me( myBCID )%curved) me( myBCID )%qVal = 0.0_rk
          me( myBCID )%requireNeighBufPost = .true.
          me( myBCID )%nNeighs = 1

        case( 'potential_neumann' )
          write(logUnit(1),*) '    Non-equilibrium extrapolation  '//  &
            &                 'treatment for '//trim(me( myBCID )%label)
          call tem_load_bc_state( bc         = me(myBCID)%BC_states      &
            &                                            %surChargeDens, &
            &                     state_name = 'surface_charge_density', &
            &                     conf       = conf,                     &
            &                     bc_handle  = sub_handle,               &
            &                     varDict    = me(myBCID)%varDict,       &
            &                     varSys     = varSys                    )

          !KM: For Neumann BC, qVal should be 0.5 to be consistent with second
          ! order extrapolation of potential field
          if (.not. me( myBCID )%curved) me( myBCID )%qVal = 0.5_rk
          me( myBCID )%requireNeighBufPost = .true.
          me( myBCID )%nNeighs = 1

        case( 'moledens_noneq_expol' )
          write(logUnit(1),*) '    Non-equilibrium extrapolation '//  &
            &                 'treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %velocity, &
            &                     state_name = 'velocity',          &
            &                     nComp      = 3,                   &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )

          call tem_load_bc_state( bc         = me(myBCID)%BC_states  &
            &                                            %moleDens,  &
            &                     state_name = 'mole_density',       &
            &                     conf       = conf,                 &
            &                     bc_handle  = sub_handle,           &
            &                     varDict    = me(myBCID)%varDict,   &
            &                     varSys     = varSys                )

          if (.not. me( myBCID )%curved) me( myBCID )%qVal = 1.0_rk
          me( myBCID )%nNeighs = 1

        case( 'moledens_neumann' )
          write(logUnit(1),*) '    Non-equilibrium extrapolation' &
            &               //' treatment for '//trim(me( myBCID )%label)

          call tem_load_bc_state( bc         = me(myBCID)%BC_states &
            &                                            %velocity, &
            &                     state_name = 'velocity',          &
            &                     nComp      = 3,                   &
            &                     conf       = conf,                &
            &                     bc_handle  = sub_handle,          &
            &                     varDict    = me(myBCID)%varDict,  &
            &                     varSys     = varSys               )

          call tem_load_bc_state( bc         = me(myBCID)%BC_states  &
            &                                            %moleFlux,  &
            &                     state_name = 'mole_flux',          &
            &                     conf       = conf,                 &
            &                     bc_handle  = sub_handle,           &
            &                     varDict    = me(myBCID)%varDict,   &
            &                     varSys     = varSys                )

          if (.not. me( myBCID )%curved) me( myBCID )%qVal = 0.5_rk
          me( myBCID )%requireNeighBufPost = .true.
          me( myBCID )%nNeighs = 1

        case( 'flekkoy_outlet' )
          write(logUnit(1),*) '   Outlet BCs for the flekkoy ' // &
            &                 'diffusion model'
          me( myBCID )%nNeighs = 2

        case( 'flekkoy_inlet' )
          write(logUnit(1),*) '   Inlet BCs for the flekkoy ' // &
            &                 'diffusion model'
          me( myBCID )%nNeighs = 0

        case( 'test' )
          write(logUnit(1),*) '   WARNING!! '
          write(logUnit(1),*) '   Test boundary conditions for testing ' // &
            &                 'neighbor info exchange'
          me( myBCID )%nNeighs = 20

        case default
          write(logUnit(1),*) 'Unknown Boundary condition kind!'
          write(logUnit(1),*) 'Stopping ...'
          call tem_abort()
        end select

        write(logUnit(3),*)'   Default qVal: ', me( myBCID )%qVal

        if ( me(myBCID)%curved ) then
           me( myBCID )%requireNeighBufPost = .true.
           !KM! \todo activate this line after implementing nonEq expol boundary
           ! condition for curved and straight boundary seperately
           ! me( myBCID )%requireNeighBufPre = .false.

           ! Set nNeighs only if its set to zero before else use already set
           ! nNeighs
           if ( me( myBCID )%nNeighs == 0 ) then
             me( myBCID )%nNeighs = 1
           end if
        end if

        write(logUnit(1),"(A,I0)") '   required neighbors: ', &
          &                        me( myBCID )%nNeighs
        ! Do special treatment for corner elements which are intersected
        ! by multiple boundaries only for moments based BC
        select case( trim( bc_header%BC_kind(iBC) ))
        case( 'pressure_momentsbased', 'moments_wall', 'velocity_momentsbased',&
          &   'moments_inflow', 'moments_outflow',                             &
          &   'spc_moments_molefrac', 'spc_moments_moleflux',                  &
          &   'spc_moments_wall', 'spc_moments_vel', 'spc_moments_outflow'     )
          me( myBCID )%treat_corner = .true.
        case default
          me( myBCID )%treat_corner = .false.
        end select

        call truncate( me = me(myBCID)%varDict )
      end if  ! sub_handle/=0
      call aot_table_close( L=conf, thandle=sub_handle )
    end do ! iBC
    call tem_horizontalSpacer(fUnit = logUnit(1))

  end subroutine mus_load_bc
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  subroutine mus_init_bc_elems( me, nElems, QQN, hasQVal )
    ! -------------------------------------------------------------------- !
    type( bc_elems_type ), intent(inout) :: me
    integer, intent(in) :: nElems
    !> Number of direction excluding center since center is never treated for
    !! bc elems
    integer, intent(in) :: QQN
    logical, intent(in) :: hasQval
    ! -------------------------------------------------------------------- !

    if ( allocated( me%elem%val ) ) then
      call destroy( me%normalInd  )
      call destroy( me%posInBcElemBuf )
      call destroy(me%elem)
      call destroy(me%normal)
      call destroy(me%bitMask)
    end if

    call init( me = me%normalInd, length=nElems )
    call init( me = me%posInBcElemBuf, length=nElems)
    call init( me = me%elem, length=nElems )
    call init( me = me%bitMask, width = QQN, length=nElems )
    call init( me = me%normal, width = 3, length=nElems )

    if ( hasQVal ) then
      if ( allocated( me%qVal%val ) ) then
        call destroy( me%qVal )
      end if

      call init( me = me%qVal, width=QQN )
    end if

  end subroutine mus_init_bc_elems
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  !> It removes non-valid elements while still maintaining the origianl order.
  !! The given bc elements (elems) contains both valid and non-valid elements.
  !! Position of valid elements are given by posInBCElem.
  !! Valid elements are moved towards the start of the elems so that they become
  !! continuous in the elems.
  !!
  subroutine rearrange_bc_elems( minLevel, maxLevel, nValid, posInBCElem, &
    &                            nElems, elemLvl )
    ! -------------------------------------------------------------------- !
    integer, intent(in) :: minLevel, maxLevel
    !> number of valid BC elements
    integer, intent(in) :: nValid(minLevel:maxLevel)
    !> BC elements information
    integer, intent(in) :: nElems(minLevel:maxLevel)
    !> their position in bc_elems_type
    integer, intent(in) :: posInBCElem(maxval(nElems),minLevel:maxLevel)
    !> BC elements information
    type( bc_elems_type ), intent(inout) :: elemLvl(minLevel:maxLevel)
    ! -------------------------------------------------------------------- !
    integer :: iElem, iLevel, oldPos
    ! -------------------------------------------------------------------- !

    write(dbgUnit(1), *) 'Inside remove solid in bcElems routine ...'

    do iLevel = minLevel, maxLevel

      ! Re arrange bc elements only when any solid are found
      if ( nValid(iLevel) < nElems(iLevel) ) then
        do iElem = 1, nValid( iLevel )
          oldPos = posInBCElem(iElem,iLevel)
          elemLvl(iLevel)%elem%val( iElem ) = elemLvl(iLevel)%elem%val( oldPos )
          elemLvl(iLevel)%bitMask%val( :,iElem ) =                             &
            &                            elemLvl(iLevel)%bitmask%val( :,oldPos )
          elemLvl(iLevel)%normal%val( :,iElem )  =                             &
            &                            elemLvl(iLevel)%normal%val( :,oldPos )
          elemLvl(iLevel)%normalInd%val( iElem ) =                             &
            &                            elemLvl(iLevel)%normalInd%val( oldPos )
          write(dbgUnit(6), *) 'elem: ', elemLvl( iLevel )%elem%val( iElem )
        end do

        ! update growing array nVals
        elemLvl(iLevel)%elem%nVals = nValid(iLevel)
        elemLvl(iLevel)%bitMask%nVals = nValid(iLevel)
        elemLvl(iLevel)%normal%nVals = nValid(iLevel)
        elemLvl(iLevel)%normalInd%nVals = nValid(iLevel)

      end if ! nValid(iLevel) == nElems(iLevel)

    end do ! iLevel

    write(dbgUnit(1), *) 'Leave  remove solid in bcElems routine ...'
    write(dbgUnit(1), *) ''

  end subroutine rearrange_bc_elems
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  !> It count valid (non-solid) elements in BC elements list.
  !! Input:
  !!   minLevel, maxLevel
  !!   LevelPointer
  !!   LevelDesc
  !!   nElems - number of BC elements
  !!   elems - positions of BC elements in tree or levelPointer
  !! Output:
  !!   nValid - number of valid BC elements
  !!   posInBCElem - positions of valid elements in BC elements list
  !!
  subroutine check_solid_in_bc( minLevel, maxLevel, levelPointer, levelDesc, &
    &                           nElems, elemLvl, nValid, posInBCElem )
    ! -------------------------------------------------------------------- !
    integer, intent(in) :: minLevel, maxLevel
    !> Level Pointer ( position of a element in level desc )
    integer, intent(in) :: levelPointer(:)
    !> Level Descriptor
    type( tem_levelDesc_type ), intent(in) :: levelDesc(minLevel:maxLevel)
    !> number of BC elements
    integer, intent(in) :: nElems(minLevel:maxLevel)
    !> BC elements list that contains their position in levelPointer
    type( bc_elems_type ), intent(in) :: elemLvl(minLevel:maxLevel)
    !> number of valid (non-solid) elements
    integer, intent(out) :: nValid(minLevel:maxLevel)
    !> positions of valid elements in globBC elements list
    integer, intent(out) :: posInBCElem(maxval(nElems),minLevel:maxLevel)
    ! -------------------------------------------------------------------- !
    integer :: iElem, iLevel, posInTotal
    ! -------------------------------------------------------------------- !

    write(dbgUnit(1), *) 'Inside check solid in bc routine ...'
    write(dbgUnit(1), *) 'If any solid are found, they are listed as below.'

    posInBCElem(:,:) = 0
    nValid(:) = 0

    do iLevel = minLevel, maxLevel

      do iElem = 1, nElems( iLevel )
        posInTotal = levelPointer( elemLvl( iLevel )%elem%val( iElem ) )
        if ( btest( levelDesc(iLevel)%property(posInTotal), prp_solid ) ) then
          write(dbgUnit(3), "(A,I2,2(A,I0))")    &
            &  'found a Solid! level: ', iLevel, &
            &  ', posInTotal: ', posInTotal,     &
            &  ', treeID: ', levelDesc(iLevel)%total(posInTotal)
        else
          nValid(iLevel) = nValid(iLevel) + 1
          posInBCElem( nValid(iLevel), iLevel ) = iElem
        end if
        write(dbgUnit(6), *) 'elem: ', elemLvl( iLevel)%elem%val( iElem )

      end do ! iElem

      if ( nElems(iLevel) /= nValid( iLevel) ) then
        write(logUnit(5),"(2(A,I0))") &
          &  ' Removed ', nElems(iLevel) - nValid(iLevel), &
          &  ' elements on level ', iLevel
      end if
      write(dbgUnit(3), *) 'nElems: ', nElems( iLevel )
      write(dbgUnit(3), *) 'nValid: ', nValid( iLevel )

    end do ! iLevel

    write(dbgUnit(1), *) 'Leave  check solid in bc routine ...'
    write(dbgUnit(1), *) ''

  end subroutine check_solid_in_bc
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  subroutine debug_glob_boundary_type( nBCs, minLevel, maxLevel, me )
    ! -------------------------------------------------------------------- !
    integer,                    intent(in) :: nBCs, minLevel, maxLevel
    type( glob_boundary_type ), intent(in) :: me( nBCs )
    ! -------------------------------------------------------------------- !
    integer :: iBC, iLevel, iElem
    ! -------------------------------------------------------------------- !

    write(dbgUnit(1),"(A)")'-- DEBUG glob_boundary_type  ----------------------'
    do iBC = 1, nBCs
      write(dbgUnit(5), "(I0,A)") iBC, ', BC label: '//trim(me(iBC)%label)
      write(dbgUnit(5), "(A,L5)")  'hasQVal: ', me(iBC)%hasQVal
      write(dbgUnit(5), "(A,L5)")  'qValInitialized: ', me(iBC)%qValInitialized
      write(dbgUnit(5), "(A,L5)")  'isWall: ', me(iBC)%isWall
      write(dbgUnit(5), "(A,I0)") 'nElems_local: ', me(iBC)%nElems_local
      write(dbgUnit(5), "(A,I0)") 'nElems_total: ', me(iBC)%nElems_total
      if ( .not. me( iBC )%isWall ) then
        do iLevel = minLevel, maxLevel
          write(dbgUnit(5), "(A,I0)") 'iLevel: ', iLevel
          write(dbgUnit(5), "(A,I0)") 'nElems: ', me(iBC)%nElems(iLevel)
          write(dbgUnit(5), "(A,I0)") 'nElems_totalLevel: ', &
            &                         me(iBC)%nElems_totalLevel(iLevel)
          write(dbgUnit(5), "(4A10)") 'elem', 'elemBuf', 'normalInd', 'normal'
          do iElem = 1, me(iBC)%nElems(iLevel)
            write(dbgUnit(5),"(6I10)")                              &
              &  me(iBC)%elemLvl(iLevel)%elem%val(iElem),           &
              &  me(iBC)%elemLvl(iLevel)%posInBcElemBuf%val(iElem), &
              &  me(iBC)%elemLvl(iLevel)%normalInd%val(iElem),      &
              &  me(iBC)%elemLvl(iLevel)%normal%val(1:3,iElem)
          end do ! iElem
          write(dbgUnit(1), "(A)") ''
        end do ! iLevel
      else ! isWall == true
        write(dbgUnit(5),"(A)") 'This is wall. No need to check bc elems'
      end if ! isWall
      write(dbgUnit(1), "(A)") ''
    end do ! iBC

  end subroutine debug_glob_boundary_type
  ! ------------------------------------------------------------------------ !

  ! ------------------------------------------------------------------------ !
  !> Set BC elements positions in LevelDesc%neigh%nghElems
  !!
  subroutine mus_set_posInNghElems( minLevel, maxLevel, nStencils, &
      &                             globBC, BC )
    ! -------------------------------------------------------------------- !
    integer, intent(in) :: minLevel, maxLevel, nStencils
    type( glob_boundary_type ), intent(in) :: globBC
    type( boundary_type ),   intent(inout) :: BC
    ! -------------------------------------------------------------------- !
    integer :: iLevel, iElem
    integer :: counter(minLevel:maxLevel,nStencils), stencilPos
    ! -------------------------------------------------------------------- !

    if ( BC%nNeighs > 0 ) then
      counter = 0
      do iLevel = minLevel, maxLevel
        do iElem = 1, globBC%nElems(iLevel)

          stencilPos = BC%elemLvl(iLevel)%stencilPos(iElem)
          counter(iLevel, stencilPos) = counter(iLevel, stencilPos) + 1
          BC%elemLvl(iLevel)%posInNghElems(iElem) = counter(iLevel, stencilPos)

        end do ! iElem
      end do ! iLevel
    end if

  end subroutine mus_set_posInNghElems
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  subroutine mus_set_bcLinks( iField, QQ, QQN, nScalars, nElems, elemLvl, &
    &                         nSize, neigh, links )
    ! -------------------------------------------------------------------- !
    integer,                intent(in) :: iField, QQ, QQN, nScalars
    !> number of BC elements
    integer,                intent(in) :: nElems, nSize
    type( bc_elems_type ),  intent(in) :: elemLvl
    integer,                intent(in) :: neigh(:)
    type( array_type ),    intent(out) :: links
    ! -------------------------------------------------------------------- !
    integer :: iElem, iDir, iLink, posInTotal
    integer, allocatable :: dirs(:)
    ! -------------------------------------------------------------------- !

    allocate( dirs( nElems * QQ ) )
    iLink = 0

    write(dbgUnit(7), "(A,I0)") 'nSize: ', nSize

    do iElem = 1, nElems
      posInTotal = elemLvl%elem%val( iElem )
      write(dbgUnit(7), "(2(A,I0))") 'iElem: ', iElem, &
        &                            ', posInTotal: ', posInTotal
      do iDir = 1, QQN
        if( elemLvl%bitmask%val( iDir, iElem )) then
          iLink = iLink + 1
          dirs(iLink) =  neigh((idir-1)* nsize+ posintotal)+( ifield-1)* qq+ nscalars*0
        end if
      end do ! iDir
    end do ! iElem

    allocate( links%val( iLink ) )
    links%val(1:iLink) = dirs(1:iLink)
    links%nVals        = iLink

    deallocate( dirs )

  end subroutine mus_set_bcLinks
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  !> Set the coefficients of bouzidi linear interpolation boundary condition.
  !!
  subroutine set_bouzidi_coeff( qVal, cIn, cOut, cNgh, cVel )
    ! -------------------------------------------------------------------- !
    real(kind=rk),  intent(in) :: qVal
    real(kind=rk), intent(out) :: cIn, cOut, cNgh, cVel
    ! -------------------------------------------------------------------- !
    if ( qVal >= 0.5_rk ) then
      cIn  = 1._rk - 0.5_rk / qVal
      cOut =         0.5_rk / qVal
      cNgh = 0.0_rk
      cVel = 0.5_rk / qVal
    else
      cIn  = 0.0_rk
      cOut =         2.0_rk * qVal
      cNgh = 1._rk - 2.0_rk * qVal
      cVel = 1._rk
    end if

  end subroutine set_bouzidi_coeff
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  !> Set necessary data for Wall Bouzidi BC
  !! bouzidi should be allocated beforehand
  subroutine mus_set_bouzidi( iLevel, QQ, QQN, nScalars, globBC, cxDirInv, &
    &                         varPos, bouzidi )
    ! -------------------------------------------------------------------- !
    integer, intent(in) :: iLevel, QQ, QQN, nScalars
    type( glob_boundary_type ), intent(in) :: globBC
    integer, intent(in)  :: cxDirInv(QQ)
    integer, intent(in)  :: varPos(:)
    !> cIn, cOut, cNgh should have size of nLinks
    type( bc_wall_bouzidi_type ), intent(inout) :: bouzidi
    ! -------------------------------------------------------------------- !
    integer :: iLink, iDir, invDir, iElem, posInBCBuf
    real(kind=rk) :: cVelTmp ! not required for wall_libb
    ! -------------------------------------------------------------------- !

    iLink = 0

    do iElem = 1, globBC%nElems(iLevel)

      posInBCBuf = globBC%elemLvl( iLevel )%posInBcElemBuf%val( iElem )

      do iDir = 1, QQN
        if( globBC%elemLvl(iLevel)%bitmask%val( iDir, iElem )) then
          invDir = cxDirInv(iDir)
          iLink = iLink + 1
          bouzidi%qVal( iLink ) = globBC%elemLvl( iLevel )       &
            &                           %qVal%val( invDir, iElem )
          call set_bouzidi_coeff(            &
            &    qVal = bouzidi%qVal(iLink), &
            &    cIn  = bouzidi%cIn(iLink),  &
            &    cOut = bouzidi%cOut(iLink), &
            &    cNgh = bouzidi%cNgh(iLink), &
            &    cVel = cVelTmp              )

          ! inPos and outPos are used in bcBuffer
          bouzidi%inPos( iLink ) = varPos(iDir) + ( posInBCBuf-1 ) * nScalars
          bouzidi%outPos( iLink ) = varPos(invDir) + ( posInBCBuf-1 ) * nScalars
          ! nghPos is used in computeNeighBuf
          bouzidi%nghPos( iLink ) = invDir + (iElem-1)*QQ
        end if
      end do ! iDir

    end do ! iElem

  end subroutine mus_set_bouzidi
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  subroutine mus_alloc_bouzidi( me, nVals, minLevel, maxLevel )
    ! -------------------------------------------------------------------- !
    type( bc_wall_bouzidi_type ), allocatable :: me(:)
    integer, intent(in) :: minLevel, maxLevel
    integer, intent(in) :: nVals(minLevel:maxLevel)
    ! -------------------------------------------------------------------- !
    integer :: iLevel
    ! -------------------------------------------------------------------- !

    allocate( me( minLevel:maxLevel ) )
    write(logUnit(7), "(2(A,I0))") 'Allocated wall libb type level: ', &
      &                             minLevel, ' - ', maxLevel

    do iLevel = minLevel, maxLevel
      if ( nVals(iLevel) > 0 ) then
        write(logUnit(9), "(2(A,I0))") 'level: ', iLevel, ', nVal: ', &
          &                            nVals(iLevel)
      end if
      if ( nVals(iLevel) >= 0 ) then
        allocate( me(iLevel)%  qVal( nVals(iLevel) ) )
        allocate( me(iLevel)%   cIn( nVals(iLevel) ) )
        allocate( me(iLevel)%  cOut( nVals(iLevel) ) )
        allocate( me(iLevel)%  cNgh( nVals(iLevel) ) )
        allocate( me(iLevel)% inPos( nVals(iLevel) ) )
        allocate( me(iLevel)%outPos( nVals(iLevel) ) )
        allocate( me(iLevel)%nghPos( nVals(iLevel) ) )
        ! Intialize with zero
        me(iLevel)%  qVal=0.0_rk
        me(iLevel)%   cIn=0.0_rk
        me(iLevel)%  cOut=0.0_rk
        me(iLevel)%  cNgh=0.0_rk
        me(iLevel)% inPos=0
        me(iLevel)%outPos=0
        me(iLevel)%nghPos=0
      end if
    end do ! iLevel

  end subroutine mus_alloc_bouzidi
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  subroutine mus_alloc_fieldBC( me, minLevel, maxLevel )
    ! -------------------------------------------------------------------- !
    type( boundary_type ) :: me
    integer, intent(in) :: minLevel, maxLevel
    ! -------------------------------------------------------------------- !

    if ( .not. allocated( me%elemLvl )) allocate( me%elemLvl(minLevel:maxLevel))
    if ( .not. allocated( me%neigh ) ) allocate( me%neigh(minLevel:maxLevel) )
    if ( .not. allocated( me%links ) ) allocate( me%links(minLevel:maxLevel) )

  end subroutine mus_alloc_fieldBC
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  !> Set necessary data for BC velocity_bounceback_qval
  subroutine mus_set_inletUbb( me, tree, stencil, nScalars,       &
    &                          globBC, levelDesc, varPos, nLinks, &
    &                          minLevel, maxLevel                 )
    ! -------------------------------------------------------------------- !
    !> setting type for bc
    type(bc_inlet_bouzidi_type), allocatable, intent(out) :: me(:)
    !> using mesh information
    type(treelmesh_type), intent(in)         :: tree
    !> for directions
    type(tem_stencilHeader_type), intent(in) :: stencil
    !> number of scalars
    integer, intent(in)                      :: nScalars
    !> for number of elements in boundary and position in buffer
    type(glob_boundary_type), intent(in)  :: globBC
    !> minimum and maximum level
    integer, intent(in)                   :: minLevel, maxLevel
    !> level descriptor
    type(tem_levelDesc_type), intent(in)  :: levelDesc(minLevel:maxLevel)
    !> for position of outgoing elements
    integer, intent(in)                   :: varPos(:)
    !> for linkwise treatment
    integer, intent(in)                   :: nLinks(minLevel:maxLevel)
    ! -------------------------------------------------------------------- !
    integer :: iLink, iDir, invDir, iElem, posInBCBuf
    integer :: iLevel
    real(kind=rk) :: dx, qVal
    real(kind=rk), dimension(3) :: bary
    ! -------------------------------------------------------------------- !

    write(logUnit(10), "(A)") 'Setting inletUbb type ...'

    allocate( me(minLevel:maxLevel) )

    ! loop over levels
    do iLevel = minLevel, maxLevel
      if (nLinks(iLevel) > 0) then
        write(logUnit(10), "(2(A,I0))") 'level: ', iLevel, ', nVal: ', &
          &                             nLinks(iLevel)
      end if
      if (nLinks(iLevel) >= 0) then
        ! allocate arrays
        allocate( me(iLevel)%  qVal(nLinks(iLevel)) )
        allocate( me(iLevel)%outPos(nLinks(iLevel)) )
        allocate( me(iLevel)%  iDir(nLinks(iLevel)) )
        allocate( me(iLevel)%  posInBuffer(nLinks(iLevel)) )
        ! Intialize arrays with zero
        do iLink = 1, nLinks(iLevel)
          me(iLevel)%  qVal(iLink) = 0.0_rk
          me(iLevel)%outPos(iLink) = 0
          me(iLevel)%  iDir(iLink) = 0
          me(iLevel)%  posInBuffer(iLink) = -99
        end do
      end if

      ! fill arrays
      iLink = 0
      dx    = tem_ElemSizeLevel(tree, iLevel)

      ! loop over elements
      do iElem = 1, globBC%nElems(iLevel)
        ! position in buffer and position of barycenter
        posInBCBuf = globBC%elemLvl( iLevel )%posInBcElemBuf%val(iElem)
        bary(:)    = tem_BaryOfId(tree, levelDesc(iLevel)          &
          &                        %total(globBC%elemLvl(iLevel)   &
          &                        %elem%val(iElem) ) )

        ! loop over directions
        do iDir = 1, stencil%QQN
          !> Bitmask is true for incoming direction
          if ( globBC%elemLvl(iLevel)%bitmask%val(iDir, iElem) ) then

            invDir  = stencil%cxDirInv(iDir)
            iLink   = iLink + 1
            ! posInBcElemBuf is needed for bcBuffer
            me(iLevel)%posInBuffer(iLink) = posInBCBuf
            ! qValues
            qVal    = globBC%elemLvl(iLevel)%qVal%val(invDir, iElem)
            !me(iLevel)%   qVal(iLink) = qVal needed here?
            ! position for outgoing elememts
            me(iLevel)% outPos(iLink) = varPos(invDir)            &
              &                                   + (posInBCBuf-1) * nScalars
            ! store directions to access stencil weight and cxDirRK
            me(iLevel)%   iDir(iLink) = iDir

          end if
        end do ! iDir

      end do ! iElem

    end do ! iLevel

  end subroutine mus_set_inletUbb
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
 subroutine mus_set_inletBfl( me, tree, stencil, nScalars, globBC,           &
    &                         levelDesc, varPos, nLinks, minLevel, maxLevel  )
    ! -------------------------------------------------------------------- !
    !> setting type for bc
    type(bc_inlet_bouzidi_type), allocatable, intent(out) :: me(:)
    !> using mesh information
    type(treelmesh_type), intent(in)         :: tree
    !> for directions
    type(tem_stencilHeader_type), intent(in) :: stencil
    !> number of scalars
    integer, intent(in)                      :: nScalars
    !> for number of elements in boundary and position in buffer
    type(glob_boundary_type), intent(in)     :: globBC
    !> minimum and maximum level
    integer, intent(in)                      :: minLevel, maxLevel
    !> level descriptor
    type(tem_levelDesc_type), intent(in)     :: levelDesc(minLevel:maxLevel)
    !> for position of elements
    integer, intent(in)                      :: varPos(:)
    !> for linkwise treatment
    integer, intent(in)          :: nLinks(minLevel:maxLevel)
    ! -------------------------------------------------------------------- !
    integer :: iLink, iDir, invDir, iElem, posInBCBuf
    integer :: iLevel
    real(kind=rk) :: dx, qVal
    real(kind=rk), dimension(3) :: bary
    ! -------------------------------------------------------------------- !

    write(logUnit(10), "(A)") 'Setting inletBFL ...'

    allocate( me(minLevel:maxLevel) )

    ! loop over levels
    do iLevel = minLevel, maxLevel
      if (nLinks(iLevel) > 0) then
        write(logUnit(10),"(2(A,I0))") 'level: ', iLevel, &
          &                            ', nVal: ', nLinks(iLevel)
      end if
      if (nLinks(iLevel) >= 0) then
        ! allocate arrays
        ! allocate( me(iLevel)%  qVal(nLinks(iLevel)) )
        allocate( me(iLevel)%   cIn(nLinks(iLevel)) )
        allocate( me(iLevel)%  cOut(nLinks(iLevel)) )
        allocate( me(iLevel)%  cNgh(nLinks(iLevel)) )
        allocate( me(iLevel)%  cVel(nLinks(iLevel)) )
        allocate( me(iLevel)% inPos(nLinks(iLevel)) )
        allocate( me(iLevel)%outPos(nLinks(iLevel)) )
        allocate( me(iLevel)%nghPos(nLinks(iLevel)) )
        allocate( me(iLevel)%  iDir(nLinks(iLevel)) )
        allocate( me(iLevel)%  posInBuffer(nLinks(iLevel)) )
        ! Intialize arrays with zero
        ! me(iLevel)%  qVal = 0.0_rk
        do iLink = 1, nLinks(iLevel)
          me(iLevel)%   cIn(iLink) = 0.0_rk
          me(iLevel)%  cOut(iLink) = 0.0_rk
          me(iLevel)%  cNgh(iLink) = 0.0_rk
          me(iLevel)% inPos(iLink) = 0
          me(iLevel)%outPos(iLink) = 0
          me(iLevel)%nghPos(iLink) = 0
          me(iLevel)%  iDir(iLink) = 0
          me(iLevel)%  posInBuffer(iLink) = -99
        end do
      end if

      ! fill arrays
      iLink = 0
      dx    = tem_ElemSizeLevel( tree, iLevel )

      ! loop over elements
      do iElem = 1, globBC%nElems(iLevel)

        ! position in buffer and position of barycenter
        posInBCBuf = globBC%elemLvl( iLevel )%posInBcElemBuf%val( iElem )
        bary(:)    = tem_BaryOfId( tree, levelDesc( iLevel )        &
          &                        %total(globBC%elemLvl ( iLevel ) &
          &                        %elem%val( iElem ) ) )

        ! loop over directions
        do iDir = 1, stencil%QQN
          !> Bitmask is true for incoming direction
          if ( globBC%elemLvl(iLevel)%bitmask%val(iDir, iElem) ) then
            invDir  = stencil%cxDirInv(iDir)
            iLink   = iLink + 1
            ! posInBcElemBuf is needed for bcBuffer
            me(iLevel)%posInBuffer(iLink) = posInBCBuf
            ! qValues
            qVal = globBC%elemLvl( iLevel )%qVal%val( invDir, iElem )
            ! coefficients for state values
            call set_bouzidi_coeff(               &
              &    qVal = qVal,                   &
              &    cIn  = me(iLevel)%cIn(iLink),  &
              &    cOut = me(iLevel)%cOut(iLink), &
              &    cNgh = me(iLevel)%cNgh(iLink), &
              &    cVel = me(iLevel)%cVel(iLink)  )
            ! positions of incoming, outgoing and neighbour elemets
            ! inPos and outPos are used in bcBuffer
            me(iLevel)%  inPos(iLink) = varPos(iDir) &
              &                        + (posInBCBuf-1) * nScalars
            me(iLevel)% outPos(iLink) = varPos(invDir) &
              &                        + (posInBCBuf-1) * nScalars
            ! nghPos is used in computeNeighBuf
            me(iLevel)% nghPos(iLink) = invDir + (iElem-1)*stencil%QQ
            ! store directions to access stencil weight and cxDirRK
            me(iLevel)%   iDir(iLink) = iDir

          end if
        end do ! iDir

      end do ! iElem

    end do ! iLevel

  end subroutine mus_set_inletBfl
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  !> Linkwise non-equilibrium extrapolation (can handle curved walls)
  !!
  !! All the coefficients are pre-calculated here and then used for the specific
  !! boundary conditions
  !! - potential_noneq_expol & potential_neumann
  !! - velocity_noneq_expol & pressure_noneq_expol
  !! - spc_moledens_noneq_expol & spc_solvent_noneq_expol
  !!
  !! The pdf is decomposed into equilibrium (eq) and non-equilibrium (neq) part:
  !! f = f_eq + f_neq
  !! - f_eq is calculated by weighting a fictitious X, which is obtained
  !!   by an extrapolation using the fluid neighbor(s)
  !! - f_neq is approximated by second-order extrapolation using the fluid
  !!   neighbor(s)
  !! - for qVal < 0.75 even the second neighbor is used for the extrapolations
  !!
  subroutine mus_set_nonEqExpol( me, curved, tree, stencil, nScalars, globBC, &
    &                            bc_neigh, pdf, levelDesc, varPos, nLinks,    &
    &                            minLevel, maxLevel  )
    ! -------------------------------------------------------------------- !
    !> minimum and maximum level
    integer, intent(in)                      :: minLevel, maxLevel
    !> setting type for bc
    type(bc_nonEqExpol_type), allocatable, intent(out) :: me(:)
    !> Curved or straight boundary
    logical, intent(in) :: curved
    !> using mesh information
    type(treelmesh_type), intent(in)         :: tree
    !> for directions
    type(tem_stencilHeader_type), intent(in) :: stencil
    !> number of scalars
    integer, intent(in)                      :: nScalars
    !> scheme global boundary type
    type(glob_boundary_type), intent(in)     :: globBC
    !> boundary neighbor
    type(bc_neigh_type), intent(in)          :: bc_neigh(minLevel:maxLevel)
    !> contains global state vector
    type( pdf_data_type ), intent(in)        :: pdf( tree%global%minlevel   &
      &                                              : tree%global%maxlevel )
    !> level descriptor
    type(tem_levelDesc_type), intent(in)     :: levelDesc(minLevel:maxLevel)
    !> for position of elements
    integer, intent(in)                      :: varPos(:)
    !> for linkwise treatment
    integer, intent(in)                      :: nLinks(minLevel:maxLevel)
    ! -------------------------------------------------------------------- !
    integer :: iLink, invDir, iElem
    integer :: iLevel, iDir, elemPos, QQ, QQN
    real(kind=rk) :: qVal
    ! -------------------------------------------------------------------- !

    write(logUnit(10), "(A)") 'Setting nonEqExpol ...'

    QQ  = stencil%QQ
    QQN = stencil%QQN

    allocate( me(minLevel:maxLevel) )

    ! loop over levels
    do iLevel = minLevel, maxLevel
      if (nLinks(iLevel) > 0) then
        write(logUnit(10),"(2(A,I0))") 'level: ', iLevel, ', nVal: ', &
          &                            nLinks(iLevel)
      end if
      if (nLinks(iLevel) >= 0) then
        ! allocate arrays
        if (curved) then
          allocate( me(iLevel)%          c_w(nLinks(iLevel)) )
          allocate( me(iLevel)%          c_f(nLinks(iLevel)) )
          allocate( me(iLevel)%         c_ff(nLinks(iLevel)) )
          allocate( me(iLevel)%      c_neq_f(nLinks(iLevel)) )
          allocate( me(iLevel)%     c_neq_ff(nLinks(iLevel)) )
        end if
        allocate( me(iLevel)%  posInBuffer(nLinks(iLevel)) )
        allocate( me(iLevel)%posInNeighBuf(nLinks(iLevel)) )
        allocate( me(iLevel)%         iDir(nLinks(iLevel)) )
        allocate( me(iLevel)%  posInBCelems(nLinks(iLevel)) )

        ! Intialize arrays
        if (curved) then
          me(iLevel)%          c_w(:) = -99.0_rk
          me(iLevel)%          c_f(:) = -99.0_rk
          me(iLevel)%         c_ff(:) = -99.0_rk
          me(iLevel)%      c_neq_f(:) = -99.0_rk
          me(iLevel)%     c_neq_ff(:) = -99.0_rk
        end if
        me(iLevel)%posInNeighBuf(:) = -99
        me(iLevel)%         iDir(:) = -99
        me(iLevel)%  posInBuffer(:) = -99
        me(iLevel)%  posInBCelems(:) = -99
      end if

      ! fill arrays
      iLink = 1

      ! loop over elements
      do iElem = 1, globBC%nElems(iLevel)
        ! loop over directions
        do iDir = 1, QQN
          !> Bitmask is true for incoming direction
          if ( globBC%elemLvl(iLevel)%bitmask%val(iDir, iElem) ) then
            ! posInBcElemBuf is needed for bcBuffer
            me(iLevel)%posInBuffer(iLink) = globBC%elemLvl( iLevel ) &
              &                                   %posInBcElemBuf%val( iElem )

            !save iDir for current link
            me(iLevel)%iDir(iLink) = iDir

            ! elemPos is needed for overnext neighbor later on
            elemPos = globBC%elemLvl(iLevel)%elem%val( iElem )
            ! determine neighbor along incoming direction
            invDir  = stencil%cxDirInv(iDir)

            ! position of the boundary element of this link in globBC elems list
            me(iLevel)%posInBCelems(iLink) = iElem

            ! Position of fluid Neigbor in neighBufferPre/Post
            ! Use neighbor along the normal direction of boundary
            ! if neighbor along direction of link is not found or
            ! boundary is defined as straight boundary
            me(iLevel)%posInNeighBuf(iLink) = iElem

            ! For curved boundary find the neighbor along the incoming link
            if (curved) then
              ! qValues
              qVal = globBC%elemLvl( iLevel )%qVal%val( invDir, iElem )

              ! coefficients for state values depending on qVal
              ! compute coeffient for physical boundary (_w)
              if ( (qVal - 0.75_rk) >= 0.0_rk ) then
                me(iLevel)%c_w(iLink) = 1.0_rk / qVal
                ! compute coeffient for fluid c_f
                me(iLevel)%c_f(iLink) = ( qVal - 1.0_rk ) / qVal
                ! compute coeffient for overnext fluid c_ff
                me(iLevel)%c_ff(iLink) = 0.0_rk
                ! compute nonEq coefficient for fluid c_nEq_f
                me(iLevel)%c_nEq_f(iLink) = 1.0_rk
                ! compute nonEq coefficient for overnext fluid c_nEq_ff
                me(iLevel)%c_nEq_ff(iLink) = 0.0_rk
              else
                me(iLevel)%c_w(iLink) = 1.0_rk + ( 2.0_rk * (1.0_rk - qVal) ) &
                  &                           / ( 1.0_rk + qVal )
                ! compute coeffient for fluid c_f
                me(iLevel)%c_f(iLink) = qVal - 1.0_rk
                ! compute coeffient for overnext fluid c_ff
                me(iLevel)%c_ff(iLink) = ( (1.0_rk - qVal)    &
                  &                       * (qVal - 1.0_rk) ) &
                  &                      / (1.0_rk + qVal)

                me(iLevel)%c_nEq_f(iLink) = qVal
                me(iLevel)%c_nEq_ff(iLink) = 1.0_rk - qVal
              end if ! qVal > 0.75
            end if ! curved

            ! increase link counter
            iLink   = iLink + 1

          end if ! bitmask
        end do ! iDir
      end do ! iElem
    end do ! iLevel

  end subroutine mus_set_nonEqExpol
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  subroutine mus_set_outletExpol( me, stencil, globBC,      &
    &                             nLinks, minLevel, maxLevel )
    ! -------------------------------------------------------------------- !
    !> setting type for bc
    type(bc_outlet_type), allocatable, intent(out) :: me(:)
    !> for directions
    type(tem_stencilHeader_type), intent(in) :: stencil
    !> number of scalars
    ! integer, intent(in)                      :: nScalars
    !> for number of elements in boundary and position in buffer
    type(glob_boundary_type), intent(in)     :: globBC
    !> minimum and maximum level
    integer, intent(in) :: minLevel, maxLevel
    !> for position of elements
    ! integer, intent(in)                      :: varPos(:)
    !> Number of links on each level
    integer, intent(in) :: nLinks(minLevel:maxLevel)
    ! -------------------------------------------------------------------- !
    integer :: iLevel, iLink, iElem, iDir
    ! -------------------------------------------------------------------- !

    write(logUnit(10), "(A)") 'Setting outlet expol ...'

    allocate( me(minLevel:maxLevel) )

    ! loop over levels
    do iLevel = minLevel, maxLevel
      if ( nLinks(iLevel) > 0) then
        write(logUnit(10), '(2(A,I0))') 'level: ', iLevel, &
          &                             ', nLinks: ', nLinks(iLevel)
      end if
      if ( nLinks(iLevel) >= 0) then
        ! allocate arrays
        allocate( me(iLevel)%statePos(nLinks(iLevel)) )
        allocate( me(iLevel)%iElem(nLinks(iLevel)) )
        ! Intialize arrays with zero
        me(iLevel)%statePos = 0
      end if

      iLink = 0

      ! loop over elements
      do iElem = 1, globBC%nElems(iLevel)
        ! position in buffer is needed afterwards
        ! posInBCBuf = globBC%elemLvl( iLevel )%posInBcElemBuf%val(iElem)

        ! loop over directions
        do iDir = 1, stencil%QQN
          ! check bitmask
          if ( globBC%elemLvl(iLevel)%bitmask%val(iDir, iElem) ) then
            ! invDir  = stencil%cxDirInv(iDir)
            iLink   = iLink + 1
            ! store position
            me(iLevel)%statePos(iLink) = iDir + (iElem-1)*stencil%QQ
            me(iLevel)%iElem(iLink) = iElem
          end if ! check bitmask

        end do ! iDir

      end do ! iElem

    end do ! iLevel

  end subroutine mus_set_outletExpol
  ! ------------------------------------------------------------------------ !


  ! ------------------------------------------------------------------------ !
  !> This routines deallocates allocatables in field%bc boundary_type for
  !! dynamic load balancing
  subroutine mus_fieldBC_cleanup( me, nBCs, minLevel, maxLevel )
    ! -------------------------------------------------------------------- !
    integer, intent(in) :: nBCs, minLevel, maxLevel
    type( boundary_type ), intent(inout) :: me(nBCs)
    ! -------------------------------------------------------------------- !
    integer :: iBC, iLevel
    ! -------------------------------------------------------------------- !

    write(dbgUnit(1),*) "Enter mus_fieldBC_cleanup"

    do iBC = 1, nBCs
      if( me(iBC)%nNeighs > 0 ) then
        do iLevel = minLevel, maxLevel
          deallocate( me(iBC)%neigh( iLevel )%posInState )
          if( me(iBC)%requireNeighBufPre_nNext  ) &
            & deallocate( me(iBC)%neigh( iLevel )%neighBufferPre_nNext )
          if( me(iBC)%requireNeighBufPre  ) &
            & deallocate( me(iBC)%neigh( iLevel )%neighBufferPre )
          if( me(iBC)%requireNeighBufPost ) &
            & deallocate( me(iBC)%neigh( iLevel )%neighBufferPost )
          deallocate( me(iBC)%elemLvl( iLevel )%stencilPos )
          deallocate( me(iBC)%elemLvl( iLevel )%posInNghElems )
        end do
      end if

      if( me(iBC)%useComputeNeigh ) then
        do iLevel = minLevel, maxLevel
          deallocate( me(iBC)%neigh( iLevel )%computeNeighBuf )
        end do
      end if

      do iLevel = minLevel, maxLevel

        ! @todo: for pure wall, following array is not allocated
        !        we should have better way to detect this.
        if ( allocated( me(iBC)%links( iLevel )%val ) ) then
          deallocate( me(iBC)%links( iLevel )%val )
        end if

        if ( allocated( me(iBC)%elemLvl( iLevel )%lodi ) ) then
          deallocate( me(iBC)%elemLvl( iLevel )%lodi )
        end if

        if ( allocated( me(iBC)%elemLvl( iLevel )%moments ) ) then
          deallocate( me(iBC)%elemLvl( iLevel )%moments )
        end if

        if ( allocated( me(iBC)%bouzidi ) ) then
          deallocate( me(iBC)%bouzidi( iLevel )%qVal )
          deallocate( me(iBC)%bouzidi( iLevel )%cIn  )
          deallocate( me(iBC)%bouzidi( iLevel )%cOut )
          deallocate( me(iBC)%bouzidi( iLevel )%cNgh )
          deallocate( me(iBC)%bouzidi( iLevel )%inPos )
          deallocate( me(iBC)%bouzidi( iLevel )%outPos )
          deallocate( me(iBC)%bouzidi( iLevel )%nghPos )
        end if

        if ( allocated( me(iBC)%inletUbbQVal ) ) then
          deallocate( me(iBC)%inletUbbQVal( iLevel )%qVal )
          deallocate( me(iBC)%inletUbbQVal( iLevel )%outPos )
          deallocate( me(iBC)%inletUbbQVal( iLevel )%iDir )
          deallocate( me(iBC)%inletUbbQVal( iLevel )%posInBuffer )
        end if

        if ( allocated( me(iBC)%inletBFL ) ) then
          deallocate( me(iBC)%inletBFL( iLevel )%cIn )
          deallocate( me(iBC)%inletBFL( iLevel )%cOut )
          deallocate( me(iBC)%inletBFL( iLevel )%cNgh )
          deallocate( me(iBC)%inletBFL( iLevel )%inPos )
          deallocate( me(iBC)%inletBFL( iLevel )%outPos )
          deallocate( me(iBC)%inletBFL( iLevel )%nghPos )
          deallocate( me(iBC)%inletBFL( iLevel )%cVel )
          deallocate( me(iBC)%inletBFL( iLevel )%iDir )
          deallocate( me(iBC)%inletBFL( iLevel )%posInBuffer )
        end if

        if ( allocated( me(iBC)%nonEqExpol ) ) then
          if ( me(iBC)%curved ) then
            deallocate( me(iBC)%nonEqExpol( iLevel )%c_w )
            deallocate( me(iBC)%nonEqExpol( iLevel )%c_f)
            deallocate( me(iBC)%nonEqExpol( iLevel )%c_ff )
            deallocate( me(iBC)%nonEqExpol( iLevel )%c_neq_f )
            deallocate( me(iBC)%nonEqExpol( iLevel )%c_neq_ff )
          end if
          deallocate( me(iBC)%nonEqExpol( iLevel )%posInNeighBuf )
          deallocate( me(iBC)%nonEqExpol( iLevel )%iDir )
          deallocate( me(iBC)%nonEqExpol( iLevel )%posInBuffer )
          deallocate( me(iBC)%nonEqExpol( iLevel )%posInBCelems )
        end if

        if ( allocated( me(iBC)%outletExpol ) ) then
          deallocate( me(iBC)%outletExpol( iLevel )%statePos )
          deallocate( me(iBC)%outletExpol( iLevel )%iElem )
        end if

        if ( allocated( me(iBC)%turbWallFunc%dataOnLvl ) ) then
          deallocate( me(iBC)%turbWallFunc%dataOnLvl( iLevel )%tVisc )
          deallocate( me(iBC)%turbWallFunc%dataOnLvl( iLevel )%velTau )
          deallocate( me(iBC)%turbWallFunc%dataOnLvl( iLevel )%distToBnd )
          deallocate( me(iBC)%turbWallFunc%dataOnLvl( iLevel )%neighDistToBnd )
        end if

      end do ! iLevel = minLevel, maxLevel

      if ( allocated( me(iBC)%links ) ) then
        deallocate( me(iBC)%neigh )
        deallocate( me(iBC)%elemLvl )
        deallocate( me(iBC)%links )
      end if

      if ( allocated( me(iBC)%bouzidi ) ) then
        deallocate( me(iBC)%bouzidi )
      end if
      if ( allocated( me(iBC)%inletUBBqVal ) ) then
        deallocate( me(iBC)%inletUBBqVal )
      end if
      if ( allocated( me(iBC)%inletBFL ) ) then
        deallocate( me(iBC)%inletBFL )
      end if
      if ( allocated( me(iBC)%nonEqExpol ) ) then
        deallocate( me(iBC)%nonEqExpol )
      end if
      if ( allocated( me(iBC)%outletExpol ) ) then
        deallocate( me(iBC)%outletExpol )
      end if
      if ( allocated( me(iBC)%turbWallFunc%dataOnLvl ) ) then
        deallocate( me(iBC)%turbWallFunc%dataOnLvl )
      end if
    end do

    write(dbgUnit(1),*) "Done!"

  end subroutine mus_fieldBC_cleanup
  ! ------------------------------------------------------------------------ !

end module mus_bc_header_module
! **************************************************************************** !