Turbulent LES Flow in a Heated Pebble Bed
In this tutorial, you will learn how to:
Couple NekRS to MOOSE for CHT for Large Eddy Simulation (LES) in a pebble bed
Manipulate the NekRS solution on each time step through custom user-defined kernels
Adjust sidesets in the NekRS mesh prior to usage
To access this tutorial,
This tutorial also requires you to download some mesh files from Box. Please download the files from the pebble_67
folder here and place these files within the same directory structure in tutorials/pebble_67
.
This tutorial requires HPC resources to run.
Geometry and Computational Model
The geometry consists of a small packed bed with 67 spheres each with diameter of cm; helium flows in the space between pebbles. A cut-away through the center of the bed is shown in Figure 1. The pebbles are packed into a cylinder with diameter of . The pebbles begin around upstream from the cylinder inlet, and the packed region has a total height of about .

Figure 1: Cutaway through the domain in a 67-pebble packed bed. Pebbles are shown in red, while the fluid region is shown in gray.
Table 1 summarizes the geometry and operating conditions of this model. The Reynolds number is based on the pebble diameter.
Table 1: Geometric and boundary condition specifications for a 67-pebble bed
Parameter | Value |
---|---|
Pebble diameter, | 6 cm |
Inlet temperature | 523 K |
Outlet pressure | 4 MPa |
Reynolds number | 1460 |
Prandtl number | 0.71 |
Power | 24 kW |
Heat Conduction Model
The MOOSE heat transfer module is used to solve for energy conservation in the solid, with the time derivative neglected in order to more quickly approach steady state. The solid mesh is shown in Figure 2; the outer surface of the pebbles is sideset 0.

Figure 2: Mesh for the solid heat conduction model
This mesh is generated using MOOSE mesh generators by building a mesh for a single sphere and repeating it 67 times, translated to the position of each pebble. The pebble positions are obtained using off-line discrete element modeling (not discussed here). Note that while chamfers are accounted for in the fluid mesh, that the 67 pebbles are not interconnected to one another for simplicity.
(tutorials/pebble_67/moose.i)The interior of the pebbles is homogenized. We will also assume a uniform power distribution in the pebbles. On the pebble surfaces, a Dirichlet temperature is provided by NekRS. We will run the solid model first, so we must specify an initial condition for the wall temperature, which we simply set to the fluid inlet temperature.
NekRS Model
NekRS solves the incompressible Navier-Stokes equations in non-dimensional form. Turbulence is modeled using LES using a high-pass filter.
The NekRS files are:
pb67.re2
: NekRS meshpb67.par
: High-level settings for the solver, boundary condition mappings to sidesets, and the equations to solvepb67.udf
: User-defined C++ functions for on-line postprocessing and model setuppb67.oudf
: User-defined OCCA kernels for boundary conditions, source terms, and other equation manipulationspb67.usr
: Legacy Fortran backend for postprocessing and input modifications
The fluid mesh is shown in Figure 3 using Paraview's "crinkle clip" feature to more clearly delineate individual elements. The pebble locations are shown in red for additional context. This mesh contains 122,284 hexahedral elements. This mesh is generated using a Voronoi cell approach Lan et al. (2021).

