atl_modg_navierstokes_kernel_module.f90 Source File


This file depends on

sourcefile~~atl_modg_navierstokes_kernel_module.f90~~EfferentGraph sourcefile~atl_modg_navierstokes_kernel_module.f90 atl_modg_navierstokes_kernel_module.f90 sourcefile~atl_cube_elem_module.f90 atl_cube_elem_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_cube_elem_module.f90 sourcefile~atl_equation_module.f90 atl_equation_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_equation_module.f90 sourcefile~atl_facedata_module.f90 atl_facedata_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_facedata_module.f90 sourcefile~atl_materialprp_module.f90 atl_materialPrp_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_materialprp_module.f90 sourcefile~atl_modg_euler_kernel_module.f90 atl_modg_euler_kernel_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_modg_euler_kernel_module.f90 sourcefile~atl_modg_scheme_module.f90 atl_modg_scheme_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_modg_scheme_module.f90 sourcefile~atl_penalization_module.f90 atl_penalization_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_penalization_module.f90 sourcefile~atl_physfluxeuler_module.f90 atl_physFluxEuler_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_physfluxeuler_module.f90 sourcefile~atl_physfluxnvrstk_module.f90 atl_physFluxNvrstk_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_physfluxnvrstk_module.f90 sourcefile~atl_scheme_module.f90 atl_scheme_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_scheme_module.f90 sourcefile~atl_viscnumflux_nvrstk_module.f90 atl_viscNumFlux_Nvrstk_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~atl_viscnumflux_nvrstk_module.f90 sourcefile~ply_oversample_module.f90 ply_oversample_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~ply_oversample_module.f90 sourcefile~ply_poly_project_module.f90 ply_poly_project_module.f90 sourcefile~atl_modg_navierstokes_kernel_module.f90->sourcefile~ply_poly_project_module.f90

Files dependent on this one

sourcefile~~atl_modg_navierstokes_kernel_module.f90~~AfferentGraph sourcefile~atl_modg_navierstokes_kernel_module.f90 atl_modg_navierstokes_kernel_module.f90 sourcefile~atl_compute_module.f90 atl_compute_module.f90 sourcefile~atl_compute_module.f90->sourcefile~atl_modg_navierstokes_kernel_module.f90 sourcefile~atl_modg_filnvrstk_kernel_module.f90 atl_modg_filNvrStk_kernel_module.f90 sourcefile~atl_compute_module.f90->sourcefile~atl_modg_filnvrstk_kernel_module.f90 sourcefile~atl_modg_filnvrstk_kernel_module.f90->sourcefile~atl_modg_navierstokes_kernel_module.f90 sourcefile~atl_fwdeuler_module.f90 atl_fwdEuler_module.f90 sourcefile~atl_fwdeuler_module.f90->sourcefile~atl_compute_module.f90 sourcefile~atl_imexrk_module.f90 atl_imexrk_module.f90 sourcefile~atl_imexrk_module.f90->sourcefile~atl_compute_module.f90 sourcefile~atl_predcor_cerk4_module.f90 atl_predcor_cerk4_module.f90 sourcefile~atl_predcor_cerk4_module.f90->sourcefile~atl_compute_module.f90 sourcefile~atl_rk4_module.f90 atl_rk4_module.f90 sourcefile~atl_rk4_module.f90->sourcefile~atl_compute_module.f90 sourcefile~atl_rktaylor_module.f90 atl_rktaylor_module.f90 sourcefile~atl_rktaylor_module.f90->sourcefile~atl_compute_module.f90 sourcefile~atl_ssprk2_module.f90 atl_ssprk2_module.f90 sourcefile~atl_ssprk2_module.f90->sourcefile~atl_compute_module.f90 sourcefile~atl_global_time_integration_module.f90 atl_global_time_integration_module.f90 sourcefile~atl_global_time_integration_module.f90->sourcefile~atl_fwdeuler_module.f90 sourcefile~atl_global_time_integration_module.f90->sourcefile~atl_imexrk_module.f90 sourcefile~atl_global_time_integration_module.f90->sourcefile~atl_predcor_cerk4_module.f90 sourcefile~atl_global_time_integration_module.f90->sourcefile~atl_rk4_module.f90 sourcefile~atl_global_time_integration_module.f90->sourcefile~atl_rktaylor_module.f90 sourcefile~atl_global_time_integration_module.f90->sourcefile~atl_ssprk2_module.f90 sourcefile~atl_container_module.f90 atl_container_module.f90 sourcefile~atl_container_module.f90->sourcefile~atl_global_time_integration_module.f90 sourcefile~atl_program_module.f90 atl_program_module.f90 sourcefile~atl_program_module.f90->sourcefile~atl_global_time_integration_module.f90

Source Code

! Copyright (c) 2014 Jens Zudrop <j.zudrop@grs-sim.de>
! Copyright (c) 2014-2016 Nikhil Anand <nikhil.anand@uni-siegen.de>
! Copyright (c) 2014-2018 Harald Klimach <harald.klimach@uni-siegen.de>
! Copyright (c) 2014-2019 Peter Vitt <peter.vitt2@uni-siegen.de>
! Copyright (c) 2016 Parid Ndreka
! Copyright (c) 2016 Tobias Girresser <tobias.girresser@student.uni-siegen.de>
! Copyright (c) 2016 Jiaxing Qi <jiaxing.qi@uni-siegen.de>
! Copyright (c) 2017 Daniel PetrĂ³ <daniel.petro@student.uni-siegen.de>
! Copyright (c) 2018 Neda Ebrahimi Pour <neda.epour@uni-siegen.de>
!
! Permission to use, copy, modify, and distribute this software for any
! purpose with or without fee is hereby granted, provided that the above
! copyright notice and this permission notice appear in all copies.
!
! THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHORS DISCLAIM ALL WARRANTIES
! WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
! MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
! ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
! WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
! ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
! OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
! **************************************************************************** !

