sandbox/ecipriano/src/navier-stokes/low-mach.h

    Incompressible Navier–Stokes solver with divergence source term (centered formulation)

    This extension of the centered.h Navier–Stokes equations solver considers situations where the velocity divergence is not null. Examples are phase change simulations, or Low Mach systems with variable physical properties.

    The modified set of equations reads:

    \partial_t\mathbf{u}+\nabla\cdot(\mathbf{u}\otimes\mathbf{u}) = \frac{1}{\rho}\left[-\nabla p + \nabla\cdot(2\mu\mathbf{D})\right] + \mathbf{a} \nabla\cdot\mathbf{u} = {\color{blue} \dot{m} \left(\dfrac{1}{\rho_g} - \dfrac{1}{\rho_l}\right)\delta_\Gamma -\color{blue} \dfrac{1}{\rho}\dfrac{D\rho}{Dt}} with the deformation tensor \mathbf{D}=[\nabla\mathbf{u} + (\nabla\mathbf{u})^T]/2.

    Field Allocations

    We define scalar fields with two possible sources of divergence: stefanflow is the phase change source term, localized at the gas-liquid interface, while drhodt considers density changes.

    #define LOW_MACH 1
    
    scalar drhodt[], * drhodtlist = NULL;
    scalar intexp[], * intexplist = NULL;
    
    bool no_advection_div = false;

    Projection Function

    We define the function that performs the projection step with the volume expansion term due to the phase change or due to density changes.

    #include "poisson.h"
    
    trace
    mgstats project_lowmach (face vector uf, scalar p,
         (const) face vector alpha = unityf,
         double dt = 1.,
         int nrelax = 4)
    {

    We allocate a local scalar field and compute the divergence of \mathbf{u}_f. The divergence is scaled by dt so that the pressure has the correct dimension.

      scalar div[];
      foreach() {
        div[] = 0.;
        foreach_dimension()
          div[] += uf.x[1] - uf.x[];
        div[] /= dt*Delta;
      }

    We add the volume expansion contribution.

      extern int nv;
      static int inv = 0;
      scalar intexp = intexplist[inv];
      scalar drhodt = drhodtlist[inv];
      foreach() {
        div[] += intexp[]/dt;
        div[] += drhodt[]/dt;
      }
      inv++;
      inv = (inv == nv) ? 0 : inv;

    We solve the Poisson problem. The tolerance (set with TOLERANCE) is the maximum relative change in volume of a cell (due to the divergence of the flow) during one timestep i.e. the non-dimensional quantity |\nabla\cdot\mathbf{u}_f|\Delta t Given the scaling of the divergence above, this gives

      mgstats mgp = poisson (p, div, alpha,
           tolerance = TOLERANCE/sq(dt), nrelax = nrelax);

    And compute \mathbf{u}_f^{n+1} using \mathbf{u}_f and p.

      foreach_face()
        uf.x[] -= dt*alpha.x[]*face_gradient_x (p, 0);
    
      return mgp;
    }
    
    #include "utils.h"
    #include "bcg.h"
    
    void advection_div (scalar * tracers, face vector u, double dt,
    		scalar * src = NULL)
    {

    If src is not provided we set all the source terms to zero.

      scalar * psrc = src;
      if (!src)
        for (scalar s in tracers) {
          const scalar zero[] = 0.;
          src = list_append (src, zero);
        }
      assert (list_len (tracers) == list_len (src));
    
      scalar f, source;
      for (f,source in tracers,src) {
        face vector flux[];
        tracer_fluxes (f, u, flux, dt, source);
    #if !EMBED
        foreach() {
          double fold = f[];
          NOT_UNUSED (fold);
          foreach_dimension() {
            if (no_advection_div)
              f[] += dt*(flux.x[] - flux.x[1] + fold*(u.x[1] - u.x[]))/(Delta*cm[]);
            else
              f[] += dt*(flux.x[] - flux.x[1])/(Delta*cm[]);
          }
        }
    #else // EMBED
        update_tracer (f, u, flux, dt);
    #endif // EMBED
      }
    
      if (!psrc)
        free (src);
    }

    We overwrite the function project in centered.h in order to call project_lowmach instead, accounting for the divergcence source terms.

    #if VELOCITY_JUMP
    #define advection(...) advection_div(__VA_ARGS__)
    #include "navier-stokes/centered-new.h"
    #undef advection
    #else
    #define project(...) project_lowmach(__VA_ARGS__)
    #define advection(...) advection_div(__VA_ARGS__)
    #include "navier-stokes/centered-new.h"
    #undef advection
    #undef project
    #endif

    We set the default divergence source term to zero (for the liquid phase)

    event defaults (i = 0) {
      drhodtlist = list_add (drhodtlist, drhodt);
      intexplist = list_add (intexplist, intexp);
    
      for (int i = 1; i < nv; i++) {
        // Density changes
        {
          scalar drhodt = new scalar;
          char name[80];
          sprintf (name, "drhodt%d", i);
          free (drhodt.name);
          drhodt.name = strdup (name);
          drhodtlist = list_add (drhodtlist, drhodt);
        }
    
        // Stefan flow
        {
          scalar intexp = new scalar;
          char name[80];
          sprintf (name, "intexp%d", i);
          free (intexp.name);
          intexp.name = strdup (name);
          intexplist = list_add (intexplist, intexp);
        }
      }
    }
    
    event cleanup (t = end) {
      for (int i = 1; i < nv; i++) {
        scalar drhodt = drhodtlist[i];
        scalar intexp = intexplist[i];
    
        delete ({drhodt,intexp});
      }
      free (drhodtlist), drhodtlist = NULL;
      free (intexplist), intexplist = NULL;
    }