Figure 3: Mesh for the NekRS CFD model; GLL points are not shown
Next, the .par
file contains problem setup information. This input solves in non-dimensional form, at a Reynolds number of 1460 and a Peclet number of 1036. Note the LES filter specification in the [GENERAL]
block.
In the .oudf
file, we define boundary conditions. The inlet boundary is set to a temperature of 0 (a dimensional temperature of ), while the fluid-solid interface will receive a heat flux from MOOSE. The inlet velocity condition is set to , with a stabilized outflow for pressure.
We also create two custom kernels which we shall use to manipulate the NekRS solution on each time step:
Clip temperature so that it is always within the range of , where is the non-dimensional temperature. This is useful to limit extreme temperatures in under-resolved regions of the mesh. We define this operation in a kernel, which we name
clipTemperature
.Increase the viscosity and conductivity at the domain outlet (where the mesh is underresolved) in order to maintain a stable solution. The properties are modified to ramp the viscosity and conductivity between heights of and in a kernel which we name
manipulateOutlet
.
Next, the .udf
file is used to setup initial conditions and conduct other initialization aspects in order to JIT-compile our kernels.
Note that our two custom kernels are fundamentally different in character. The clipTemperature
kernel is a postprocessing activity, so we explicitly call that kernel on each time step in the UDF_ExecuteStep
function. Alternatively, the manipulateOutletKernel
is changing the viscosity and conductivity near the outlet, which is modifying the actual governing equations themselves. NekRS contains a udf
object which has multiple pointers to kernels which actually touch the governing equations. For our case, the udf.properties
pointer is the interface for modifying the fluid properties, so we pass a function wrapping our kernel (uservp
) to that variable so that NekRS can call the kernel at the appropriate location during execution. You can find the function signatures for the other interfaces in nekRS/src/udf/udf.hpp
:
You have already seen the udf.sEqnSource
for applying a volumetric heat source in NekRS from MOOSE.
Finally, for this tutorial we are also using the legacy Fortran backend to modify some of our boundary condition sidesets (for demonstration's sake, the mesh may have been generated without sideset numbers, which NekRS needs). We do this manipulation in the pb67.usr
file, in the usrdat2()
subroutine.
The following lists how to interpret these various Fortran entities:
iel
is a loop variable name used to indicate a loop over elementsifc
is a loop variable name used to indicate a loop over the faces on an elementcbc(ifc, iel, 1)
: sideset name for velocitycbc(ifc, iel, 2)
: sideset name for temperaturebc(5, ifc, iel, 1)
: sideset ID for velocityboundaryID(ifc, iel)
: sideset ID for velocityboundaryIDt(ifc, iel)
: sideset ID for temperature
In other words, this function is essentially (i) changing the names associated with sidesets and then (ii) using those new names to define IDs for the sidesets.
CHT Coupling
In this section, MOOSE and NekRS are coupled for CHT.
Solid Input Files
The solid phase is solved with the MOOSE heat transfer module, and is described in the moose.i
input. First we set up the mesh using a SphereMeshGenerator for each pebble and then repeating it 67 times at each pebble location.
Next, we define a temperature variable temp
, and specify the governing equations and boundary conditions we will apply.
The MOOSE heat transfer module will receive a wall temperature from NekRS in the form of an AuxVariable, so we define a receiver variable for the temperature, as nek_temp
. The MOOSE heat transfer module will also send heat flux to NekRS, which we compute as another AuxVariable named flux
, which we compute with a DiffusionFluxAux auxiliary kernel.
We define a number of postprocessors for querying the solution as well as for normalizing the heat flux.
(tutorials/pebble_67/moose.i)Finally, we add a TransientMultiApp that will run a MOOSE-wrapped NekRS simulation. Then, we add three different transfers to/from NekRS:
MultiAppNearestNodeTransfer to send the heat flux from MOOSE to NekRS
MultiAppNearestNodeTransfer to send temperature from NekRS to MOOSE
MultiAppPostprocessorTransfer to normalize the heat flux sent to NekRS
Fluid Input Files
The Nek wrapping is described in the nek.i
input file. We first define a few file-local variables, and then build a mesh mirror with a NekRSMesh. By setting boundary = '4'
, we indicate that boundary 4 will be coupled via CHT.
Next, we define additional parameters to describe how NekRS interacts with MOOSE with the NekRSProblem. The NekRS input files are in nondimensional form, so we must indicate all the characteristic scales so that data transfers with a dimensional MOOSE application are performed correctly.
(tutorials/pebble_67/nek.i)For time stepping, we will allow NekRS to control its own time stepping by using the NekTimeStepper.
(tutorials/pebble_67/nek.i)Then, we define a few postprocessors to use for querying the solution.
(tutorials/pebble_67/nek.i)Execution and Postprocessing
To run the coupled NekRS-MOOSE calculation,
This will run with 500 MPI processes.
When the simulation has finished, you will have created a number of different output files:
moose_out.e
, an Exodus file with the MOOSE solution for the NekRS-MOOSE calculationmoose_out_nek0.e
, an Exodus file with the NekRS solution that was ultimately transferred in/out of MOOSEpb670.f<n>
, a (custom format) NekRS output file with the NekRS solution; to convert to a format viewable in Paraview, consult the NekRS documentation
Figure 4 shows the instantaneous velocity (left) and temperature (right) predicted by NekRS, in non-dimensional form. This solution is visualized from the NekRS native output files.

Figure 4: Instantaneous velocity (left) and temperature (right) predicted by NekRS, in non-dimensional form.
References
- Y. Lan, P. Fischer, E. Merzari, and M. Min.
All-hex meshing strategies for densely packed spheres.
arXiv, 2021.
URL: https://arxiv.org/abs/2106.00196.[BibTeX]