FluidX3D Documentation - How to get started?

0. Install GPU Drivers and OpenCL Runtime

(click to expand section)
  • Windows

    GPUs
    • Download and install the AMD/Intel/Nvidia GPU Drivers, which contain the OpenCL Runtime.
    • Reboot.
    CPUs
  • Linux

    AMD GPUs
    • Download and install AMD GPU Drivers, which contain the OpenCL Runtime, with:
      1
      2
      3
      4
      5
      6
      7
      8
      9
      sudo apt update && sudo apt upgrade -y
      sudo apt install -y g++ git make ocl-icd-libopencl1 ocl-icd-opencl-dev
      mkdir -p ~/amdgpu
      wget -P ~/amdgpu https://repo.radeon.com/amdgpu-install/6.1.3/ubuntu/jammy/amdgpu-install_6.1.60103-1_all.deb
      sudo apt install -y ~/amdgpu/amdgpu-install*.deb
      sudo amdgpu-install -y --usecase=graphics,rocm,opencl --opencl=rocr
      sudo usermod -a -G render,video $(whoami)
      rm -r ~/amdgpu
      sudo shutdown -r now
    Intel GPUs
    • Intel GPU Drivers come already installed since Linux Kernel 6.2, but they don’t contain the OpenCL Runtime.
    • The the OpenCL Runtime has to be installed separately with:
      1
      2
      3
      4
      sudo apt update && sudo apt upgrade -y
      sudo apt install -y g++ git make ocl-icd-libopencl1 ocl-icd-opencl-dev intel-opencl-icd
      sudo usermod -a -G render $(whoami)
      sudo shutdown -r now
    Nvidia GPUs
    • Download and install Nvidia GPU Drivers, which contain the OpenCL Runtime, with:
      1
      2
      3
      sudo apt update && sudo apt upgrade -y
      sudo apt install -y g++ git make ocl-icd-libopencl1 ocl-icd-opencl-dev nvidia-driver-550
      sudo shutdown -r now
    CPUs
    • Option 1: Download and install the oneAPI DPC++ Compiler and oneTBB with:
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      export OCLV="2024.18.6.0.02_rel"
      export TBBV="2021.13.0"
      sudo apt update && sudo apt upgrade -y
      sudo apt install -y g++ git make ocl-icd-libopencl1 ocl-icd-opencl-dev
      sudo mkdir -p ~/cpurt /opt/intel/oclcpuexp_${OCLV} /etc/OpenCL/vendors /etc/ld.so.conf.d
      sudo wget -P ~/cpurt https://github.com/intel/llvm/releases/download/2024-WW25/oclcpuexp-${OCLV}.tar.gz
      sudo wget -P ~/cpurt https://github.com/oneapi-src/oneTBB/releases/download/v${TBBV}/oneapi-tbb-${TBBV}-lin.tgz
      sudo tar -zxvf ~/cpurt/oclcpuexp-${OCLV}.tar.gz -C /opt/intel/oclcpuexp_${OCLV}
      sudo tar -zxvf ~/cpurt/oneapi-tbb-${TBBV}-lin.tgz -C /opt/intel
      echo /opt/intel/oclcpuexp_${OCLV}/x64/libintelocl.so | sudo tee /etc/OpenCL/vendors/intel_expcpu.icd
      echo /opt/intel/oclcpuexp_${OCLV}/x64 | sudo tee /etc/ld.so.conf.d/libintelopenclexp.conf
      sudo ln -sf /opt/intel/oneapi-tbb-${TBBV}/lib/intel64/gcc4.8/libtbb.so /opt/intel/oclcpuexp_${OCLV}/x64
      sudo ln -sf /opt/intel/oneapi-tbb-${TBBV}/lib/intel64/gcc4.8/libtbbmalloc.so /opt/intel/oclcpuexp_${OCLV}/x64
      sudo ln -sf /opt/intel/oneapi-tbb-${TBBV}/lib/intel64/gcc4.8/libtbb.so.12 /opt/intel/oclcpuexp_${OCLV}/x64
      sudo ln -sf /opt/intel/oneapi-tbb-${TBBV}/lib/intel64/gcc4.8/libtbbmalloc.so.2 /opt/intel/oclcpuexp_${OCLV}/x64
      sudo ldconfig -f /etc/ld.so.conf.d/libintelopenclexp.conf
      sudo rm -r ~/cpurt
    • Option 2: Download and install PoCL with:
      1
      2
      sudo apt update && sudo apt upgrade -y
      sudo apt install -y g++ git make ocl-icd-libopencl1 ocl-icd-opencl-dev pocl-opencl-icd
  • Android

    ARM GPUs
    • Download the Termux .apk and install it.
    • In the Termux app, run:
      1
      2
      apt update && apt upgrade -y
      apt install -y clang git make

  • 1. Download FluidX3D

    • Download and unzip the source code, or clone with:
      1
      git clone https://github.com/ProjectPhysX/FluidX3D.git && cd FluidX3D
    • To update FluidX3D:
      • Make a backup of your changes.
      • Run:
        1
        2
        3
        git stash
        git pull origin master
        git stash pop

    2. Compiling the Source Code

    • There is no “installation” of FluidX3D. Instead, you have to compile the source code yourself.
    • I have made this as easy as possible and this documentation will guide you through it. Nontheless, some basic programming experience with C++ would be good for the setup scripts.
    • First, compile the code as-is; this is the standard FP32 benchmark test case. By default, the fastest installed GPU will be selected automatically. Compile time is about 5 seconds.

    Windows

    • Download and install Visual Studio Community. In Visual Studio Installer, add:
      • Desktop development with C++
      • MSVC v142
      • Windows 10 SDK
    • Open FluidX3D.sln in Visual Studio Community.
    • Compile and run by clicking the ► Local Windows Debugger button.
    • To select a specific GPU, open Windows CMD in the FluidX3D folder (type cmd in File Explorer in the directory field and press Enter), then run bin\FluidX3D.exe 0 to select device 0. You can also select multiple GPUs with bin\FluidX3D.exe 0 1 3 6 if the setup is configured as multi-GPU.

    Linux / macOS / Android

    • Compile and run with:
      1
      2
      chmod +x make.sh
      ./make.sh
    • Compiling requires g++ with C++17, which is supported since version 8 (check with g++ --version). If you have make installed (check with make --version), compiling will will be faster using multiple CPU cores; otherwise compiling falls back to using a single CPU core.
    • To select a specific GPU, enter ./make.sh 0 to compile+run, or bin/FluidX3D 0 to run on device 0. You can also select multiple GPUs with bin/FluidX3D 0 1 3 6 if the setup is configured as multi-GPU.
    • Operating system (Linux/macOS/Android) and X11 support (required for INTERACTIVE_GRAPHICS) are detected automatically. In case problems arise, you can still manually select target=... in make.sh.
    • On macOS and Android, INTERACTIVE_GRAPHICS mode is not supported, as no X11 is available. You can still use INTERACTIVE_GRAPHICS_ASCII though, or render video to the hard drive with regular GRAPHICS mode.

    3. Go through Sample Setups

    • Now open src/setup.cpp. In here are all the sample setups, each one being a void main_setup() {...} function block written in C++. Uncomment one of them, maybe start top-to-bottom.
    • In the line where the main_setup() function starts, it says “required extensions in defines.hpp:”, followed by a list of extensions in capital letters. Head over to src/defines.hpp and comment out
      1
      //#define BENCHMARK
      with a //. Then, uncomment all of the extensions required for the setup by removing the // in front of the corresponding line.
    • Finally, compile and run the setup with the ► Local Windows Debugger button (Windows) or ./make.sh (Linux/macOS/Android).
    • Once the interactive graphics window opens, press key P to start/pause the simulation, and press H to show the help menu for keyboard controls and visualization settings.
    • Go through some of the sample setups this way, get familiar with their code structure and test the graphics mode.

    4. Keyboard/Mouse Controls for INTERACTIVE_GRAPHICS/_ASCII

    Key Function
    P start/pause the simulation
    H show/hide help menu for keyboard controls and visualization settings
    Esc
    Alt+F4
    quit
    Mouse
    I
    J K L
    rotate camera
    Scrollwheel
    + -
    zoom (centered camera mode) or camera movement speed (free camera mode)
    Mouseclick
    U
    toggle rotation with Mouse and angle snap rotation with I J K L
    F toggle centered/free camera mode
    W
    A S D
    Space C
    move free camera
    Y X adjust camera field of view
    R toggle camera autorotation
    G print current camera position/rotation in console as copy/paste command
    V toggle stereoscopic rendering for VR
    B toggle VR-goggles/3D-TV mode for stereoscopic rendering
    N M adjust eye distance for stereoscopic rendering
    1 flag wireframe / solid surface (and force vectors on solid cells or surface pressure if the extension is used)
    2 velocity field
    3 streamlines
    4 vorticity (velocity-colored Q-criterion isosurface)
    5 rasterized free surface
    6 raytraced free surface
    7 particles
    T toggle slice visualization mode
    Z toggle field visualization mode
    Q E move slice in slice visualization mode

    5. Writing your own Setups

    The LBM Class

    • For initializing the simulation box, use call
      1
      LBM lbm(Nx, Ny, Nz, nu, ...);
      constructor. Nx/Ny/Nz is the grid resolution and nu is the kinematic shear viscosity in LBM units.
    • To use multiple GPUs, use
      1
      LBM lbm(Nx, Ny, Nz, Dx, Dy, Dz, nu, ...);
      with Dx/Dy/Dz indicating how many domains (GPUs) there are in each spatial direction. The product Dx×Dy×Dz is the total number of domains (GPUs).
    • As long as the lbm object is in scope, you can access the memory. As soon as it goes out of scope, all memory associated with the current simulation is freed again.
    • The grid resolution Nx/Ny/Nz ultimately determines the VRAM occupation. Quite often it’s not obvious at which resolution you’ll overshoot the VRAM capacity of the GPU(s). To aid with this, there is the function:
      1
      const uint3 lbm_N = resolution(float3(1.0f, 2.0f, 0.5f), 2000u);
      This takes as inputs the desired aspect ratio of the simulation box and the VRAM occupation in MB, and returns the grid resolution as a uint3 with .x/.y/.z components. You can also directly feed the uint3 into the LBM constructor as resolution:
      1
      LBM lbm(lbm_N, nu, ...);

    Unit Conversion

    • The LBM simulation uses a different unit system from SI units, where density rho=1 and velocity u≈0.001-0.1, because floating-point arithmetic is most accurate close to 1.
    • To ease unit conversion from SI to LBM units and back, there is the units.hpp struct. By calling
      1
      units.set_m_kg_s(lbm_length, lbm_velocity, lbm_density=1, si_length, si_velocity, si_density);
      the base unit conversion factors [m], [kg], [s] are calculated and stored in the units struct. Thereafter, any of the conversion functions from src/units.hpp can be used to go from SI to LBM units and back, such as lbm_nu = units.nu(si_nu) to convert the kinematic viscosity from SI to LBM units.
    • A good beginner example for this is the “aerodynamics of a cow“ setup.

    Initial and Boundary Conditions

    • If not explicitly set, by default all cells have the default values rho=1, u=0, flags=0.
    • The initial/boundary conditions of single grid cells are set in a parallelized loop that iterates over the entire grid:
      1
      2
      3
      const uint Nx=lbm.get_Nx(), Ny=lbm.get_Ny(), Nz=lbm.get_Nz(); parallel_for(lbm.get_N(), [&](ulong n) { uint x=0u, y=0u, z=0u; lbm.coordinates(n, x, y, z);
      // ...
      });
      Within this loop, you can set the density, velocity and flags of each cell individually by assigning values to lbm.rho[n], lbm.u.x[n], lbm.u.y[n], lbm.u.z[n] and lbm.flags[n]. The n here is the linearized 3D grid index, corresponding to an (x|y|z) position via the function lbm.coordinates(n, x, y, z).
    • For example, to set solid walls at the left and right sides of the simulation box, write
      1
      if(y==0u||y==Ny-1u) lbm.flags[n] = TYPE_S;
      within the loop.
    • All box sides where no solid (TYPE_S) or other boundary type are set will remain periodic boundaries.
    • Primitive geometry, such as spheres, ellipsoids, cubes, cuboids, cylinders, codes, pipes, triangles, inclined planes, or toruses can be set with the functions from shapes.hpp. Example to insert a cylinder:
      1
      if(cylinder(x, y, z, lbm.center(), float3(axis_x, axis_z, axis_z), radius) lbm.flags[n] = TYPE_S;
    • The non-moving no-slip mid-grid bounce-back boundaries (TYPE_S) are always available without further extensions. “No-slip bounce-back” refers to the property that the flow velocity directly at the boundary is 0 (no-slip condition). “Mid-grid” refers to the boundary being located exactly in the middle between the boundary cells and adjacent fluid cells.
    • For inflow/outflow boundaries, you need to enable (uncomment) the EQUILIBRIUM_BOUNDARIES extension. Then, for the specific inflow/outflow cells, set the flag lbm.flags[n] = TYPE_E and on the same cell specify either a density lbm.rho[n] unequal to 1 or a velocity lbm.u.x[n]/lbm.u.y[n]/lbm.u.z[n] unequal to 0, or a combination of both. TYPE_E cells enforce the specified density/velocity value and absorb any incoming shockwaves.
    • For moving solid boundaries, you need to enable (uncomment) the MOVING_BOUNDARIES extension. Then, for the specific solid cells, set the flag lbm.flags[n] = TYPE_S and on the same cell specify a velocity lbm.u.x[n]/lbm.u.y[n]/lbm.u.z[n] unequal to 0. TYPE_S cells reflect any incoming shockwaves.
    • If strict mass conservation is required (for example flow through a linear pipe), use periodic boundaries (i.e. don’t set any boundary type on the cells at these simulation box sides), and drive the flow with a volume force (equivalent to a pressure gradient). Therefore you need to enable (uncomment) the VOLUME_FORCE extension, and in the LBM constuctor set the force per volume (fx|fy|fz):
      1
      LBM lbm(Nx, Ny, Nz, nu, fx, fy, fz);
      These force per volume values should not exceed 0.001 in magnitude.

    Running the Simulation

    • Call lbm.run() (without input parameter, it’s infinite time steps) to initialize and execute the setup, or lbm.run(time_steps) to execute only a specific number of time steps.
    • If you have a more complicated simulation loop where you periodically compute time steps and render images for a video or export data, don’t forget to place an lbm.run(0u) before that loop. This copies the initial/boundary conditions from CPU RAM to GPU VRAM and initializes the simulation on the GPU, without computing a time step. Without initialization, there is no data in VRAM yet for rendering.

    Loading .stl Files

    • For more complex geometries, you can load .stl triangle meshes and voxelize them to the Cartesian simulation grid on the GPU(s).
    • Create a FluidX3D/stl/ folder next to the FluidX3D/src/ folder and download the geometry from websites like Thingiverse, or create your own.
    • Only binary .stl files are supported. For conversion from other formats or for splitting composite geometries like helicopter hull and rotors, I recommend Microsoft 3D Builder on Windows or Blender on Windows/Linux.
    • Load and voxelize simple .stl files directly with
      1
      lbm.voxelize_stl(get_exe_path()+"../stl/mesh.stl", center, rotation, size);
      This automatically repositions/rescales the mesh to the specified center. Use lbm.center() for the simulation box center, or add an offset with a +float3(offset_x, offset_y, offset_z). You can generate and multiply together a rotation matrix like this (example: rotation around the z-axis by 180°, then around the x-axis by 90°):
      1
      float3x3 rotation = float3x3(float3(1, 0, 0), radians(90.0f))*float3x3(float3(0, 0, 1), radians(180.0f));
    • To load composite geometries with several parts without automatic mesh repositioning/rescaling, use
      1
      2
      3
      4
      5
      6
      7
      8
      9
      Mesh* mesh_1 = read_stl(const string& path, const float scale=1.0f, const float3x3& rotation=float3x3(1.0f), const float3& offset=float3(0.0f)); // load mesh without automatic repositioning/rescaling
      Mesh* mesh_2 = read_stl(const string& path, const float scale=1.0f, const float3x3& rotation=float3x3(1.0f), const float3& offset=float3(0.0f));
      mesh_1->scale(const float scale); // manually scale meshes
      mesh_2->scale(const float scale);
      mesh_1->translate(const float3& translation); // manually reposition meshes
      mesh_2->translate(const float3& translation);
      lbm.voxelize_mesh_on_device(mesh_1); // voxelize meshes on GPU
      lbm.voxelize_mesh_on_device(mesh_2);

      to load the meshes from the .stl files, manually scale/reposition all parts of the mesh the same time, and finally voxelize them on the GPU.
    • To aid with repositioning the mesh, there is lbm.center() for the center of the simulation box, as well as the min/max bounding-box coordinates of the mesh mesh->pmin/mesh->pmax, each a float3 with (x|y|z) components.
    • Rotating geometries have to be periodically revoxelized, about every 1-10 LBM time steps. In the main simulation loop in the main_setup() function, first rotate the triangle mesh, then revoxelize on GPU, then compute a few LBM time steps:
      1
      2
      3
      4
      5
      6
      7
      8
      const uint lbm_T = 100000u; // number of LBM time steps to simulate
      const uint lbm_dt = 4u; // number of LBM time steps between each mesh revoxelization
      lbm.run(0u); // initialize simulation
      while(lbm.get_t()<lbm_T) { // main simulation loop
      mesh->rotate(float3x3(float3(0, 0, 1), lbm_omega*(float)lbm_dt)); // rotate the triangle mesh
      lbm.voxelize_mesh_on_device(mesh, TYPE_S, center, float3(0.0f), float3(0.0f, 0.0f, lbm_omega)); // revoxelize the rotated triangle mesh, provide the instantaneous angular velocity vector for moving boundaries
      lbm.run(lbm_dt); // run lbm_dt LBM time steps
      }
      Here lbm_omega is the angular velocity in radians per time step, lbm_dt is the number of simulated time steps between revoxelizations, and float3(0.0f, 0.0f, lbm_omega) is the instantaneous angular velocity as a vector along the axis of rotation. The largest displacement of the outermost cells should not exceed 1 cell between revoxelizations; set lbm_omega = lbm_u/lbm_radius accordingly.
    • Have a look at the “Cessna 172“ and “Bell 222“ setups for some examples.

    Video Rendering

    • For video rendering, disable (comment out) INTERACTIVE_GRAPHICS and INTERACTIVE_GRAPHICS_ASCII and enable (uncomment) GRAPHICS in src/defines.hpp.
    • Set the video resolution as GRAPHICS_FRAME_WIDTH/GRAPHICS_FRAME_HEIGHT and the background color as GRAPHICS_BACKGROUND_COLOR. You can also adjust the other GRAPHICS_... options there, such as semi-transparent rendering mode, or adjust the color scale for velocity with GRAPHICS_U_MAX.
    • A basic loop for rendering video in the main_setup() function looks like this:
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      lbm.graphics.visualization_modes = VIS_FLAG_LATTICE|VIS_Q_CRITERION; // set visualization modes, see all available visualization mode macros (VIZ_...) in defines.hpp
      const uint lbm_T = 10000u; // number of LBM time steps to simulate
      lbm.run(0u, lbm_T); // initialize simulation
      while(lbm.get_t()<lbm_T) { // main simulation loop
      if(lbm.graphics.next_frame(lbm_T, 25.0f)) { // render enough frames for 25 seconds of 60fps video
      lbm.graphics.set_camera_free(float3(2.5f*(float)Nx, 0.0f*(float)Ny, 0.0f*(float)Nz), 0.0f, 0.0f, 50.0f); // set camera to position 1
      lbm.graphics.write_frame(get_exe_path()+"export/camera_1/"); // export image from camera position 1
      lbm.graphics.set_camera_centered(-40.0f, 20.0f, 78.0f, 1.25f); // set camera to position 2
      lbm.graphics.write_frame(get_exe_path()+"export/camera_2/"); // export image from camera position 2
      }
      lbm.run(1u, lbm_T); // run 1 LBM time step
      }
    • To find suitable camera placement, run the simulation at low resolution in INTERACTIVE_GRAPHICS mode, rotate/move the camera to the desired position, click the Mouse to disable mouse rotation, and press G to print the current camera settings as a copy-paste command in the console. Alt+Tab to the console and copy the camera placement command by selecting it with the mouse and right-clicking, then paste it into the main_setup() function.
    • To fly the camera along a smooth path through a list of provided keyframe camera placements, use catmull_rom splines:
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      while(lbm.get_t()<=lbm_T) { // main simulation loop
      if(lbm.graphics.next_frame(lbm_T, 30.0f)) {
      const float t = (float)lbm.get_t()/(float)lbm_T;
      vector<float3> camera_positions = {
      float3(-0.282220f*(float)Nx, 0.529221f*(float)Ny, 0.304399f*(float)Nz),
      float3( 0.806921f*(float)Nx, 0.239912f*(float)Ny, 0.436880f*(float)Nz),
      float3( 1.129724f*(float)Nx, -0.130721f*(float)Ny, 0.352759f*(float)Nz),
      float3( 0.595601f*(float)Nx, -0.504690f*(float)Ny, 0.203096f*(float)Nz),
      float3(-0.056776f*(float)Nx, -0.591919f*(float)Ny, -0.416467f*(float)Nz)
      };
      vector<float> camera_rx = {
      116.0f,
      25.4f,
      -10.6f,
      -45.6f,
      -94.6f
      };
      vector<float> camera_ry = {
      26.0f,
      33.3f,
      20.3f,
      25.3f,
      -16.7f
      };
      const float camera_fov = 90.0f;
      lbm.graphics.set_camera_free(catmull_rom(camera_positions, t), catmull_rom(camera_rx, t), catmull_rom(camera_ry, t), camera_fov);
      lbm.graphics.write_frame(get_exe_path()+"export/");
      }
      lbm.run(1u, lbm_T);
      }
    • The visualization mode(s) can be specified as lbm.graphics.visualization_modes with the VIS_... macros. You can also set the lbm.graphics.slice_mode (0=no slice, 1=x, 2=y, 3=z, 4=xz, 5=xyz, 6=yz, 7=xy) and reposition the slices with lbm.graphics.slice_x/lbm.graphics.slice_y/lbm.graphics.slice_z.
    • Exported frames will automatically be assigned the current simulation time step in their name, in the format bin/export/image-123456789.png.
    • To convert the rendered .png images to video, use FFmpeg:
      1
      ffmpeg -framerate 60 -pattern_type glob -i "export/*/image-*.png" -c:v libx264 -pix_fmt yuv420p -b:v 24M "video.mp4"

    Data Export

    • At any point in time, you can export volumetric data as binary .vtk files with:
      1
      2
      3
      4
      5
      6
      7
      lbm.rho.write_device_to_vtk(); // density
      lbm.u.write_device_to_vtk(); // velocity
      lbm.flags.write_device_to_vtk(); // flags
      lbm.F.write_device_to_vtk(); // force, only for FORCE_FIELD extension
      lbm.phi.write_device_to_vtk(); // fill fraction, only for SURFACE extension
      lbm.T.write_device_to_vtk(); // temperature, only for TEMPERATURE extension
      lbm.write_mesh_to_vtk(const Mesh* mesh); // for exporting triangle meshes
    • These functions first pull the data from the GPU(s) into CPU RAM, and then write it to the hard drive.
    • If unit conversion with units.set_m_kg_s(...) was specified, the data in exported .vtk files is automaticlally converted to SI units.
    • Exported files will automatically be assigned the current simulation time step in their name, in the format bin/export/u-123456789.vtk.
    • Be aware that these volumetric files can be gigantic in file size, tens of GigaByte for a single file.
    • You can view/evaluate the .vtk files for example in ParaView.
    • It is recommended to use the C++ functionality in the main_setup() function directly to extract the data of interest and selectively only write that to the hard drive. Therefore, call lbm.u.read_from_device() to copy the data from the GPU(s) to CPU RAM, and then you can access it directly, for example
      1
      const float lbm_velocity_x = lbm.u.x[lbm.index(x, y, z)];
      to get the x-velocity at the position (x|y|z) in LBM units.
    • To convert the velocity from LBM to SI units, use
      1
      const float si_velocity_x = units.si_u(lbm_velocity_x);
      after having done unit conversion with units.set_m_kg_s(...).
    • You can also export the .stl triangle meshes to binary .vtk files with:
      1
      lbm.write_mesh_to_vtk(const Mesh* mesh);

    Lift/Drag Forces

    • Enable (uncomment) the FORCE_FIELD extension. This extension allows computing boundary forces on every solid cell (TYPE_S) individually, as well as placing an individual volume force on every fluid cell (not used here).
    • In the main_setup() function’s main simulation loop, alternatingly call:
      1
      2
      lbm.run(lbm_dt); // run lbm_dt LBM time steps
      lbm.calculate_force_on_boundaries(); // compute boundary forces on GPU on all solid cells (TYPE_S)
      The latter computes the boundary forces on the GPU into the lbm.F field in VRAM.
    • To copy lbm.F from GPU VRAM to CPU RAM, call:
      1
      lbm.F.read_from_device();
      You can then access the boundary forces at each individual cell with:
      1
      float lbm_force_x_n = lbm.F.x[lbm.index(x, y, z)];
    • To sum over all the individual boundary cells that belong to the body, to get the total force on the body, first voxelize the body with
      1
      lbm.voxelize_mesh_on_device(mesh, TYPE_S|TYPE_X);
      with the additional TYPE_X flagging, and then call
      1
      const float3 lbm_force = lbm.calculate_force_on_object(TYPE_S|TYPE_X);
      to sum over all cells marked TYPE_S|TYPE_X that belong to the body. You can also use TYPE_Y for this.
    • Finally, convert from LBM to SI units with
      1
      const float si_force_x = units.si_F(lbm_force.x);
      after having done unit conversion with units.set_m_kg_s(...).
    • See the “Ahmed body” setup for an example. Note that in the highly turbulent regime, computed body forces are too large by up to a factor 2, because even large resolution is not enough to fully capture the turbulent boundary layer. A wall function is needed, I’ll scan literature on it.

    6. Further LBM Extensions

    By now you’re already familiar with the additional boundary types through extensions VOLUME_FORCE, FORCE_FIELD, EQUILIBRIUM_BOUNDARIES, and MOVING_BOUNDARIES. The remaining available model extensions are briefly outlined here:

    SURFACE Extension

    • To simulate free water surfaces, enable (uncomment) the SURFACE extension.
    • All cells then get 3 additional flags: TYPE_F (fluid), TYPE_I (interface), and TYPE_G (gas). Fluid cells are computed with regular LBM. Interface cells account for the extra surface tension forces, if the surface tension coefficient sigma is set greater than 0 in the LBM constructor; the interface is always 1 cell layer thick. Gas cells are not simulated at all and are essentially treated as vacuum.
    • If not set otherwise in the initial conditions, all cells are initialized as TYPE_G by default. As initial conditions, set all cells that should be fluid to
      1
      lbm.flags[n] = TYPE_F;
      The interface layer will be automatically initialized during initialization with lbm.run(0u).
    • Addidionally to the 3 flags, each cell also gets assigned a fill level lbm.phi[n]: 1 for fluid cells (TYPE_F), ]0,1[ for interface cells (TYPE_I), and 0 for gas cells (TYPE_G). You can set this fill level at initialization, additionally to the cell flag. Do not forget to set the cell flag. If lbm.phi[n] is not set manually, it will automatically be initialized such that all fluid cells get phi=1, all interface cells get phi=0.5, and all gas clls get phi=0 assigned.
    • For a simple example, see the “dam break“ setup. A more advanced sample setup for free surfaces is the “raindrop impact“.

    TEMPERATURE Extension

    • With the TEMPERATURE extension, FluidX3D can model thermal convection flows. This extension automatically also enables the VOLUME_FORCE extension.
    • In the LBM constructor, you then need to set the volume force (fx|fy|fz), the thermal diffusion coefficient alpha, and the thermal expansion coefficient beta, all in LBM units:
      1
      LBM lbm(Nx, Ny, Nz, nu, fx, fy, fz, 0.0f, alpha, beta); // the "0.0f" is for the surface tension coefficient sigma which is not used here and has to remain 0
    • With the extension, each grid cell gets an additional temperature lbm.T[n] (in LBM units) assigned. The default temperature in LBM units is 1.
    • To set temperature boundary conditions, use the flag TYPE_T and for the same cells assign a temperature unequal to 1:
      1
      2
      lbm.flags[n] = TYPE_T; // make the cell n a temperature boundary
      lbm.T[n] = 1.2f; // set this temperature boundary hotter than average
    • See the “Rayleigh-Benard convection“ and “thermal convection“ setups for two examples.

    SUBGRID Extension

    • Fluid flow is characterized by the Reynolds number

      Re = x·unu

      with a characteristic length scale x, a characteristic velocity u and the kinematic shear viscosity nu. Larger length scale, larger velocity or smaller viscosity all mean larger Reynolds number.
    • The Reynolds number is a unit-less number. A low value Re < 2300 means laminar flow, a high value Re > 2900 means turbulent flow. In between is a transitional regime.
    • For very large Reynolds number Re > 100000, the LBM solver becomes unstable, as tiny, very fast rotating vortices can be present in the flow field, and too fast velocity and shear rate makes the simulation blow up.
    • To tackle this problem, there is subgrid models that model vortices smaller than single grid cells. This works by increasing the effective vscosity where the shear rate is large and lots of small eddies are assumed to be present. Coincidentally, locations of high shear rate and low viscosity cause instability, so increasing effective viscosity there keeps the simulation stable.
    • The subgrid model in FLuidX3D is the Smagorinsky-Lilly model. You can enable it with the SUBGRID extension.
    • There is no additional performance cost for this extension.

    PARTICLES Extension

    • By default, the LBM is a grid-based simulation, so there are no particles.
    • But the PARTICLES extension allows to add particles to the simulation, either as passive tracers or as 2-way-coupled particles that can do floating/sedimentation.
    • For passive tracers, only enable the PARTICLES extension, and in the LBM constructor simply add the particle count:
      1
      LBM lbm(Nx, Ny, Nz, nu, 50000u); // this will create 50000 particles
    • Then, in initialization, make a loop over all particles (outside of the initialization loop that iterates over all grid cells):
      1
      2
      3
      4
      5
      6
      uint seed = 42u;
      for(ulong n=0ull; n<lbm.particles->length(); n++) {
      lbm.particles->x[n] = random_symmetric(seed, 0.5f*lbm.size().x); // this will palce the particles randomly anywhere in the simulation box
      lbm.particles->y[n] = random_symmetric(seed, 0.5f*lbm.size().y);
      lbm.particles->z[n] = random_symmetric(seed, 0.5f*lbm.size().z);
      }
    • Note that the position (0|0|0) for particles corresponds to the simulation box center.
    • For 2-way-coupled particles, additionally enable the VOLUME_FORCE and FORCE_FIELD extensions, and in the LBM constructor add the particle density (in LBM units) unequal to 1:
      1
      LBM lbm(Nx, Ny, Nz, nu, 50000u, 1.2f); // this will create 50000 particles that are more dense than the fluid and will sink to the bottom

    7. Suitable Parameters and Simulation Instability

    • Sometimes in the velocity field or streamlines visualization, you will see fuzzyness, or something that looks like a rapidly growing white crystal, blowing up from a certain point and filling the entire simulation box. This is instability, i.e. when velocities turn NaN or Inf.
    • Often times, the cause of instability is an unfortunate choice of unsuitable parameters:
      • too high/low density rho (ideally should be very close to 1 at all times)
      • too high velocity u (must never exceed 0.57 anywhere in the box, ideally should be somewhere around 0.075, but can be as small as 0.001)
      • too low kinematic shear viscosity nu (ideally close to 1/6, becomes unstable when it’s very very close to 0 (then enable the SUBGRID extension), and should not exceed 3)
      • too high force per volume (fx|fy|fz) (should not exceed 0.001 in magnitude)
      • too high surface tension coefficient sigma (should not exceed 0.1)
    • The best parametrization for LBM simulations is an art in itself and needs some practice.
    Author

    Zongbao Chen

    Posted on

    2024-10-17

    Updated on

    2024-09-27

    Licensed under

    Comments