!> author: Jens Zudrop
!! Module for routines and datatypes of MOdal Discontinuous Galerkin (MODG)
!! scheme for the compressible Navier-Stokes equation. This scheme is a spectral
!! scheme for linear, convection dominated
!! partial differential equation systems.
module atl_modg_navierstokes_kernel_module
  use env_module,                    only: rk
  use tem_faceData_module,           only: tem_faceIterator_type

  use ply_poly_project_module,       only: ply_poly_project_type, &
    &                                      assignment(=),         &
    &                                      ply_poly_project_m2n,  &
    &                                      ply_poly_project_n2m
  use ply_oversample_module,         only: ply_convert2oversample,   &
   &                                       ply_convertFromoversample

  use atl_equation_module,           only: atl_equations_type
  use atl_cube_elem_module,          only: atl_cube_elem_type
  use atl_scheme_module,             only: atl_scheme_type
  use atl_modg_scheme_module,        only: atl_modg_scheme_type
  use atl_facedata_module,           only: atl_facedata_type
  use atl_modg_euler_kernel_module,              &
    & only: atl_modg_euler_oneDim_numFlux_const, &
    &       atl_modg_euler_oneDim_numFlux_nonconst
  use atl_physFluxEuler_module,      only: atl_physFluxEuler
  use atl_physFluxNvrStk_module,     only: atl_viscPhysFluxNavierStokes
  use atl_viscNumFlux_Nvrstk_module, only: atl_viscNavierStokes
  use atl_materialPrp_module,        only: atl_material_type
  use atl_penalization_module,       only: atl_penalizationData_type

  implicit none

  private

  public :: atl_modg_navierstokes_numFlux,          &
    & atl_modg_navierstokes_physFlux_const,         &
    & atl_modg_navierstokes_physFlux_NonConst,      &
    & atl_modg_navierstokes_penalization_const,     &
    & atl_modg_navierstokes_penalization_NonConst,  &
    & atl_modg_stabViscNavierStokes_oneDim_numFlux, &
    & atl_modg_viscNavierStokes_oneDim_numFlux


