Skip to content

KBoychev/fvm_shallow_water

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

19 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Finite volume method for shallow water equations

A 2D unstructured finite volume method (FVM) shallow water solver written in C++. Fluxes can be evaluated with the Lax–Friedrichs or the Roe method. Implicit and explicit time stepping is available. Expicit time stepping can be performed with the Euler, the Runge-Kutta 2nd-order, and the Runge-Kutta 4th-order methods. The jacobian used for the implicit time stepping is numerically evaluated and the linear system of equations is solved with the BiCGSTAB method. The Eigen C++ template library for linear algebra is used along with OpenMP.

Grid

Dam break grid

Water height

Dam break water height

Effect of the flux method; Gray surface is Lax-Friedrichs, red is Roe

Dam break flux method

Effect of the integration method; Gray surface is Euler, red is Runge-Kutta 2nd order, blue is Runge-Kutta 4th order

Dam break integration method

The code solves the following partial differential equations

$\frac{\partial Q}{\partial t} + \frac{\partial F_{x}}{\partial x} + \frac{\partial F_{y}}{\partial y}=0$,

where

$Q=[h,hu,hv]^T$,

$F_{x}=[hu,hu^2+\frac{1}{2}gh^2,huv]^T$,

$F_{y}=[hv,hvu,hv^2+\frac{1}{2}gh^2]^T$.

Integrating the partial differential equations with respect to the finite volume V gives

$\frac{d}{dt}\int_{V}\mathbf{Q}dV+\int_{V}\frac{\partial F_{x}}{\partial x}+\frac{\partial F_{y}}{\partial y}dV=0$,

which can be simplified to

$\frac{d}{dt}\int_{V}\mathbf{Q}dV+\int_{V}\nabla\cdot\mathbf{F}dV=0$.

Application of the divergence theorem gives

$\frac{d}{dt}\int_{V}\mathbf{Q}dV+\int_{S}\mathbf{F}\cdot\mathbf{n}dS=0$.

Further simplification leads to

$\frac{d}{dt}\int_{V}\mathbf{Q}dV=-\mathbf{R}=-\int_{S}\mathbf{F}\cdot\mathbf{n}dS$.

The flux across the cell faces is approximated with the Roe or Lax-Friedrichs Riemann solvers. The conservative variables are assumed constant over the cell which results in a first-order spatial approximation. For higher-order approximation, the gradients at each cell centre must be calculated and used to reconstruct the values at the cell faces. To avoid spurious oscillations, due to discontinuities, a limiter must also be introduced.

$\int_{S}\mathbf{F}\cdot\mathbf{n}dS\approx F_{1}(\mathbf{Q},\mathbf{Q_{1}},\mathbf{n_{1}})S_{1}+F_{2}(\mathbf{Q},\mathbf{Q_{2}},\mathbf{n_{2}})S_{2}+F_{3}(\mathbf{Q},\mathbf{Q_{3}},\mathbf{n_{3}})S_{3}$

The time derivative is approximated with the forward Euler method.

$\frac{d}{dt}\int_{V}\mathbf{Q}dV\approx \frac{\mathbf{Q}^{n+1}-\mathbf{Q}^{n}}{\Delta t}V$

The resulting discretised PDE is then given by:

$\frac{\mathbf{Q}^{n+1}-\mathbf{Q}^{n}}{\Delta t}=-\mathbf{R}(\mathbf{Q})=-\frac{1}{V}\left(F_{1}(\mathbf{Q},\mathbf{Q_{1}},\mathbf{n_{1}})S_{1}+F_{2}(\mathbf{Q},\mathbf{Q_{2}},\mathbf{n_{2}})S_{2}+F_{3}(\mathbf{Q},\mathbf{Q_{3}},\mathbf{n_{3}})S_{3}\right)$.

An explicit scheme is obtained if the residual is evaluated at the current time level, n.

$\mathbf{Q}^{n+1}=\mathbf{Q}^{n}-\Delta t \mathbf{R}(\mathbf{Q}^{n})$

An implicit scheme is obtained if the residual is evaluated at the next time level, n+1. Since the residual at n+1 is unknown, Taylor series can be used to approximate it. The higher-order terms in the Taylor series are neglected. The solution becomes more expensive as one needs to evaluate the derivative of the residual with respect to the vector of conserved variables. The current method uses finite differences to evaluate the Jacobian matrix.

$\left(\frac{1}{\Delta t}\mathbf{I}+\frac{\partial\mathbf{R}}{\partial\mathbf{Q}}\right)\Delta\mathbf{Q}=-\mathbf{R}(\mathbf{Q}^{n})$

Compile

The code requires the Eigen C++ template library. Download the library and extract its contents to a folder named Eigen inside the includes folder (includes/Eigen). Once extracted use make to compile the code. The sw_solver binary will be created in the bin folder.

Run

To run the code navigate to the folder of the case e.g. dam_coarse_lf and run ../bin/sw_solver dam. The program will solve the dam break problem using the Lax-Friedrichs method for 1000 time steps with a time step of 0.003. The full list of options is:

  • dt - timestep
  • N_timesteps - number of timesteps
  • riemann_solver - Riemman solver; 1 - Lax-Friedrichs, 2 - Roe
  • output_frequency - Results output frequency; 0 - no output, i where i > 0 - every ith iteration
  • integrator - Integration method; 0 - Euler, 1 - Runge-Kutta 2nd-order, 2 - Runge-Kutta 4th order
  • implicit - Solve the equation explicitly or implicitly; 0 - Explicit, 1 - Implicit

Note: If the implicit option is set to 1 the integrator option is not used.

About

FVM for shallow water equations

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published