contains


  ! ************************************************************************ !
  !> Calculate the physical flux for the MODG scheme and
  !! Navier-Stokes equation (with constant penalizations).
  subroutine atl_modg_navierstokes_physFlux_const( equation, res, state, &
    & iElem, iDir, penalizationData, poly_proj, material, nodal_data,    &
    & nodal_gradData, nodal_res, elemLength, scheme_min, scheme_current  )
    ! -------------------------------------------------------------------- !
    !> The equation you solve.
    type(atl_equations_type), intent(in) :: equation
    !> To store the resulting phy flux in modal form
    real(kind=rk), intent(inout)     :: res(:,:)
    !> The state of the equation
    real(kind=rk), intent(in), optional :: state(:,:)
    !> The current Element
    integer, intent(in) :: iElem
    !> The current Direction
    integer, intent(in) :: iDir
    !> The penalization data
    type(atl_penalizationData_type), intent(inout) :: penalizationData
    !> Poly project
    type(ply_poly_project_type), intent(inout) :: poly_proj
    !> Material description for the faces on the current level
    type(atl_material_type), intent(inout) :: material
    !> The state in nodal form
    real(kind=rk), intent(in), optional :: nodal_data(:,:)
    real(kind=rk), intent(in), optional :: nodal_GradData(:,:,:)
    real(kind=rk), intent(inout) :: nodal_res(:,:)
    !> Length of the element
    real(kind=rk), intent(in) :: ElemLength
    !> The scheme information
    type(atl_scheme_type), intent(inout) :: scheme_min
    type(atl_scheme_type), intent(inout) :: scheme_current
    ! -------------------------------------------------------------------- !
    integer :: iPoint
    integer :: rot(5), derRot(3)
    ! -------------------------------------------------------------------- !

    ! get the rotation for the physical flux calculation in y direction
    rot = equation%varRotation(iDir)%varTransformIndices(1:5)
    ! get rotation for the derivatives
    derRot = equation%varRotation(iDir)%derTransformIndices(2:4) &
      &      - equation%varRotation(iDir)%derTransformIndices(1)


    ! Calculate the physical flux point by point within this cell
    do iPoint = 1, poly_proj%body_3D%nquadpoints
      scheme_min%temp_nodal(iPoint,rot,1) = atl_physFluxEuler(                &
        & state        = nodal_data(iPoint,rot),                              &
        & isenCoeff    = equation%euler%isen_coef,                            &
        & penalty_char = material%material_dat%elemMaterialData(1)            &
        &                                     %materialDat(iElem, 1, 1),      &
        & U_o          = material%material_dat%elemMaterialData(1)            &
        &                                     %materialDat(iElem, 1, iDir+1), &
        & porosity     = equation%euler%porosity                              )
    end do

    ! Calculate viscous physical flux point by point within this cell
    do iPoint = 1, poly_proj%body_3D%nquadpoints
      scheme_min%temp_nodal(iPoint,rot,2) = atl_viscPhysFluxNavierStokes( &
        & state          = nodal_data(iPoint,rot),                        &
        & state_gradient = nodal_GradData(iPoint,rot, DerRot),            &
        & mu             = equation%NavierStokes%mu,                      &
        & lambda         = equation%NavierStokes%lambda,                  &
        & thermCond      = equation%NavierStokes%therm_cond,              &
        & heatCap        = equation%euler%cv                              )
    end do

    ! Add up the nodal data
    nodal_res(:,:) = scheme_min%temp_nodal(:,:,1) - scheme_min%temp_nodal(:,:,2)


  end subroutine atl_modg_navierstokes_physFlux_const
  ! ************************************************************************ !


  ! ************************************************************************ !
  ! Calculate the penalization terms (for density, momentum, energy)
  ! The penalization terms are calculated in the sammer manner as
  ! the physical fluxes, i.e. in a nodal-modal approach
  subroutine atl_modg_navierstokes_penalization_const(equation, poly_proj, &
    & nodal_data, scheme_min, penalizationData, iElem, material            )
    ! -------------------------------------------------------------------- !
    !> The equation you solve.
    type(atl_equations_type), intent(in) :: equation
    !> Poly project
    type(ply_poly_project_type), intent(inout) :: poly_proj
    !> The state in nodal form
    real(kind=rk), intent(in), optional :: nodal_data(:,:)
    !> The scheme information
    type(atl_scheme_type), intent(inout) :: scheme_min
    !> The penalization data
    type(atl_penalizationData_type), intent(inout) :: penalizationData
    !> The current Element
    integer, intent(in) :: iElem
    !> Material description for the faces on the current level
    type(atl_material_type), intent(inout) :: material
    ! -------------------------------------------------------------------- !
    integer :: iPoint
    integer :: glob_elem_ind
    real(kind=rk) :: temperature, pressure
    real(kind=rk) :: penalization(5)
    ! -------------------------------------------------------------------- !
    ! Calculate the penalization terms (for density, momentum, energy)
    ! The penalization terms are calculated in the samme way as the physical
    ! fluxes, i.e. in a nodal-modal approach
    ! Penalization for momentum and energy in pointwise manner
    !> @todo PV 20150820 Get the correct penalization data here
    penalization = material%material_dat%elemMaterialData(1)  &
      &                                 %materialDat(iElem,1,:)
    glob_elem_ind = material%material_desc%computeElems(1)%totElemIndices(iElem)

    do iPoint = 1, poly_proj%body_3D%nquadpoints
      pressure                                   &
        & = (equation%euler%isen_coef-1.0_rk)    &
        &   * ( nodal_data(iPoint,5)             &
        &     - 0.5_rk                           &
        &       * sum(nodal_data(iPoint,2:4)**2) &
        &       / nodal_data(iPoint,1) )
      temperature = pressure / ( nodal_data(iPoint,1) * equation%euler%R )

      ! ... momentum
      scheme_min%temp_nodal(iPoint,1:3,1)         &
        & = (-1.0_rk)                             &
        &   * penalization(1)                     &
        &   * ( nodal_data(iPoint,2:4)            &
        &     / nodal_data(iPoint,1)              &
        &     - penalization(2:4) )               &
        &   / equation%euler%viscous_permeability
      ! ... energy
      scheme_min%temp_nodal(iPoint,4,1)           &
        & = (-1.0_rk)                             &
        &   * penalization(1)                     &
        &   * ( temperature - penalization(5) )   &
        &   / equation%euler%thermal_permeability
    end do

    ! Transform penalizations back to modal space
    call ply_poly_project_n2m( me         = poly_proj,                    &
      &                        dim        = 3,                            &
      &                        nVars      = 4,                            &
      &                        nodal_data = scheme_min%temp_nodal(:,:,1), &
      &                        modal_data = scheme_min%temp_modal(:,:,1)  )

    ! ... no penalization for density

    penalizationdata%penalization_data(glob_elem_ind,:,1) = 0.0_rk

    ! --> oversamp modal space for penalty terms (momentum + energy)
    call ply_convertFromOversample(                                           &
      & modalCoeffs = scheme_min%temp_over(:,:,1),                            &
      & poly_proj   = poly_proj,                                              &
      & nDim        = 3,                                                      &
      & state       = penalizationdata%penalization_data(glob_elem_ind,:,2:5) )


  end subroutine atl_modg_navierstokes_penalization_const
  ! ************************************************************************ !


  ! ************************************************************************ !
  !> Calculate the physical flux for the MODG scheme and
  !! Navier-Stokes equation (with non-constant penalizations).
  subroutine atl_modg_navierstokes_physFlux_NonConst( equation, res, state, &
    & iElem, iDir, penalizationData, poly_proj, material, nodal_data,       &
    & nodal_GradData, nodal_res, elemLength, scheme_min, scheme_current     )
    ! -------------------------------------------------------------------- !
    !> The equation you solve.
    type(atl_equations_type), intent(in) :: equation
    !> To store the resulting phy flux in modal form
    real(kind=rk), intent(inout) :: res(:,:)
    !> The state of the equation
    real(kind=rk), intent(in), optional :: state(:,:)
    !> The current Element
    integer, intent(in) :: iElem
    !> The current Direction
    integer, intent(in) :: iDir
    !> The penalization data
    type(atl_penalizationData_type), intent(inout) :: penalizationData
    !> Poly project
    type(ply_poly_project_type), intent(inout) :: poly_proj
    !> Material description for the faces on the current level
    type(atl_material_type), intent(inout) :: material
    !> The state in nodal form
    real(kind=rk), intent(in), optional :: nodal_data(:,:)
    real(kind=rk), intent(in), optional :: nodal_GradData(:,:,:)
    real(kind=rk), intent(inout) :: nodal_res(:,:)
    !> Length of the element
    real(kind=rk), intent(in) :: ElemLength
    !> The scheme information
    type(atl_scheme_type), intent(inout) :: scheme_min
    type(atl_scheme_type), intent(inout) :: scheme_current
    ! -------------------------------------------------------------------- !
    integer :: iPoint
    ! Loop var for all the fluid elements
    integer :: rot(5), derRot(3)
    real(kind=rk) :: penalization(poly_proj%body_3D%nquadpoints)
    ! -------------------------------------------------------------------- !
    ! get correct amount of quadrature points and degree due to projection
    ! method. oversamp_dof and oversamp_degree is used for the oversampling
    ! loop
    penalization = material%material_dat%elemMaterialData(2)  &
      &                                 %materialDat(iElem,:,1)


    ! get the rotation for the physical flux calculation
    rot = equation%varRotation(iDir)%varTransformIndices(1:5)
    ! get rotation for the derivatives
    derRot = equation%varRotation(iDir)%derTransformIndices(2:4) &
      &        - equation%varRotation(iDir)%derTransformIndices(1)


    do iPoint = 1, poly_proj%body_3D%nquadpoints
      scheme_min%temp_nodal(iPoint,rot,1) = atl_physFluxEuler(        &
        & state        = nodal_data(iPoint,rot),                      &
        & isenCoeff    = equation%euler%isen_coef,                    &
        & penalty_char = penalization(iPoint),                        &
        & porosity     = equation%euler%porosity,                     &
        & U_o          = material%material_dat                        &
        &                        %elemMaterialData(2)                 &
        &                        %materialDat(iElem, iPoint, iDir + 1))
    end do

    do iPoint = 1, poly_proj%body_3D%nquadpoints
      scheme_min%temp_nodal(iPoint,rot,2) = atl_viscPhysFluxNavierStokes( &
        & state          = nodal_data(iPoint,rot),                        &
        & state_gradient = nodal_GradData(iPoint,rot,derRot),             &
        & mu             = equation%NavierStokes%mu,                      &
        & lambda         = equation%NavierStokes%lambda,                  &
        & thermCond      = equation%NavierStokes%therm_cond,              &
        & heatCap        = equation%euler%cv                              )
    end do

    ! Add up the nodal data
    nodal_res(:,:) =  scheme_min%temp_nodal(:,:,1)   &
      &                 - scheme_min%temp_nodal(:,:,2)


  end subroutine atl_modg_navierstokes_physFlux_nonconst
  ! ************************************************************************ !



  ! ************************************************************************ !
  ! Calculate the penalization terms (for density, momentum, energy)
  ! The penalization terms are calculated in the sammer manner as
  ! the physical fluxes, i.e. in a nodal-modal approach
  subroutine atl_modg_navierstokes_penalization_nonconst( equation, poly_proj, &
    & nodal_data, scheme_min, penalizationData, iElem, material                )
    ! -------------------------------------------------------------------- !
    !> The equation you solve.
    type(atl_equations_type), intent(in) :: equation
    !> Poly project
    type(ply_poly_project_type), intent(inout) :: poly_proj
    !> The state in nodal form
    real(kind=rk), intent(in), optional :: nodal_data(:,:)
    !> The scheme information
    type(atl_scheme_type), intent(inout) :: scheme_min
    !> The penalization data
    type(atl_penalizationData_type), intent(inout) :: penalizationData
    !> The current Element
    integer, intent(in) :: iElem
    !> Material description for the faces on the current level
    type(atl_material_type), intent(inout) :: material
    ! -------------------------------------------------------------------- !
    integer :: iPoint
    integer :: glob_elem_ind
    real(kind=rk) :: temperature, pressure
    real(kind=rk) :: penalization(5)
    ! -------------------------------------------------------------------- !

    ! Calculate the penalization terms (for density, momentum, energy)
    ! The penalization terms are calculated in the samme way as the physical
    ! fluxes,
    ! i.e. in a nodal-modal approach
    ! Penalization for momentum and energy in pointwise manner
    !> @todo PV 20150820 Get the correct penalization data here
    penalization = material%material_dat%elemMaterialData(1)  &
      &                                 %materialDat(iElem,1,:)
    glob_elem_ind = material%material_desc%computeElems(1)%totElemIndices(iElem)

    do iPoint = 1, poly_proj%body_3D%nquadpoints
      pressure = (equation%euler%isen_coef-1.0_rk) &
        & * ( nodal_data(iPoint,5)                 &
        &   - 0.5_rk                               &
        &     * sum(nodal_data(iPoint,2:4)**2)     &
        &     / nodal_data(iPoint,1) )
      temperature =  pressure / ( nodal_data(iPoint,1) * equation%euler%R )

      ! ... momentum
      scheme_min%temp_nodal(iPoint,1:3,1) = (-1.0_rk) &
        & * penalization(1)                           &
        & * ( nodal_data(iPoint,2:4)                  &
        &   / nodal_data(iPoint,1)                    &
        &   - penalization(2:4) )                     &
        & / equation%euler%viscous_permeability
      ! ... energy
      scheme_min%temp_nodal(iPoint,4,1) = (-1.0_rk) &
        & * penalization(1)                         &
        & * ( temperature                           &
        &   - penalization(5) )                     &
        & / equation%euler%thermal_permeability
    end do

    ! Transform penalizations back to modal space
    call ply_poly_project_n2m( me         = poly_proj,                    &
      &                        dim        = 3,                            &
      &                        nVars      = 4,                            &
      &                        nodal_data = scheme_min%temp_nodal(:,:,1), &
      &                        modal_data = scheme_min%temp_modal(:,:,1)  )

    ! ... no penalization for density

    penalizationdata%penalization_data(glob_elem_ind,:,1) = 0.0_rk

    ! --> oversamp modal space for penalty terms (momentum + energy)
    call ply_convertFromOversample(                                           &
      & modalCoeffs = scheme_min%temp_over(:,:,1),                            &
      & poly_proj   = poly_proj,                                              &
      & nDim        = 3,                                                      &
      & state       = penalizationdata%penalization_data(glob_elem_ind,:,2:5) )

  end subroutine atl_modg_navierstokes_penalization_nonconst
  ! ************************************************************************ !


  ! ************************************************************************ !
  !> Calculate the numerical flux for Navier-Stokes equation and MODG scheme
  subroutine atl_modg_navierstokes_numFlux( mesh, equation, facedata, scheme, &
    &                                       poly_proj, material               )
    ! -------------------------------------------------------------------- !
    !> The mesh you are working with.
    type(atl_cube_elem_type), intent(in) :: mesh
    !> The equation you solve.
    type(atl_equations_type), intent(in) :: equation
    !> The face representation of the state.
    type(atl_facedata_type), intent(inout) :: facedata
    !> Parameters of the modal dg scheme
    type(atl_modg_scheme_type), intent(in) :: scheme
    !> Parameter for used projection
    type(ply_poly_project_type), intent(inout) :: poly_proj
    !> Material description for the faces on the current level
    type(atl_material_type), intent(inout) :: material
    ! -------------------------------------------------------------------- !
    integer :: iDir
    ! -------------------------------------------------------------------- !

    ! Numerical flux for faces in all 2 spatial face directions (x and y dir)
    do iDir = 1, 3

      ! convective part of the Euler equations (F^*)
      ! ... fluxes for constant penalization parameters
      call atl_modg_euler_oneDim_numFlux_const(                                &
        & equation       = equation,                                           &
        & nSides         = size(material%material_desc%computeFace(iDir,1)     &
        &                                             %leftPos),               &
        & faceRep        = facedata%faceRep(iDir)%dat,                         &
        & faceFlux       = facedata%faceFlux(iDir)%dat,                        &
        & leftPos        = material%material_desc%computeFace(iDir,1)%leftPos, &
        & rightPos       = material%material_desc%computeFace(iDir,1)          &
        &                                        %rightPos,                    &
        & poly_proj      = poly_proj,                                          &
        & varRotation    = equation%varRotation(iDir)                          &
        &                          %varTransformIndices(1:5),                  &
        & material_left  = material%material_dat%faceMaterialData(iDir,1)      &
        &                                       %leftElemMaterialDat,          &
        & material_right = material%material_dat%faceMaterialData(iDir,1)      &
        &                                       %rightElemMaterialDat          )

      ! ... fluxes for non-constant penalization parameters
      call atl_modg_euler_oneDim_numFlux_nonconst(                             &
        & equation       = equation,                                           &
        & nSides         = size(material%material_desc%computeFace(iDir,2)     &
        &                                             %leftPos),               &
        & faceRep        = facedata%faceRep(iDir)%dat,                         &
        & faceFlux       = facedata%faceFlux(iDir)%dat,                        &
        & leftPos        = material%material_desc%computeFace(iDir,2)%leftPos, &
        & rightPos       = material%material_desc%computeFace(iDir,2)          &
        &                                        %rightPos,                    &
        & poly_proj      = poly_proj,                                          &
        & varRotation    = equation%varRotation(iDir)                          &
        &                          %varTransformIndices(1:5),                  &
        & material_left  = material%material_dat%faceMaterialData(iDir,2)      &
        &                                       %leftElemMaterialDat,          &
        & material_right = material%material_dat%faceMaterialData(iDir,2)      &
        &                                       %rightElemMaterialDat          )

      ! viscous numerical flux of the Navier-Stokes equation (sigma^*)
      call atl_modg_viscNavierStokes_oneDim_numFlux(       &
        & equation   = equation,                           &
        & facedata   = facedata,                           &
        & scheme     = scheme,                             &
        & faces      = mesh%faces%faces(iDir)%computeFace, &
        & faceDir    = iDir,                               &
        & poly_proj  = poly_proj,                          &
        & elemLength = mesh%length                         )

      ! stabilization viscous numerical flux of the Navier-Stokes equation (u^*)
      call atl_modg_stabViscNavierStokes_oneDim_numFlux(  &
        & equation  = equation,                           &
        & facedata  = facedata,                           &
        & scheme    = scheme,                             &
        & faces     = mesh%faces%faces(iDir)%computeFace, &
        & faceDir   = iDir,                               &
        & poly_proj = poly_proj                           )

    end do

  end subroutine atl_modg_navierstokes_numFlux
  ! ************************************************************************ !


  ! ************************************************************************ !
  !> Numerical flux calculation for viscous part of the Navier-Stokes equation
  !! across the faces in a single spatial direction.
  subroutine atl_modg_viscNavierStokes_oneDim_numFlux( equation, facedata, &
    & scheme, faces, faceDir, poly_proj, elemLength                        )
    ! -------------------------------------------------------------------- !
    !> The equation you solve.
    type(atl_equations_type), intent(in) :: equation
    !> The face state if the equation
    type(atl_facedata_type), intent(inout) :: facedata
    !> Parameters of the modal dg scheme
    type(atl_modg_scheme_type), intent(in) :: scheme
    !> The faces to calculate the fluxes for.
    type(tem_faceIterator_type), intent(in) :: faces
    !> The spatial direction of the faces you calc the fluxes for, use the
    !! following:
    !! 1 --> x direction. \n
    !! 2 --> y direction. \n
    !! 3 --> z direction.
    integer, intent(in) :: faceDir
    !> Parameter for used projection
    type(ply_poly_project_type), intent(inout) :: poly_proj
    !> The length of an element
    real(kind=rk), intent(in) :: elemLength
    ! -------------------------------------------------------------------- !
    ! Loop vars
    integer :: iPoint
    ! Modal coefficients for elements left and right of the face:
    ! First dimension is the number of modal coefficients on the face, second
    ! is the number of variables.
    real(kind=rk), allocatable :: leftModalCoeff(:,:),        &
      &                           rightModalCoeff(:,:),       &
      &                           leftModalCoeff_gradX(:,:),  &
      &                           rightModalCoeff_gradX(:,:), &
      &                           leftModalCoeff_gradY(:,:),  &
      &                           rightModalCoeff_gradY(:,:), &
      &                           leftModalCoeff_gradZ(:,:),  &
      &                           rightModalCoeff_gradZ(:,:)
    real(kind=rk) :: flux(equation%varSys%nScalars)
    ! Loop var for the faces.
    integer :: iside
    ! Element positions of the left and right element of the face.
    integer :: left_neighbor, right_neighbor
    ! The rotation indices for the flux calculation
    integer :: varRotation(equation%varSys%nScalars), gradRot(3)
    ! Nodal representation on the face (for left and right neighbor)
    real(kind=rk), allocatable :: pointValLeft(:,:), pointValRight(:,:)
    ! Nodal representations of gradients on the face (for left and right
    ! neighbor)
    real(kind=rk), allocatable :: pointValLeft_grad(:,:,:)
    real(kind=rk), allocatable :: pointValRight_grad(:,:,:)
    ! Nodal representation of the numerical flux
    real(kind=rk), allocatable :: nodalNumFlux(:,:)
    ! Modal representation of the flux on the face
    real(kind=rk), allocatable :: numFluxBuffer(:,:)
    ! Loop over variables (due to single variable FPTs)
    ! integer :: d, e, f, g, h, i, j, m
    integer :: nQuadPoints, overSamp_dofs, nFaceDofs
    integer :: iVP, nPVars, nScalars, iVar
    real(kind=rk) :: penaltyIP
    ! -------------------------------------------------------------------- !

    penaltyIP = alt_get_penaltyIP(                     &
      & maxPolyDegree = scheme%maxPolyDegree,          &
      & elemLength    = elemLength,                    &
      & ip_param      = equation%navierstokes%ip_param )

    ! get correct amount of quadrature points and degree due to projection
    ! method. oversamp_dof is used for the oversampling loop
    nquadpoints = poly_proj%body_2D%nquadpoints
    ! ndofs         = poly_proj%body_2D%ndofs
    oversamp_dofs = poly_proj%body_2D%oversamp_dofs
    nScalars = equation%varSys%nScalars
    nFaceDofs = size(facedata%faceFlux(faceDir)%dat,2)
    nPVars = nFaceDofs * nScalars

    allocate(  leftModalCoeff(oversamp_dofs, nScalars), &
      &       rightModalCoeff(oversamp_dofs, nScalars), &
      &  leftModalCoeff_gradX(oversamp_dofs, nScalars), &
      & rightModalCoeff_gradX(oversamp_dofs, nScalars), &
      &  leftModalCoeff_gradY(oversamp_dofs, nScalars), &
      & rightModalCoeff_gradY(oversamp_dofs, nScalars), &
      &  leftModalCoeff_gradZ(oversamp_dofs, nScalars), &
      & rightModalCoeff_gradZ(oversamp_dofs, nScalars)  )

    allocate(numFluxBuffer(oversamp_dofs, nScalars))
    allocate(nodalNumFlux(nQuadPoints, nScalars))

    allocate( pointValLeft(nQuadPoints,  nScalars),  &
      & pointValRight(nQuadPoints, nScalars),        &
      & pointValLeft_grad( nQuadPoints, nScalars,3), &
      & pointValRight_grad(nQuadPoints, nScalars,3)  )

    ! The permutation indices we apply to enable numerical flux calculation
    ! across faces in x direction.
    varRotation(:) = equation%varRotation(faceDir)          &
      &                      %varTransformIndices(1:nScalars)
    gradRot = equation%varRotation(faceDir)%derTransformIndices(2:4) &
      &       - equation%varRotation(faceDir)%derTransformIndices(1)


    ! Loop over all fluid the faces in x direction
    FaceLoop: do iside = 1, size(faces%leftPos)
      ! Get the fluid neighbors for this face.

      left_neighbor = faces%leftPos(iside)
      right_neighbor = faces%rightPos(iside)

      ! for the left element, we have to access the right face values
      ! and for the right, we have to acess the left face values.
      ! --> modal space
      ! Modal coeffs of the state (left and right)
      call ply_convert2oversample(                             &
        & state       = facedata%faceRep(faceDir)              &
        &               %dat(left_neighbor, :, 1:nScalars, 2), &
        & poly_proj   = poly_proj,                             &
        & nDim        = 2,                                     &
        & modalCoeffs = leftModalCoeff,                        &
        & nScalars    = nScalars                               )
      call ply_convert2oversample(                              &
        & state       = facedata%faceRep(faceDir)               &
        &               %dat(right_neighbor, :, 1:nScalars, 1), &
        & poly_proj   = poly_proj,                              &
        & nDim        = 2,                                      &
        & modalCoeffs = rightModalCoeff,                        &
        & nScalars    = nScalars                                )

      ! Modal coeffs of the gradient in x direction (left and right)
      call ply_convert2oversample(                                        &
        & state       = facedata%faceRep(faceDir)                         &
        &               %dat(left_neighbor, :, nScalars+1:2*nScalars, 2), &
        & poly_proj   = poly_proj,                                        &
        & nDim        = 2,                                                &
        & modalCoeffs = leftModalCoeff_gradX,                             &
        & nScalars    = nScalars                                          )
      call ply_convert2oversample(                                         &
        & state       = facedata%faceRep(faceDir)                          &
        &               %dat(right_neighbor, :, nScalars+1:2*nScalars, 1), &
        & poly_proj   = poly_proj,                                         &
        & nDim        = 2,                                                 &
        & modalCoeffs = rightModalCoeff_gradX,                             &
        & nScalars    = nScalars                                           )

      ! Modal coeffs of the gradient in y direction (left and right)
      call ply_convert2oversample(                                          &
        & state       = facedata%faceRep(faceDir)                           &
        &               %dat(left_neighbor, :, 2*nScalars+1:3*nScalars, 2), &
        & poly_proj   = poly_proj,                                          &
        & nDim        = 2,                                                  &
        & modalCoeffs = leftModalCoeff_gradY,                               &
        & nScalars    = nScalars                                            )
      call ply_convert2oversample(                                           &
        & state       = facedata%faceRep(faceDir)                            &
        &               %dat(right_neighbor, :, 2*nScalars+1:3*nScalars, 1), &
        & poly_proj   = poly_proj,                                           &
        & nDim        = 2,                                                   &
        & modalCoeffs = rightModalCoeff_gradY,                               &
        & nScalars    = nScalars                                             )

      ! Modal coeffs of the gradient in z direction (left and right)
      call ply_convert2oversample(                                          &
        & state       = facedata%faceRep(faceDir)                           &
        &               %dat(left_neighbor, :, 3*nScalars+1:4*nScalars, 2), &
        & poly_proj   = poly_proj,                                          &
        & nDim        = 2,                                                  &
        & modalCoeffs = leftModalCoeff_gradZ,                               &
        & nScalars    = nScalars                                            )
      call ply_convert2oversample(                                           &
        & state       = facedata%faceRep(faceDir)                            &
        &               %dat(right_neighbor, :, 3*nScalars+1:4*nScalars, 1), &
        & poly_proj   = poly_proj,                                           &
        & nDim        = 2,                                                   &
        & modalCoeffs = rightModalCoeff_gradZ,                               &
        & nScalars    = nScalars                                             )
      ! --> oversamp modal space

      ! transform the 2D modal representation to nodal surface points
      ! State left
      call ply_poly_project_m2n( me         = poly_proj,     &
        &                        dim        = 2,             &
        &                        nVars      = nScalars,      &
        &                        nodal_data = pointValLeft,  &
        &                        modal_data = leftModalCoeff )
      ! Gradient of state - left
      call ply_poly_project_m2n( me         = poly_proj,                &
        &                        dim        = 2,                        &
        &                        nVars      = nScalars,                 &
        &                        nodal_data = pointValLeft_grad(:,:,1), &
        &                        modal_data = leftModalCoeff_gradX      )
      call ply_poly_project_m2n( me         = poly_proj,                &
        &                        dim        = 2,                        &
        &                        nVars      = nScalars,                 &
        &                        nodal_data = pointValLeft_grad(:,:,2), &
        &                        modal_data = leftModalCoeff_gradY      )
      call ply_poly_project_m2n( me         = poly_proj,                &
        &                        dim        = 2,                        &
        &                        nVars      = nScalars,                 &
        &                        nodal_data = pointValLeft_grad(:,:,3), &
        &                        modal_data = leftModalCoeff_gradZ      )

      ! State right
      call ply_poly_project_m2n( me         = poly_proj,      &
        &                        dim        = 2,              &
        &                        nVars      = nScalars,       &
        &                        nodal_data = pointValRight,  &
        &                        modal_data = rightModalCoeff )
      ! Gradient of state - right
      call ply_poly_project_m2n( me         = poly_proj,                 &
        &                        dim        = 2,                         &
        &                        nVars      = nScalars,                  &
        &                        nodal_data = pointValRight_grad(:,:,1), &
        &                        modal_data = rightModalCoeff_gradX      )
      call ply_poly_project_m2n( me         = poly_proj,                 &
        &                        dim        = 2,                         &
        &                        nVars      = nScalars,                  &
        &                        nodal_data = pointValRight_grad(:,:,2), &
        &                        modal_data = rightModalCoeff_gradY      )
      call ply_poly_project_m2n( me         = poly_proj,                 &
        &                        dim        = 2,                         &
        &                        nVars      = nScalars,                  &
        &                        nodal_data = pointValRight_grad(:,:,3), &
        &                        modal_data = rightModalCoeff_gradZ      )

      ! --> oversamp nodal space`

      ! for each of the surface points calculate the numerical flux
      do iPoint = 1, nQuadPoints

        call atl_viscNavierStokes(                                       &
          & left       = pointValLeft(iPoint,varRotation),               &
          & left_grad  = pointValLeft_grad(iPoint,varRotation,gradRot),  &
          & right      = pointValRight(iPoint,varRotation),              &
          & right_grad = pointValRight_grad(iPoint,varRotation,gradRot), &
          & mu         = equation%NavierStokes%mu,                       &
          & lambda     = equation%NavierStokes%lambda,                   &
          & thermCond  = equation%NavierStokes%therm_cond,               &
          & heatCap    = equation%euler%cv,                              &
          & penaltyIP  = penaltyIP,                                      &
          & flux       = flux                                            )

        nodalNumFlux(iPoint,varRotation) = flux

      end do !iVP = 1,nPVars

      ! transform back to modal space (facial polynomial)
      call ply_poly_project_n2m( me         = poly_proj,    &
        &                        dim        = 2,            &
        &                        nVars      = nScalars,     &
        &                        nodal_data = nodalNumFlux, &
        &                        modal_data = numFluxBuffer )
      ! --> oversamp modal space

      ! Store the modal coefficients of the numerical flux. For the left
      ! element we have calculated the flux on the right face and vice versa.
      do iVP = 1,nPVars
        iVar = (iVP-1)/nFaceDofs + 1
        iPoint = iVP - (iVar-1)*nFaceDofs

        facedata%faceFlux(faceDir)%dat(left_neighbor, iPoint, iVar, 2)       &
          & = facedata%faceFlux(faceDir)%dat(left_neighbor, iPoint, iVar, 2) &
          &   - numFluxBuffer(iPoint, iVar)
        facedata%faceFlux(faceDir)%dat(right_neighbor, iPoint, iVar, 1)       &
          & = facedata%faceFlux(faceDir)%dat(right_neighbor, iPoint, iVar, 1) &
          &   - numFluxBuffer(iPoint, iVar)
      end do
      ! --> modal space

    end do FaceLoop

  end subroutine atl_modg_viscNavierStokes_oneDim_numFlux
  ! ************************************************************************ !


  ! ************************************************************************ !
  !> Return the penalty parameter for the IP discretizations of higher order
  !! equations.
  function alt_get_penaltyIP(maxPolyDegree, elemLength, ip_param) &
      & result(penaltyIP)
    ! -------------------------------------------------------------------- !
    !> The maximal polynomial degree of the discretization (starting from 0 for
    !! FVM)
    integer, intent(in) :: maxPolyDegree
    !> The length of an element.
    real(kind=rk), intent(in) :: elemLength
    !> The Interior Penalty paramter (should be large enough to ensure
    !! stability)
    real(kind=rk), intent(in) :: ip_param
    !> The resulting penalty parameter
    real(kind=rk) :: penaltyIP
    ! -------------------------------------------------------------------- !

    penaltyIP = ip_param                                         &
      &           * real((maxPolyDegree+1)*(maxPolyDegree+3),rk) &
      &           / 3.0_rk                                       &
      &           / elemLength

  end function alt_get_penaltyIP
  ! ************************************************************************ !


  ! ************************************************************************ !
  !> Numerical flux calculation for stab-viscous part of the Navier-Stokes
  !! equation across the faces in a single spatial direction.
  subroutine atl_modg_stabViscNavierStokes_oneDim_numFlux( equation, facedata, &
    & scheme, faces, faceDir, poly_proj                                        )
    ! -------------------------------------------------------------------- !
    !> The equation you solve.
    type(atl_equations_type), intent(in) :: equation
    !> The face state if the equation
    type(atl_facedata_type), intent(inout) :: facedata
    !> Parameters of the modal dg scheme
    type(atl_modg_scheme_type), intent(in) :: scheme
    !> The faces to calculate the fluxes for.
    type(tem_faceIterator_type), intent(in) :: faces
    !> The spatial direction of the faces you calc the fluxes for, use the
    !! following:
    !! 1 --> x direction. \n
    !! 2 --> y direction. \n
    !! 3 --> z direction.
    integer, intent(in) :: faceDir
    !> Parameter for used projection
    type(ply_poly_project_type), intent(inout) :: poly_proj
    ! -------------------------------------------------------------------- !
    ! Loop vars
    integer :: iDof
    ! Loop var for the faces.
    integer :: iside
    ! Element positions of the left and right element of the face.
    integer :: left_neighbor, right_neighbor
    real(kind=rk) :: numFluxBuffer
    integer :: iVar
    integer :: ndofs
    integer :: iVP, nPVars, nScalars
    integer :: nSides
    ! -------------------------------------------------------------------- !

    ! get correct amount of quadrature points and degree due to projection
    ! method. oversamp_dof and oversamp_degree is used for the oversampling
    ! loop
    ndofs = poly_proj%body_2D%ndofs

    nSides = size(faces%leftPos)
    nScalars = equation%varSys%nScalars
    nPVars = nDofs*equation%varSys%nScalars


    ! Loop over all fluid the faces in x direction
    FaceLoop: do iside = 1, nSides
      ! Get the fluid neighbors for this face.
      left_neighbor = faces%leftPos(iside)
      right_neighbor = faces%rightPos(iside)

      ! Calc the flux in modal way
      do iVP = 1,nPVars
        iVar = (iVP-1) / nDofs + 1
        iDof = iVP - (iVar-1)*nDofs

        numFluxBuffer                                                         &
          & = ( facedata%faceRep(faceDir)%dat(left_neighbor,iDof,iVar,2)      &
          &     + facedata%faceRep(faceDir)%dat(right_neighbor,iDof,iVar,1) ) &
          &   / 2.0_rk

        ! Store the modal coefficients of the numerical flux. For the left
        ! element we have calculated the flux on the right face and vice versa.
        facedata%faceFlux(faceDir)%dat(left_neighbor,iDof,iVar+nScalars,2) &
          & = numFluxBuffer
        facedata%faceFlux(faceDir)%dat(right_neighbor,iDof,iVar+nScalars,1) &
          & = numFluxBuffer
      end do

    end do FaceLoop


  end subroutine atl_modg_stabViscNavierStokes_oneDim_numFlux
  ! ************************************************************************ !

end module atl_modg_navierstokes_kernel_module