xref: /petsc/src/ts/tutorials/multirate/ex6.c (revision 2a1887a77e7b2c6e00dd0ba96d1387c839460237)
1c4762a1bSJed Brown /*
2c4762a1bSJed Brown   Note:
36aad120cSJose E. Roman     -hratio is the ratio between mesh size of coarse grids and fine grids
4c4762a1bSJed Brown     -ts_rk_dtratio is the ratio between the large stepsize and the small stepsize
5c4762a1bSJed Brown */
6c4762a1bSJed Brown 
7c4762a1bSJed Brown static const char help[] = "1D periodic Finite Volume solver in slope-limiter form with semidiscrete time stepping.\n"
8c4762a1bSJed Brown                            "  advection   - Constant coefficient scalar advection\n"
9c4762a1bSJed Brown                            "                u_t       + (a*u)_x               = 0\n"
10c4762a1bSJed Brown                            "  for this toy problem, we choose different meshsizes for different sub-domains (slow-fast-slow), say\n"
11c4762a1bSJed Brown                            "                hxs  = hratio*hxf \n"
12c4762a1bSJed Brown                            "  where hxs and hxf are the grid spacings for coarse and fine grids respectively.\n"
13c4762a1bSJed Brown                            "  exact       - Exact Riemann solver which usually needs to perform a Newton iteration to connect\n"
14c4762a1bSJed Brown                            "                the states across shocks and rarefactions\n"
15c4762a1bSJed Brown                            "  simulation  - use reference solution which is generated by smaller time step size to be true solution,\n"
16c4762a1bSJed Brown                            "                also the reference solution should be generated by user and stored in a binary file.\n"
17c4762a1bSJed Brown                            "  characteristic - Limit the characteristic variables, this is usually preferred (default)\n"
18c4762a1bSJed Brown                            "Several initial conditions can be chosen with -initial N\n\n"
19c4762a1bSJed Brown                            "The problem size should be set with -da_grid_x M\n\n";
20c4762a1bSJed Brown 
21c4762a1bSJed Brown #include <petscts.h>
22c4762a1bSJed Brown #include <petscdm.h>
23c4762a1bSJed Brown #include <petscdmda.h>
24c4762a1bSJed Brown #include <petscdraw.h>
25c4762a1bSJed Brown #include "finitevolume1d.h"
26c4762a1bSJed Brown 
RangeMod(PetscReal a,PetscReal xmin,PetscReal xmax)27d71ae5a4SJacob Faibussowitsch static inline PetscReal RangeMod(PetscReal a, PetscReal xmin, PetscReal xmax)
28d71ae5a4SJacob Faibussowitsch {
299371c9d4SSatish Balay   PetscReal range = xmax - xmin;
309371c9d4SSatish Balay   return xmin + PetscFmodReal(range + PetscFmodReal(a, range), range);
319371c9d4SSatish Balay }
32c4762a1bSJed Brown 
33c4762a1bSJed Brown /* --------------------------------- Advection ----------------------------------- */
34c4762a1bSJed Brown typedef struct {
35c4762a1bSJed Brown   PetscReal a; /* advective velocity */
36c4762a1bSJed Brown } AdvectCtx;
37c4762a1bSJed Brown 
PhysicsRiemann_Advect(void * vctx,PetscInt m,const PetscScalar * uL,const PetscScalar * uR,PetscScalar * flux,PetscReal * maxspeed)38d71ae5a4SJacob Faibussowitsch static PetscErrorCode PhysicsRiemann_Advect(void *vctx, PetscInt m, const PetscScalar *uL, const PetscScalar *uR, PetscScalar *flux, PetscReal *maxspeed)
39d71ae5a4SJacob Faibussowitsch {
40c4762a1bSJed Brown   AdvectCtx *ctx = (AdvectCtx *)vctx;
41c4762a1bSJed Brown   PetscReal  speed;
42c4762a1bSJed Brown 
43c4762a1bSJed Brown   PetscFunctionBeginUser;
44c4762a1bSJed Brown   speed     = ctx->a;
45c4762a1bSJed Brown   flux[0]   = PetscMax(0, speed) * uL[0] + PetscMin(0, speed) * uR[0];
46c4762a1bSJed Brown   *maxspeed = speed;
473ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
48c4762a1bSJed Brown }
49c4762a1bSJed Brown 
PhysicsCharacteristic_Advect(void * vctx,PetscInt m,const PetscScalar * u,PetscScalar * X,PetscScalar * Xi,PetscReal * speeds)50d71ae5a4SJacob Faibussowitsch static PetscErrorCode PhysicsCharacteristic_Advect(void *vctx, PetscInt m, const PetscScalar *u, PetscScalar *X, PetscScalar *Xi, PetscReal *speeds)
51d71ae5a4SJacob Faibussowitsch {
52c4762a1bSJed Brown   AdvectCtx *ctx = (AdvectCtx *)vctx;
53c4762a1bSJed Brown 
54c4762a1bSJed Brown   PetscFunctionBeginUser;
55c4762a1bSJed Brown   X[0]      = 1.;
56c4762a1bSJed Brown   Xi[0]     = 1.;
57c4762a1bSJed Brown   speeds[0] = ctx->a;
583ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
59c4762a1bSJed Brown }
60c4762a1bSJed Brown 
PhysicsSample_Advect(void * vctx,PetscInt initial,FVBCType bctype,PetscReal xmin,PetscReal xmax,PetscReal t,PetscReal x,PetscReal * u)61d71ae5a4SJacob Faibussowitsch static PetscErrorCode PhysicsSample_Advect(void *vctx, PetscInt initial, FVBCType bctype, PetscReal xmin, PetscReal xmax, PetscReal t, PetscReal x, PetscReal *u)
62d71ae5a4SJacob Faibussowitsch {
63c4762a1bSJed Brown   AdvectCtx *ctx = (AdvectCtx *)vctx;
64c4762a1bSJed Brown   PetscReal  a   = ctx->a, x0;
65c4762a1bSJed Brown 
66c4762a1bSJed Brown   PetscFunctionBeginUser;
67c4762a1bSJed Brown   switch (bctype) {
68d71ae5a4SJacob Faibussowitsch   case FVBC_OUTFLOW:
69d71ae5a4SJacob Faibussowitsch     x0 = x - a * t;
70d71ae5a4SJacob Faibussowitsch     break;
71d71ae5a4SJacob Faibussowitsch   case FVBC_PERIODIC:
72d71ae5a4SJacob Faibussowitsch     x0 = RangeMod(x - a * t, xmin, xmax);
73d71ae5a4SJacob Faibussowitsch     break;
74d71ae5a4SJacob Faibussowitsch   default:
75d71ae5a4SJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_UNKNOWN_TYPE, "unknown BCType");
76c4762a1bSJed Brown   }
77c4762a1bSJed Brown   switch (initial) {
78d71ae5a4SJacob Faibussowitsch   case 0:
79d71ae5a4SJacob Faibussowitsch     u[0] = (x0 < 0) ? 1 : -1;
80d71ae5a4SJacob Faibussowitsch     break;
81d71ae5a4SJacob Faibussowitsch   case 1:
82d71ae5a4SJacob Faibussowitsch     u[0] = (x0 < 0) ? -1 : 1;
83d71ae5a4SJacob Faibussowitsch     break;
84d71ae5a4SJacob Faibussowitsch   case 2:
85d71ae5a4SJacob Faibussowitsch     u[0] = (0 < x0 && x0 < 1) ? 1 : 0;
86d71ae5a4SJacob Faibussowitsch     break;
87d71ae5a4SJacob Faibussowitsch   case 3:
88d71ae5a4SJacob Faibussowitsch     u[0] = PetscSinReal(2 * PETSC_PI * x0);
89d71ae5a4SJacob Faibussowitsch     break;
90d71ae5a4SJacob Faibussowitsch   case 4:
91d71ae5a4SJacob Faibussowitsch     u[0] = PetscAbs(x0);
92d71ae5a4SJacob Faibussowitsch     break;
93d71ae5a4SJacob Faibussowitsch   case 5:
94d71ae5a4SJacob Faibussowitsch     u[0] = (x0 < 0 || x0 > 0.5) ? 0 : PetscSqr(PetscSinReal(2 * PETSC_PI * x0));
95d71ae5a4SJacob Faibussowitsch     break;
96d71ae5a4SJacob Faibussowitsch   case 6:
97d71ae5a4SJacob Faibussowitsch     u[0] = (x0 < 0) ? 0 : ((x0 < 1) ? x0 : ((x0 < 2) ? 2 - x0 : 0));
98d71ae5a4SJacob Faibussowitsch     break;
99d71ae5a4SJacob Faibussowitsch   case 7:
100d71ae5a4SJacob Faibussowitsch     u[0] = PetscPowReal(PetscSinReal(PETSC_PI * x0), 10.0);
101d71ae5a4SJacob Faibussowitsch     break;
102d71ae5a4SJacob Faibussowitsch   default:
103d71ae5a4SJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_UNKNOWN_TYPE, "unknown initial condition");
104c4762a1bSJed Brown   }
1053ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
106c4762a1bSJed Brown }
107c4762a1bSJed Brown 
PhysicsCreate_Advect(FVCtx * ctx)108d71ae5a4SJacob Faibussowitsch static PetscErrorCode PhysicsCreate_Advect(FVCtx *ctx)
109d71ae5a4SJacob Faibussowitsch {
110c4762a1bSJed Brown   AdvectCtx *user;
111c4762a1bSJed Brown 
112c4762a1bSJed Brown   PetscFunctionBeginUser;
1139566063dSJacob Faibussowitsch   PetscCall(PetscNew(&user));
114c4762a1bSJed Brown   ctx->physics2.sample2         = PhysicsSample_Advect;
115c4762a1bSJed Brown   ctx->physics2.riemann2        = PhysicsRiemann_Advect;
116c4762a1bSJed Brown   ctx->physics2.characteristic2 = PhysicsCharacteristic_Advect;
117c4762a1bSJed Brown   ctx->physics2.destroy         = PhysicsDestroy_SimpleFree;
118c4762a1bSJed Brown   ctx->physics2.user            = user;
119c4762a1bSJed Brown   ctx->physics2.dof             = 1;
1209566063dSJacob Faibussowitsch   PetscCall(PetscStrallocpy("u", &ctx->physics2.fieldname[0]));
121c4762a1bSJed Brown   user->a = 1;
122d0609cedSBarry Smith   PetscOptionsBegin(ctx->comm, ctx->prefix, "Options for advection", "");
123d71ae5a4SJacob Faibussowitsch   {
124d71ae5a4SJacob Faibussowitsch     PetscCall(PetscOptionsReal("-physics_advect_a", "Speed", "", user->a, &user->a, NULL));
125d71ae5a4SJacob Faibussowitsch   }
126d0609cedSBarry Smith   PetscOptionsEnd();
1273ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
128c4762a1bSJed Brown }
129c4762a1bSJed Brown 
FVSample_2WaySplit(FVCtx * ctx,DM da,PetscReal time,Vec U)130d71ae5a4SJacob Faibussowitsch PetscErrorCode FVSample_2WaySplit(FVCtx *ctx, DM da, PetscReal time, Vec U)
131d71ae5a4SJacob Faibussowitsch {
132c4762a1bSJed Brown   PetscScalar   *u, *uj, xj, xi;
133c4762a1bSJed Brown   PetscInt       i, j, k, dof, xs, xm, Mx;
134c4762a1bSJed Brown   const PetscInt N = 200;
135c4762a1bSJed Brown   PetscReal      hs, hf;
136c4762a1bSJed Brown 
137c4762a1bSJed Brown   PetscFunctionBeginUser;
1383c633725SBarry Smith   PetscCheck(ctx->physics2.sample2, PETSC_COMM_SELF, PETSC_ERR_SUP, "Physics has not provided a sampling function");
1399566063dSJacob Faibussowitsch   PetscCall(DMDAGetInfo(da, 0, &Mx, 0, 0, 0, 0, 0, &dof, 0, 0, 0, 0, 0));
1409566063dSJacob Faibussowitsch   PetscCall(DMDAGetCorners(da, &xs, 0, 0, &xm, 0, 0));
1419566063dSJacob Faibussowitsch   PetscCall(DMDAVecGetArray(da, U, &u));
1429566063dSJacob Faibussowitsch   PetscCall(PetscMalloc1(dof, &uj));
143c4762a1bSJed Brown   hs = (ctx->xmax - ctx->xmin) * 3.0 / 8.0 / ctx->sf;
144c4762a1bSJed Brown   hf = (ctx->xmax - ctx->xmin) / 4.0 / (ctx->fs - ctx->sf);
145c4762a1bSJed Brown   for (i = xs; i < xs + xm; i++) {
146c4762a1bSJed Brown     if (i < ctx->sf) {
147c4762a1bSJed Brown       xi = ctx->xmin + 0.5 * hs + i * hs;
148c4762a1bSJed Brown       /* Integrate over cell i using trapezoid rule with N points. */
149c4762a1bSJed Brown       for (k = 0; k < dof; k++) u[i * dof + k] = 0;
150c4762a1bSJed Brown       for (j = 0; j < N + 1; j++) {
151c4762a1bSJed Brown         xj = xi + hs * (j - N / 2) / (PetscReal)N;
1529566063dSJacob Faibussowitsch         PetscCall((*ctx->physics2.sample2)(ctx->physics2.user, ctx->initial, ctx->bctype, ctx->xmin, ctx->xmax, time, xj, uj));
153c4762a1bSJed Brown         for (k = 0; k < dof; k++) u[i * dof + k] += ((j == 0 || j == N) ? 0.5 : 1.0) * uj[k] / N;
154c4762a1bSJed Brown       }
155c4762a1bSJed Brown     } else if (i < ctx->fs) {
156c4762a1bSJed Brown       xi = ctx->xmin + ctx->sf * hs + 0.5 * hf + (i - ctx->sf) * hf;
157c4762a1bSJed Brown       /* Integrate over cell i using trapezoid rule with N points. */
158c4762a1bSJed Brown       for (k = 0; k < dof; k++) u[i * dof + k] = 0;
159c4762a1bSJed Brown       for (j = 0; j < N + 1; j++) {
160c4762a1bSJed Brown         xj = xi + hf * (j - N / 2) / (PetscReal)N;
1619566063dSJacob Faibussowitsch         PetscCall((*ctx->physics2.sample2)(ctx->physics2.user, ctx->initial, ctx->bctype, ctx->xmin, ctx->xmax, time, xj, uj));
162c4762a1bSJed Brown         for (k = 0; k < dof; k++) u[i * dof + k] += ((j == 0 || j == N) ? 0.5 : 1.0) * uj[k] / N;
163c4762a1bSJed Brown       }
164c4762a1bSJed Brown     } else {
165c4762a1bSJed Brown       xi = ctx->xmin + ctx->sf * hs + (ctx->fs - ctx->sf) * hf + 0.5 * hs + (i - ctx->fs) * hs;
166c4762a1bSJed Brown       /* Integrate over cell i using trapezoid rule with N points. */
167c4762a1bSJed Brown       for (k = 0; k < dof; k++) u[i * dof + k] = 0;
168c4762a1bSJed Brown       for (j = 0; j < N + 1; j++) {
169c4762a1bSJed Brown         xj = xi + hs * (j - N / 2) / (PetscReal)N;
1709566063dSJacob Faibussowitsch         PetscCall((*ctx->physics2.sample2)(ctx->physics2.user, ctx->initial, ctx->bctype, ctx->xmin, ctx->xmax, time, xj, uj));
171c4762a1bSJed Brown         for (k = 0; k < dof; k++) u[i * dof + k] += ((j == 0 || j == N) ? 0.5 : 1.0) * uj[k] / N;
172c4762a1bSJed Brown       }
173c4762a1bSJed Brown     }
174c4762a1bSJed Brown   }
1759566063dSJacob Faibussowitsch   PetscCall(DMDAVecRestoreArray(da, U, &u));
1769566063dSJacob Faibussowitsch   PetscCall(PetscFree(uj));
1773ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
178c4762a1bSJed Brown }
179c4762a1bSJed Brown 
SolutionErrorNorms_2WaySplit(FVCtx * ctx,DM da,PetscReal t,Vec X,PetscReal * nrm1)180d71ae5a4SJacob Faibussowitsch static PetscErrorCode SolutionErrorNorms_2WaySplit(FVCtx *ctx, DM da, PetscReal t, Vec X, PetscReal *nrm1)
181d71ae5a4SJacob Faibussowitsch {
182c4762a1bSJed Brown   Vec                Y;
183c4762a1bSJed Brown   PetscInt           i, Mx;
184c4762a1bSJed Brown   const PetscScalar *ptr_X, *ptr_Y;
185c4762a1bSJed Brown   PetscReal          hs, hf;
186c4762a1bSJed Brown 
187c4762a1bSJed Brown   PetscFunctionBeginUser;
1889566063dSJacob Faibussowitsch   PetscCall(VecGetSize(X, &Mx));
1899566063dSJacob Faibussowitsch   PetscCall(VecDuplicate(X, &Y));
1909566063dSJacob Faibussowitsch   PetscCall(FVSample_2WaySplit(ctx, da, t, Y));
191c4762a1bSJed Brown   hs = (ctx->xmax - ctx->xmin) * 3.0 / 8.0 / ctx->sf;
192c4762a1bSJed Brown   hf = (ctx->xmax - ctx->xmin) / 4.0 / (ctx->fs - ctx->sf);
1939566063dSJacob Faibussowitsch   PetscCall(VecGetArrayRead(X, &ptr_X));
1949566063dSJacob Faibussowitsch   PetscCall(VecGetArrayRead(Y, &ptr_Y));
195c4762a1bSJed Brown   for (i = 0; i < Mx; i++) {
196c4762a1bSJed Brown     if (i < ctx->sf || i > ctx->fs - 1) *nrm1 += hs * PetscAbs(ptr_X[i] - ptr_Y[i]);
197c4762a1bSJed Brown     else *nrm1 += hf * PetscAbs(ptr_X[i] - ptr_Y[i]);
198c4762a1bSJed Brown   }
1999566063dSJacob Faibussowitsch   PetscCall(VecRestoreArrayRead(X, &ptr_X));
2009566063dSJacob Faibussowitsch   PetscCall(VecRestoreArrayRead(Y, &ptr_Y));
2019566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&Y));
2023ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
203c4762a1bSJed Brown }
204c4762a1bSJed Brown 
FVRHSFunction_2WaySplit(TS ts,PetscReal time,Vec X,Vec F,void * vctx)205d71ae5a4SJacob Faibussowitsch PetscErrorCode FVRHSFunction_2WaySplit(TS ts, PetscReal time, Vec X, Vec F, void *vctx)
206d71ae5a4SJacob Faibussowitsch {
207c4762a1bSJed Brown   FVCtx       *ctx = (FVCtx *)vctx;
208c4762a1bSJed Brown   PetscInt     i, j, k, Mx, dof, xs, xm, sf = ctx->sf, fs = ctx->fs;
209c4762a1bSJed Brown   PetscReal    hxf, hxs, cfl_idt = 0;
210c4762a1bSJed Brown   PetscScalar *x, *f, *slope;
211c4762a1bSJed Brown   Vec          Xloc;
212c4762a1bSJed Brown   DM           da;
213c4762a1bSJed Brown 
214c4762a1bSJed Brown   PetscFunctionBeginUser;
2159566063dSJacob Faibussowitsch   PetscCall(TSGetDM(ts, &da));
2169566063dSJacob Faibussowitsch   PetscCall(DMGetLocalVector(da, &Xloc));                                 /* Xloc contains ghost points                                     */
2179566063dSJacob Faibussowitsch   PetscCall(DMDAGetInfo(da, 0, &Mx, 0, 0, 0, 0, 0, &dof, 0, 0, 0, 0, 0)); /* Mx is the number of center points                              */
218c4762a1bSJed Brown   hxs = (ctx->xmax - ctx->xmin) * 3.0 / 8.0 / ctx->sf;
219c4762a1bSJed Brown   hxf = (ctx->xmax - ctx->xmin) / 4.0 / (ctx->fs - ctx->sf);
2209566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalBegin(da, X, INSERT_VALUES, Xloc)); /* X is solution vector which does not contain ghost points       */
2219566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalEnd(da, X, INSERT_VALUES, Xloc));
222c4762a1bSJed Brown 
223dd8e379bSPierre Jolivet   PetscCall(VecZeroEntries(F)); /* F is the right-hand side function corresponds to center points */
224c4762a1bSJed Brown 
2259566063dSJacob Faibussowitsch   PetscCall(DMDAVecGetArray(da, Xloc, &x));
2269566063dSJacob Faibussowitsch   PetscCall(DMDAVecGetArray(da, F, &f));
2279566063dSJacob Faibussowitsch   PetscCall(DMDAGetArray(da, PETSC_TRUE, &slope)); /* contains ghost points                                           */
228c4762a1bSJed Brown 
2299566063dSJacob Faibussowitsch   PetscCall(DMDAGetCorners(da, &xs, 0, 0, &xm, 0, 0));
230c4762a1bSJed Brown 
231c4762a1bSJed Brown   if (ctx->bctype == FVBC_OUTFLOW) {
232c4762a1bSJed Brown     for (i = xs - 2; i < 0; i++) {
233c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[j];
234c4762a1bSJed Brown     }
235c4762a1bSJed Brown     for (i = Mx; i < xs + xm + 2; i++) {
236c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[(xs + xm - 1) * dof + j];
237c4762a1bSJed Brown     }
238c4762a1bSJed Brown   }
239c4762a1bSJed Brown   for (i = xs - 1; i < xs + xm + 1; i++) {
240c4762a1bSJed Brown     struct _LimitInfo info;
241c4762a1bSJed Brown     PetscScalar      *cjmpL, *cjmpR;
242c4762a1bSJed Brown     /* Determine the right eigenvectors R, where A = R \Lambda R^{-1} */
2439566063dSJacob Faibussowitsch     PetscCall((*ctx->physics2.characteristic2)(ctx->physics2.user, dof, &x[i * dof], ctx->R, ctx->Rinv, ctx->speeds));
244c4762a1bSJed Brown     /* Evaluate jumps across interfaces (i-1, i) and (i, i+1), put in characteristic basis */
2459566063dSJacob Faibussowitsch     PetscCall(PetscArrayzero(ctx->cjmpLR, 2 * dof));
246c4762a1bSJed Brown     cjmpL = &ctx->cjmpLR[0];
247c4762a1bSJed Brown     cjmpR = &ctx->cjmpLR[dof];
248c4762a1bSJed Brown     for (j = 0; j < dof; j++) {
249c4762a1bSJed Brown       PetscScalar jmpL, jmpR;
250c4762a1bSJed Brown       jmpL = x[(i + 0) * dof + j] - x[(i - 1) * dof + j];
251c4762a1bSJed Brown       jmpR = x[(i + 1) * dof + j] - x[(i + 0) * dof + j];
252c4762a1bSJed Brown       for (k = 0; k < dof; k++) {
253c4762a1bSJed Brown         cjmpL[k] += ctx->Rinv[k + j * dof] * jmpL;
254c4762a1bSJed Brown         cjmpR[k] += ctx->Rinv[k + j * dof] * jmpR;
255c4762a1bSJed Brown       }
256c4762a1bSJed Brown     }
257c4762a1bSJed Brown     /* Apply limiter to the left and right characteristic jumps */
258c4762a1bSJed Brown     info.m   = dof;
259c4762a1bSJed Brown     info.hxs = hxs;
260c4762a1bSJed Brown     info.hxf = hxf;
261c4762a1bSJed Brown     (*ctx->limit2)(&info, cjmpL, cjmpR, ctx->sf, ctx->fs, i, ctx->cslope);
262c4762a1bSJed Brown     for (j = 0; j < dof; j++) {
263c4762a1bSJed Brown       PetscScalar tmp = 0;
264c4762a1bSJed Brown       for (k = 0; k < dof; k++) tmp += ctx->R[j + k * dof] * ctx->cslope[k];
265c4762a1bSJed Brown       slope[i * dof + j] = tmp;
266c4762a1bSJed Brown     }
267c4762a1bSJed Brown   }
268c4762a1bSJed Brown 
269c4762a1bSJed Brown   for (i = xs; i < xs + xm + 1; i++) {
270c4762a1bSJed Brown     PetscReal    maxspeed;
271c4762a1bSJed Brown     PetscScalar *uL, *uR;
272c4762a1bSJed Brown     uL = &ctx->uLR[0];
273c4762a1bSJed Brown     uR = &ctx->uLR[dof];
274c4762a1bSJed Brown     if (i < sf) { /* slow region */
275c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
276c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
277c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
278c4762a1bSJed Brown       }
2799566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
280c4762a1bSJed Brown       if (i > xs) {
281c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(i - 1) * dof + j] -= ctx->flux[j] / hxs;
282c4762a1bSJed Brown       }
283c4762a1bSJed Brown       if (i < xs + xm) {
284c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[i * dof + j] += ctx->flux[j] / hxs;
285c4762a1bSJed Brown       }
286c4762a1bSJed Brown     } else if (i == sf) { /* interface between the slow region and the fast region */
287c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
288c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
289c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxf / 2;
290c4762a1bSJed Brown       }
2919566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
292c4762a1bSJed Brown       if (i > xs) {
293c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(i - 1) * dof + j] -= ctx->flux[j] / hxs;
294c4762a1bSJed Brown       }
295c4762a1bSJed Brown       if (i < xs + xm) {
296c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[i * dof + j] += ctx->flux[j] / hxf;
297c4762a1bSJed Brown       }
298c4762a1bSJed Brown     } else if (i > sf && i < fs) { /* fast region */
299c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
300c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxf / 2;
301c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxf / 2;
302c4762a1bSJed Brown       }
3039566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
304c4762a1bSJed Brown       if (i > xs) {
305c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(i - 1) * dof + j] -= ctx->flux[j] / hxf;
306c4762a1bSJed Brown       }
307c4762a1bSJed Brown       if (i < xs + xm) {
308c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[i * dof + j] += ctx->flux[j] / hxf;
309c4762a1bSJed Brown       }
310c4762a1bSJed Brown     } else if (i == fs) { /* interface between the fast region and the slow region */
311c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
312c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxf / 2;
313c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
314c4762a1bSJed Brown       }
3159566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
316c4762a1bSJed Brown       if (i > xs) {
317c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(i - 1) * dof + j] -= ctx->flux[j] / hxf;
318c4762a1bSJed Brown       }
319c4762a1bSJed Brown       if (i < xs + xm) {
320c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[i * dof + j] += ctx->flux[j] / hxs;
321c4762a1bSJed Brown       }
322c4762a1bSJed Brown     } else { /* slow region */
323c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
324c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
325c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
326c4762a1bSJed Brown       }
3279566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
328c4762a1bSJed Brown       cfl_idt = PetscMax(cfl_idt, PetscAbsScalar(maxspeed / hxs)); /* Max allowable value of 1/Delta t */
329c4762a1bSJed Brown       if (i > xs) {
330c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(i - 1) * dof + j] -= ctx->flux[j] / hxs;
331c4762a1bSJed Brown       }
332c4762a1bSJed Brown       if (i < xs + xm) {
333c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[i * dof + j] += ctx->flux[j] / hxs;
334c4762a1bSJed Brown       }
335c4762a1bSJed Brown     }
336c4762a1bSJed Brown   }
3379566063dSJacob Faibussowitsch   PetscCall(DMDAVecRestoreArray(da, Xloc, &x));
3389566063dSJacob Faibussowitsch   PetscCall(DMDAVecRestoreArray(da, F, &f));
3399566063dSJacob Faibussowitsch   PetscCall(DMDARestoreArray(da, PETSC_TRUE, &slope));
3409566063dSJacob Faibussowitsch   PetscCall(DMRestoreLocalVector(da, &Xloc));
341462c564dSBarry Smith   PetscCallMPI(MPIU_Allreduce(&cfl_idt, &ctx->cfl_idt, 1, MPIU_SCALAR, MPIU_MAX, PetscObjectComm((PetscObject)da)));
342c4762a1bSJed Brown   if (0) {
343c4762a1bSJed Brown     /* We need to a way to inform the TS of a CFL constraint, this is a debugging fragment */
344c4762a1bSJed Brown     PetscReal dt, tnow;
3459566063dSJacob Faibussowitsch     PetscCall(TSGetTimeStep(ts, &dt));
3469566063dSJacob Faibussowitsch     PetscCall(TSGetTime(ts, &tnow));
347c4762a1bSJed Brown     if (dt > 0.5 / ctx->cfl_idt) {
348ac530a7eSPierre Jolivet       if (1) PetscCall(PetscPrintf(ctx->comm, "Stability constraint exceeded at t=%g, dt %g > %g\n", (double)tnow, (double)dt, (double)(1 / (2 * ctx->cfl_idt))));
349ac530a7eSPierre Jolivet       else SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_UNKNOWN_TYPE, "Stability constraint exceeded, %g > %g", (double)dt, (double)(ctx->cfl / ctx->cfl_idt));
350c4762a1bSJed Brown     }
351c4762a1bSJed Brown   }
3523ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
353c4762a1bSJed Brown }
354c4762a1bSJed Brown 
355c4762a1bSJed Brown /* --------------------------------- Finite Volume Solver for slow components ----------------------------------- */
FVRHSFunctionslow_2WaySplit(TS ts,PetscReal time,Vec X,Vec F,void * vctx)356d71ae5a4SJacob Faibussowitsch PetscErrorCode FVRHSFunctionslow_2WaySplit(TS ts, PetscReal time, Vec X, Vec F, void *vctx)
357d71ae5a4SJacob Faibussowitsch {
358c4762a1bSJed Brown   FVCtx       *ctx = (FVCtx *)vctx;
359c4762a1bSJed Brown   PetscInt     i, j, k, Mx, dof, xs, xm, islow = 0, sf = ctx->sf, fs = ctx->fs, lsbwidth = ctx->lsbwidth, rsbwidth = ctx->rsbwidth;
360c4762a1bSJed Brown   PetscReal    hxs, hxf, cfl_idt = 0;
361c4762a1bSJed Brown   PetscScalar *x, *f, *slope;
362c4762a1bSJed Brown   Vec          Xloc;
363c4762a1bSJed Brown   DM           da;
364c4762a1bSJed Brown 
365c4762a1bSJed Brown   PetscFunctionBeginUser;
3669566063dSJacob Faibussowitsch   PetscCall(TSGetDM(ts, &da));
3679566063dSJacob Faibussowitsch   PetscCall(DMGetLocalVector(da, &Xloc));
3689566063dSJacob Faibussowitsch   PetscCall(DMDAGetInfo(da, 0, &Mx, 0, 0, 0, 0, 0, &dof, 0, 0, 0, 0, 0));
369c4762a1bSJed Brown   hxs = (ctx->xmax - ctx->xmin) * 3.0 / 8.0 / ctx->sf;
370c4762a1bSJed Brown   hxf = (ctx->xmax - ctx->xmin) / 4.0 / (ctx->fs - ctx->sf);
3719566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalBegin(da, X, INSERT_VALUES, Xloc));
3729566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalEnd(da, X, INSERT_VALUES, Xloc));
3739566063dSJacob Faibussowitsch   PetscCall(VecZeroEntries(F));
3749566063dSJacob Faibussowitsch   PetscCall(DMDAVecGetArray(da, Xloc, &x));
3759566063dSJacob Faibussowitsch   PetscCall(VecGetArray(F, &f));
3769566063dSJacob Faibussowitsch   PetscCall(DMDAGetArray(da, PETSC_TRUE, &slope));
3779566063dSJacob Faibussowitsch   PetscCall(DMDAGetCorners(da, &xs, 0, 0, &xm, 0, 0));
378c4762a1bSJed Brown 
379c4762a1bSJed Brown   if (ctx->bctype == FVBC_OUTFLOW) {
380c4762a1bSJed Brown     for (i = xs - 2; i < 0; i++) {
381c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[j];
382c4762a1bSJed Brown     }
383c4762a1bSJed Brown     for (i = Mx; i < xs + xm + 2; i++) {
384c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[(xs + xm - 1) * dof + j];
385c4762a1bSJed Brown     }
386c4762a1bSJed Brown   }
387c4762a1bSJed Brown   for (i = xs - 1; i < xs + xm + 1; i++) {
388c4762a1bSJed Brown     struct _LimitInfo info;
389c4762a1bSJed Brown     PetscScalar      *cjmpL, *cjmpR;
390c4762a1bSJed Brown     if (i < sf - lsbwidth + 1 || i > fs + rsbwidth - 2) { /* slow components and the first and last fast components */
391c4762a1bSJed Brown       /* Determine the right eigenvectors R, where A = R \Lambda R^{-1} */
3929566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.characteristic2)(ctx->physics2.user, dof, &x[i * dof], ctx->R, ctx->Rinv, ctx->speeds));
393c4762a1bSJed Brown       /* Evaluate jumps across interfaces (i-1, i) and (i, i+1), put in characteristic basis */
3949566063dSJacob Faibussowitsch       PetscCall(PetscArrayzero(ctx->cjmpLR, 2 * dof));
395c4762a1bSJed Brown       cjmpL = &ctx->cjmpLR[0];
396c4762a1bSJed Brown       cjmpR = &ctx->cjmpLR[dof];
397c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
398c4762a1bSJed Brown         PetscScalar jmpL, jmpR;
399c4762a1bSJed Brown         jmpL = x[(i + 0) * dof + j] - x[(i - 1) * dof + j];
400c4762a1bSJed Brown         jmpR = x[(i + 1) * dof + j] - x[(i + 0) * dof + j];
401c4762a1bSJed Brown         for (k = 0; k < dof; k++) {
402c4762a1bSJed Brown           cjmpL[k] += ctx->Rinv[k + j * dof] * jmpL;
403c4762a1bSJed Brown           cjmpR[k] += ctx->Rinv[k + j * dof] * jmpR;
404c4762a1bSJed Brown         }
405c4762a1bSJed Brown       }
406c4762a1bSJed Brown       /* Apply limiter to the left and right characteristic jumps */
407c4762a1bSJed Brown       info.m   = dof;
408c4762a1bSJed Brown       info.hxs = hxs;
409c4762a1bSJed Brown       info.hxf = hxf;
410c4762a1bSJed Brown       (*ctx->limit2)(&info, cjmpL, cjmpR, ctx->sf, ctx->fs, i, ctx->cslope);
411c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
412c4762a1bSJed Brown         PetscScalar tmp = 0;
413c4762a1bSJed Brown         for (k = 0; k < dof; k++) tmp += ctx->R[j + k * dof] * ctx->cslope[k];
414c4762a1bSJed Brown         slope[i * dof + j] = tmp;
415c4762a1bSJed Brown       }
416c4762a1bSJed Brown     }
417c4762a1bSJed Brown   }
418c4762a1bSJed Brown 
419c4762a1bSJed Brown   for (i = xs; i < xs + xm + 1; i++) {
420c4762a1bSJed Brown     PetscReal    maxspeed;
421c4762a1bSJed Brown     PetscScalar *uL, *uR;
422c4762a1bSJed Brown     uL = &ctx->uLR[0];
423c4762a1bSJed Brown     uR = &ctx->uLR[dof];
424c4762a1bSJed Brown     if (i < sf - lsbwidth) { /* slow region */
425c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
426c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
427c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
428c4762a1bSJed Brown       }
4299566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
430c4762a1bSJed Brown       cfl_idt = PetscMax(cfl_idt, PetscAbsScalar(maxspeed / hxs)); /* Max allowable value of 1/Delta t */
431c4762a1bSJed Brown       if (i > xs) {
432c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(islow - 1) * dof + j] -= ctx->flux[j] / hxs;
433c4762a1bSJed Brown       }
434c4762a1bSJed Brown       if (i < xs + xm) {
435c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[islow * dof + j] += ctx->flux[j] / hxs;
436c4762a1bSJed Brown         islow++;
437c4762a1bSJed Brown       }
438c4762a1bSJed Brown     }
439c4762a1bSJed Brown     if (i == sf - lsbwidth) { /* interface between the slow region and the fast region */
440c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
441c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
442c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
443c4762a1bSJed Brown       }
4449566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
445c4762a1bSJed Brown       if (i > xs) {
446c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(islow - 1) * dof + j] -= ctx->flux[j] / hxs;
447c4762a1bSJed Brown       }
448c4762a1bSJed Brown     }
449c4762a1bSJed Brown     if (i == fs + rsbwidth) { /* slow region */
450c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
451c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
452c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
453c4762a1bSJed Brown       }
4549566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
455c4762a1bSJed Brown       if (i < xs + xm) {
456c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[islow * dof + j] += ctx->flux[j] / hxs;
457c4762a1bSJed Brown         islow++;
458c4762a1bSJed Brown       }
459c4762a1bSJed Brown     }
460c4762a1bSJed Brown     if (i > fs + rsbwidth) { /* slow region */
461c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
462c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
463c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
464c4762a1bSJed Brown       }
4659566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
466c4762a1bSJed Brown       if (i > xs) {
467c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(islow - 1) * dof + j] -= ctx->flux[j] / hxs;
468c4762a1bSJed Brown       }
469c4762a1bSJed Brown       if (i < xs + xm) {
470c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[islow * dof + j] += ctx->flux[j] / hxs;
471c4762a1bSJed Brown         islow++;
472c4762a1bSJed Brown       }
473c4762a1bSJed Brown     }
474c4762a1bSJed Brown   }
4759566063dSJacob Faibussowitsch   PetscCall(DMDAVecRestoreArray(da, Xloc, &x));
4769566063dSJacob Faibussowitsch   PetscCall(VecRestoreArray(F, &f));
4779566063dSJacob Faibussowitsch   PetscCall(DMDARestoreArray(da, PETSC_TRUE, &slope));
4789566063dSJacob Faibussowitsch   PetscCall(DMRestoreLocalVector(da, &Xloc));
479462c564dSBarry Smith   PetscCallMPI(MPIU_Allreduce(&cfl_idt, &ctx->cfl_idt, 1, MPIU_SCALAR, MPIU_MAX, PetscObjectComm((PetscObject)da)));
4803ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
481c4762a1bSJed Brown }
482c4762a1bSJed Brown 
FVRHSFunctionslowbuffer_2WaySplit(TS ts,PetscReal time,Vec X,Vec F,void * vctx)483d71ae5a4SJacob Faibussowitsch PetscErrorCode FVRHSFunctionslowbuffer_2WaySplit(TS ts, PetscReal time, Vec X, Vec F, void *vctx)
484d71ae5a4SJacob Faibussowitsch {
485c4762a1bSJed Brown   FVCtx       *ctx = (FVCtx *)vctx;
486c4762a1bSJed Brown   PetscInt     i, j, k, Mx, dof, xs, xm, islow = 0, sf = ctx->sf, fs = ctx->fs, lsbwidth = ctx->lsbwidth, rsbwidth = ctx->rsbwidth;
487c4762a1bSJed Brown   PetscReal    hxs, hxf;
488c4762a1bSJed Brown   PetscScalar *x, *f, *slope;
489c4762a1bSJed Brown   Vec          Xloc;
490c4762a1bSJed Brown   DM           da;
491c4762a1bSJed Brown 
492c4762a1bSJed Brown   PetscFunctionBeginUser;
4939566063dSJacob Faibussowitsch   PetscCall(TSGetDM(ts, &da));
4949566063dSJacob Faibussowitsch   PetscCall(DMGetLocalVector(da, &Xloc));
4959566063dSJacob Faibussowitsch   PetscCall(DMDAGetInfo(da, 0, &Mx, 0, 0, 0, 0, 0, &dof, 0, 0, 0, 0, 0));
496c4762a1bSJed Brown   hxs = (ctx->xmax - ctx->xmin) * 3.0 / 8.0 / ctx->sf;
497c4762a1bSJed Brown   hxf = (ctx->xmax - ctx->xmin) / 4.0 / (ctx->fs - ctx->sf);
4989566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalBegin(da, X, INSERT_VALUES, Xloc));
4999566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalEnd(da, X, INSERT_VALUES, Xloc));
5009566063dSJacob Faibussowitsch   PetscCall(VecZeroEntries(F));
5019566063dSJacob Faibussowitsch   PetscCall(DMDAVecGetArray(da, Xloc, &x));
5029566063dSJacob Faibussowitsch   PetscCall(VecGetArray(F, &f));
5039566063dSJacob Faibussowitsch   PetscCall(DMDAGetArray(da, PETSC_TRUE, &slope));
5049566063dSJacob Faibussowitsch   PetscCall(DMDAGetCorners(da, &xs, 0, 0, &xm, 0, 0));
505c4762a1bSJed Brown 
506c4762a1bSJed Brown   if (ctx->bctype == FVBC_OUTFLOW) {
507c4762a1bSJed Brown     for (i = xs - 2; i < 0; i++) {
508c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[j];
509c4762a1bSJed Brown     }
510c4762a1bSJed Brown     for (i = Mx; i < xs + xm + 2; i++) {
511c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[(xs + xm - 1) * dof + j];
512c4762a1bSJed Brown     }
513c4762a1bSJed Brown   }
514c4762a1bSJed Brown   for (i = xs - 1; i < xs + xm + 1; i++) {
515c4762a1bSJed Brown     struct _LimitInfo info;
516c4762a1bSJed Brown     PetscScalar      *cjmpL, *cjmpR;
517c4762a1bSJed Brown     if ((i > sf - lsbwidth - 2 && i < sf + 1) || (i > fs - 2 && i < fs + rsbwidth + 1)) {
518c4762a1bSJed Brown       /* Determine the right eigenvectors R, where A = R \Lambda R^{-1} */
5199566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.characteristic2)(ctx->physics2.user, dof, &x[i * dof], ctx->R, ctx->Rinv, ctx->speeds));
520c4762a1bSJed Brown       /* Evaluate jumps across interfaces (i-1, i) and (i, i+1), put in characteristic basis */
5219566063dSJacob Faibussowitsch       PetscCall(PetscArrayzero(ctx->cjmpLR, 2 * dof));
522c4762a1bSJed Brown       cjmpL = &ctx->cjmpLR[0];
523c4762a1bSJed Brown       cjmpR = &ctx->cjmpLR[dof];
524c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
525c4762a1bSJed Brown         PetscScalar jmpL, jmpR;
526c4762a1bSJed Brown         jmpL = x[(i + 0) * dof + j] - x[(i - 1) * dof + j];
527c4762a1bSJed Brown         jmpR = x[(i + 1) * dof + j] - x[(i + 0) * dof + j];
528c4762a1bSJed Brown         for (k = 0; k < dof; k++) {
529c4762a1bSJed Brown           cjmpL[k] += ctx->Rinv[k + j * dof] * jmpL;
530c4762a1bSJed Brown           cjmpR[k] += ctx->Rinv[k + j * dof] * jmpR;
531c4762a1bSJed Brown         }
532c4762a1bSJed Brown       }
533c4762a1bSJed Brown       /* Apply limiter to the left and right characteristic jumps */
534c4762a1bSJed Brown       info.m   = dof;
535c4762a1bSJed Brown       info.hxs = hxs;
536c4762a1bSJed Brown       info.hxf = hxf;
537c4762a1bSJed Brown       (*ctx->limit2)(&info, cjmpL, cjmpR, ctx->sf, ctx->fs, i, ctx->cslope);
538c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
539c4762a1bSJed Brown         PetscScalar tmp = 0;
540c4762a1bSJed Brown         for (k = 0; k < dof; k++) tmp += ctx->R[j + k * dof] * ctx->cslope[k];
541c4762a1bSJed Brown         slope[i * dof + j] = tmp;
542c4762a1bSJed Brown       }
543c4762a1bSJed Brown     }
544c4762a1bSJed Brown   }
545c4762a1bSJed Brown 
546c4762a1bSJed Brown   for (i = xs; i < xs + xm + 1; i++) {
547c4762a1bSJed Brown     PetscReal    maxspeed;
548c4762a1bSJed Brown     PetscScalar *uL, *uR;
549c4762a1bSJed Brown     uL = &ctx->uLR[0];
550c4762a1bSJed Brown     uR = &ctx->uLR[dof];
551c4762a1bSJed Brown     if (i == sf - lsbwidth) {
552c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
553c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
554c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
555c4762a1bSJed Brown       }
5569566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
557c4762a1bSJed Brown       if (i < xs + xm) {
558c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[islow * dof + j] += ctx->flux[j] / hxs;
559c4762a1bSJed Brown         islow++;
560c4762a1bSJed Brown       }
561c4762a1bSJed Brown     }
562c4762a1bSJed Brown     if (i > sf - lsbwidth && i < sf) {
563c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
564c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
565c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
566c4762a1bSJed Brown       }
5679566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
568c4762a1bSJed Brown       if (i > xs) {
569c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(islow - 1) * dof + j] -= ctx->flux[j] / hxs;
570c4762a1bSJed Brown       }
571c4762a1bSJed Brown       if (i < xs + xm) {
572c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[islow * dof + j] += ctx->flux[j] / hxs;
573c4762a1bSJed Brown         islow++;
574c4762a1bSJed Brown       }
575c4762a1bSJed Brown     }
576c4762a1bSJed Brown     if (i == sf) { /* interface between the slow region and the fast region */
577c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
578c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
579c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxf / 2;
580c4762a1bSJed Brown       }
5819566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
582c4762a1bSJed Brown       if (i > xs) {
583c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(islow - 1) * dof + j] -= ctx->flux[j] / hxs;
584c4762a1bSJed Brown       }
585c4762a1bSJed Brown     }
586c4762a1bSJed Brown     if (i == fs) { /* interface between the fast region and the slow region */
587c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
588c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxf / 2;
589c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
590c4762a1bSJed Brown       }
5919566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
592c4762a1bSJed Brown       if (i < xs + xm) {
593c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[islow * dof + j] += ctx->flux[j] / hxs;
594c4762a1bSJed Brown         islow++;
595c4762a1bSJed Brown       }
596c4762a1bSJed Brown     }
597c4762a1bSJed Brown     if (i > fs && i < fs + rsbwidth) {
598c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
599c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
600c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
601c4762a1bSJed Brown       }
6029566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
603c4762a1bSJed Brown       if (i > xs) {
604c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(islow - 1) * dof + j] -= ctx->flux[j] / hxs;
605c4762a1bSJed Brown       }
606c4762a1bSJed Brown       if (i < xs + xm) {
607c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[islow * dof + j] += ctx->flux[j] / hxs;
608c4762a1bSJed Brown         islow++;
609c4762a1bSJed Brown       }
610c4762a1bSJed Brown     }
611c4762a1bSJed Brown     if (i == fs + rsbwidth) {
612c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
613c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
614c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
615c4762a1bSJed Brown       }
6169566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
617c4762a1bSJed Brown       if (i > xs) {
618c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(islow - 1) * dof + j] -= ctx->flux[j] / hxs;
619c4762a1bSJed Brown       }
620c4762a1bSJed Brown     }
621c4762a1bSJed Brown   }
6229566063dSJacob Faibussowitsch   PetscCall(DMDAVecRestoreArray(da, Xloc, &x));
6239566063dSJacob Faibussowitsch   PetscCall(VecRestoreArray(F, &f));
6249566063dSJacob Faibussowitsch   PetscCall(DMDARestoreArray(da, PETSC_TRUE, &slope));
6259566063dSJacob Faibussowitsch   PetscCall(DMRestoreLocalVector(da, &Xloc));
6263ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
627c4762a1bSJed Brown }
628c4762a1bSJed Brown 
629c4762a1bSJed Brown /* --------------------------------- Finite Volume Solver for fast  parts ----------------------------------- */
FVRHSFunctionfast_2WaySplit(TS ts,PetscReal time,Vec X,Vec F,void * vctx)630d71ae5a4SJacob Faibussowitsch PetscErrorCode FVRHSFunctionfast_2WaySplit(TS ts, PetscReal time, Vec X, Vec F, void *vctx)
631d71ae5a4SJacob Faibussowitsch {
632c4762a1bSJed Brown   FVCtx       *ctx = (FVCtx *)vctx;
633c4762a1bSJed Brown   PetscInt     i, j, k, Mx, dof, xs, xm, ifast = 0, sf = ctx->sf, fs = ctx->fs;
634c4762a1bSJed Brown   PetscReal    hxs, hxf;
635c4762a1bSJed Brown   PetscScalar *x, *f, *slope;
636c4762a1bSJed Brown   Vec          Xloc;
637c4762a1bSJed Brown   DM           da;
638c4762a1bSJed Brown 
639c4762a1bSJed Brown   PetscFunctionBeginUser;
6409566063dSJacob Faibussowitsch   PetscCall(TSGetDM(ts, &da));
6419566063dSJacob Faibussowitsch   PetscCall(DMGetLocalVector(da, &Xloc));
6429566063dSJacob Faibussowitsch   PetscCall(DMDAGetInfo(da, 0, &Mx, 0, 0, 0, 0, 0, &dof, 0, 0, 0, 0, 0));
643c4762a1bSJed Brown   hxs = (ctx->xmax - ctx->xmin) * 3.0 / 8.0 / ctx->sf;
644c4762a1bSJed Brown   hxf = (ctx->xmax - ctx->xmin) / 4.0 / (ctx->fs - ctx->sf);
6459566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalBegin(da, X, INSERT_VALUES, Xloc));
6469566063dSJacob Faibussowitsch   PetscCall(DMGlobalToLocalEnd(da, X, INSERT_VALUES, Xloc));
6479566063dSJacob Faibussowitsch   PetscCall(VecZeroEntries(F));
6489566063dSJacob Faibussowitsch   PetscCall(DMDAVecGetArray(da, Xloc, &x));
6499566063dSJacob Faibussowitsch   PetscCall(VecGetArray(F, &f));
6509566063dSJacob Faibussowitsch   PetscCall(DMDAGetArray(da, PETSC_TRUE, &slope));
6519566063dSJacob Faibussowitsch   PetscCall(DMDAGetCorners(da, &xs, 0, 0, &xm, 0, 0));
652c4762a1bSJed Brown 
653c4762a1bSJed Brown   if (ctx->bctype == FVBC_OUTFLOW) {
654c4762a1bSJed Brown     for (i = xs - 2; i < 0; i++) {
655c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[j];
656c4762a1bSJed Brown     }
657c4762a1bSJed Brown     for (i = Mx; i < xs + xm + 2; i++) {
658c4762a1bSJed Brown       for (j = 0; j < dof; j++) x[i * dof + j] = x[(xs + xm - 1) * dof + j];
659c4762a1bSJed Brown     }
660c4762a1bSJed Brown   }
661c4762a1bSJed Brown   for (i = xs - 1; i < xs + xm + 1; i++) {
662c4762a1bSJed Brown     struct _LimitInfo info;
663c4762a1bSJed Brown     PetscScalar      *cjmpL, *cjmpR;
664c4762a1bSJed Brown     if (i > sf - 2 && i < fs + 1) {
6659566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.characteristic2)(ctx->physics2.user, dof, &x[i * dof], ctx->R, ctx->Rinv, ctx->speeds));
6669566063dSJacob Faibussowitsch       PetscCall(PetscArrayzero(ctx->cjmpLR, 2 * dof));
667c4762a1bSJed Brown       cjmpL = &ctx->cjmpLR[0];
668c4762a1bSJed Brown       cjmpR = &ctx->cjmpLR[dof];
669c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
670c4762a1bSJed Brown         PetscScalar jmpL, jmpR;
671c4762a1bSJed Brown         jmpL = x[(i + 0) * dof + j] - x[(i - 1) * dof + j];
672c4762a1bSJed Brown         jmpR = x[(i + 1) * dof + j] - x[(i + 0) * dof + j];
673c4762a1bSJed Brown         for (k = 0; k < dof; k++) {
674c4762a1bSJed Brown           cjmpL[k] += ctx->Rinv[k + j * dof] * jmpL;
675c4762a1bSJed Brown           cjmpR[k] += ctx->Rinv[k + j * dof] * jmpR;
676c4762a1bSJed Brown         }
677c4762a1bSJed Brown       }
678c4762a1bSJed Brown       /* Apply limiter to the left and right characteristic jumps */
679c4762a1bSJed Brown       info.m   = dof;
680c4762a1bSJed Brown       info.hxs = hxs;
681c4762a1bSJed Brown       info.hxf = hxf;
682c4762a1bSJed Brown       (*ctx->limit2)(&info, cjmpL, cjmpR, ctx->sf, ctx->fs, i, ctx->cslope);
683c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
684c4762a1bSJed Brown         PetscScalar tmp = 0;
685c4762a1bSJed Brown         for (k = 0; k < dof; k++) tmp += ctx->R[j + k * dof] * ctx->cslope[k];
686c4762a1bSJed Brown         slope[i * dof + j] = tmp;
687c4762a1bSJed Brown       }
688c4762a1bSJed Brown     }
689c4762a1bSJed Brown   }
690c4762a1bSJed Brown 
691c4762a1bSJed Brown   for (i = xs; i < xs + xm + 1; i++) {
692c4762a1bSJed Brown     PetscReal    maxspeed;
693c4762a1bSJed Brown     PetscScalar *uL, *uR;
694c4762a1bSJed Brown     uL = &ctx->uLR[0];
695c4762a1bSJed Brown     uR = &ctx->uLR[dof];
696c4762a1bSJed Brown     if (i == sf) { /* interface between the slow region and the fast region */
697c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
698c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxs / 2;
699c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxf / 2;
700c4762a1bSJed Brown       }
7019566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
702c4762a1bSJed Brown       if (i < xs + xm) {
703c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[ifast * dof + j] += ctx->flux[j] / hxf;
704c4762a1bSJed Brown         ifast++;
705c4762a1bSJed Brown       }
706c4762a1bSJed Brown     }
707c4762a1bSJed Brown     if (i > sf && i < fs) { /* fast region */
708c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
709c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxf / 2;
710c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxf / 2;
711c4762a1bSJed Brown       }
7129566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
713c4762a1bSJed Brown       if (i > xs) {
714c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(ifast - 1) * dof + j] -= ctx->flux[j] / hxf;
715c4762a1bSJed Brown       }
716c4762a1bSJed Brown       if (i < xs + xm) {
717c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[ifast * dof + j] += ctx->flux[j] / hxf;
718c4762a1bSJed Brown         ifast++;
719c4762a1bSJed Brown       }
720c4762a1bSJed Brown     }
721c4762a1bSJed Brown     if (i == fs) { /* interface between the fast region and the slow region */
722c4762a1bSJed Brown       for (j = 0; j < dof; j++) {
723c4762a1bSJed Brown         uL[j] = x[(i - 1) * dof + j] + slope[(i - 1) * dof + j] * hxf / 2;
724c4762a1bSJed Brown         uR[j] = x[(i - 0) * dof + j] - slope[(i - 0) * dof + j] * hxs / 2;
725c4762a1bSJed Brown       }
7269566063dSJacob Faibussowitsch       PetscCall((*ctx->physics2.riemann2)(ctx->physics2.user, dof, uL, uR, ctx->flux, &maxspeed));
727c4762a1bSJed Brown       if (i > xs) {
728c4762a1bSJed Brown         for (j = 0; j < dof; j++) f[(ifast - 1) * dof + j] -= ctx->flux[j] / hxf;
729c4762a1bSJed Brown       }
730c4762a1bSJed Brown     }
731c4762a1bSJed Brown   }
7329566063dSJacob Faibussowitsch   PetscCall(DMDAVecRestoreArray(da, Xloc, &x));
7339566063dSJacob Faibussowitsch   PetscCall(VecRestoreArray(F, &f));
7349566063dSJacob Faibussowitsch   PetscCall(DMDARestoreArray(da, PETSC_TRUE, &slope));
7359566063dSJacob Faibussowitsch   PetscCall(DMRestoreLocalVector(da, &Xloc));
7363ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
737c4762a1bSJed Brown }
738c4762a1bSJed Brown 
main(int argc,char * argv[])739d71ae5a4SJacob Faibussowitsch int main(int argc, char *argv[])
740d71ae5a4SJacob Faibussowitsch {
741c4762a1bSJed Brown   char              lname[256] = "mc", physname[256] = "advect", final_fname[256] = "solution.m";
742c4762a1bSJed Brown   PetscFunctionList limiters = 0, physics = 0;
743c4762a1bSJed Brown   MPI_Comm          comm;
744c4762a1bSJed Brown   TS                ts;
745c4762a1bSJed Brown   DM                da;
746c4762a1bSJed Brown   Vec               X, X0, R;
747c4762a1bSJed Brown   FVCtx             ctx;
748c4762a1bSJed Brown   PetscInt          i, k, dof, xs, xm, Mx, draw = 0, count_slow, count_fast, islow = 0, ifast = 0, islowbuffer = 0, *index_slow, *index_fast, *index_slowbuffer;
749c4762a1bSJed Brown   PetscBool         view_final = PETSC_FALSE;
750c4762a1bSJed Brown   PetscReal         ptime;
751c4762a1bSJed Brown 
752327415f7SBarry Smith   PetscFunctionBeginUser;
7539566063dSJacob Faibussowitsch   PetscCall(PetscInitialize(&argc, &argv, 0, help));
754c4762a1bSJed Brown   comm = PETSC_COMM_WORLD;
7559566063dSJacob Faibussowitsch   PetscCall(PetscMemzero(&ctx, sizeof(ctx)));
756c4762a1bSJed Brown 
757c4762a1bSJed Brown   /* Register limiters to be available on the command line */
7589566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "upwind", Limit2_Upwind));
7599566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "lax-wendroff", Limit2_LaxWendroff));
7609566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "beam-warming", Limit2_BeamWarming));
7619566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "fromm", Limit2_Fromm));
7629566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "minmod", Limit2_Minmod));
7639566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "superbee", Limit2_Superbee));
7649566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "mc", Limit2_MC));
7659566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&limiters, "koren3", Limit2_Koren3));
766c4762a1bSJed Brown 
767c4762a1bSJed Brown   /* Register physical models to be available on the command line */
7689566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&physics, "advect", PhysicsCreate_Advect));
769c4762a1bSJed Brown 
770c4762a1bSJed Brown   ctx.comm   = comm;
771c4762a1bSJed Brown   ctx.cfl    = 0.9;
772c4762a1bSJed Brown   ctx.bctype = FVBC_PERIODIC;
773c4762a1bSJed Brown   ctx.xmin   = -1.0;
774c4762a1bSJed Brown   ctx.xmax   = 1.0;
775d0609cedSBarry Smith   PetscOptionsBegin(comm, NULL, "Finite Volume solver options", "");
7769566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-xmin", "X min", "", ctx.xmin, &ctx.xmin, NULL));
7779566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-xmax", "X max", "", ctx.xmax, &ctx.xmax, NULL));
7789566063dSJacob Faibussowitsch   PetscCall(PetscOptionsFList("-limit", "Name of flux imiter to use", "", limiters, lname, lname, sizeof(lname), NULL));
7799566063dSJacob Faibussowitsch   PetscCall(PetscOptionsInt("-draw", "Draw solution vector, bitwise OR of (1=initial,2=final,4=final error)", "", draw, &draw, NULL));
7809566063dSJacob Faibussowitsch   PetscCall(PetscOptionsString("-view_final", "Write final solution in ASCII MATLAB format to given file name", "", final_fname, final_fname, sizeof(final_fname), &view_final));
7819566063dSJacob Faibussowitsch   PetscCall(PetscOptionsInt("-initial", "Initial condition (depends on the physics)", "", ctx.initial, &ctx.initial, NULL));
7829566063dSJacob Faibussowitsch   PetscCall(PetscOptionsBool("-exact", "Compare errors with exact solution", "", ctx.exact, &ctx.exact, NULL));
7839566063dSJacob Faibussowitsch   PetscCall(PetscOptionsBool("-simulation", "Compare errors with reference solution", "", ctx.simulation, &ctx.simulation, NULL));
7849566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-cfl", "CFL number to time step at", "", ctx.cfl, &ctx.cfl, NULL));
7859566063dSJacob Faibussowitsch   PetscCall(PetscOptionsEnum("-bc_type", "Boundary condition", "", FVBCTypes, (PetscEnum)ctx.bctype, (PetscEnum *)&ctx.bctype, NULL));
7869566063dSJacob Faibussowitsch   PetscCall(PetscOptionsInt("-hratio", "Spacing ratio", "", ctx.hratio, &ctx.hratio, NULL));
787d0609cedSBarry Smith   PetscOptionsEnd();
788c4762a1bSJed Brown 
789c4762a1bSJed Brown   /* Choose the limiter from the list of registered limiters */
7909566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListFind(limiters, lname, &ctx.limit2));
7913c633725SBarry Smith   PetscCheck(ctx.limit2, PETSC_COMM_SELF, PETSC_ERR_ARG_UNKNOWN_TYPE, "Limiter '%s' not found", lname);
792c4762a1bSJed Brown 
793c4762a1bSJed Brown   /* Choose the physics from the list of registered models */
794c4762a1bSJed Brown   {
795c4762a1bSJed Brown     PetscErrorCode (*r)(FVCtx *);
7969566063dSJacob Faibussowitsch     PetscCall(PetscFunctionListFind(physics, physname, &r));
7973c633725SBarry Smith     PetscCheck(r, PETSC_COMM_SELF, PETSC_ERR_ARG_UNKNOWN_TYPE, "Physics '%s' not found", physname);
798c4762a1bSJed Brown     /* Create the physics, will set the number of fields and their names */
7999566063dSJacob Faibussowitsch     PetscCall((*r)(&ctx));
800c4762a1bSJed Brown   }
801c4762a1bSJed Brown 
802c4762a1bSJed Brown   /* Create a DMDA to manage the parallel grid */
8039566063dSJacob Faibussowitsch   PetscCall(DMDACreate1d(comm, DM_BOUNDARY_PERIODIC, 50, ctx.physics2.dof, 2, NULL, &da));
8049566063dSJacob Faibussowitsch   PetscCall(DMSetFromOptions(da));
8059566063dSJacob Faibussowitsch   PetscCall(DMSetUp(da));
806c4762a1bSJed Brown   /* Inform the DMDA of the field names provided by the physics. */
807c4762a1bSJed Brown   /* The names will be shown in the title bars when run with -ts_monitor_draw_solution */
80848a46eb9SPierre Jolivet   for (i = 0; i < ctx.physics2.dof; i++) PetscCall(DMDASetFieldName(da, i, ctx.physics2.fieldname[i]));
8099566063dSJacob Faibussowitsch   PetscCall(DMDAGetInfo(da, 0, &Mx, 0, 0, 0, 0, 0, &dof, 0, 0, 0, 0, 0));
8109566063dSJacob Faibussowitsch   PetscCall(DMDAGetCorners(da, &xs, 0, 0, &xm, 0, 0));
811c4762a1bSJed Brown 
812c4762a1bSJed Brown   /* Set coordinates of cell centers */
8139566063dSJacob Faibussowitsch   PetscCall(DMDASetUniformCoordinates(da, ctx.xmin + 0.5 * (ctx.xmax - ctx.xmin) / Mx, ctx.xmax + 0.5 * (ctx.xmax - ctx.xmin) / Mx, 0, 0, 0, 0));
814c4762a1bSJed Brown 
815c4762a1bSJed Brown   /* Allocate work space for the Finite Volume solver (so it doesn't have to be reallocated on each function evaluation) */
8169566063dSJacob Faibussowitsch   PetscCall(PetscMalloc4(dof * dof, &ctx.R, dof * dof, &ctx.Rinv, 2 * dof, &ctx.cjmpLR, 1 * dof, &ctx.cslope));
8179566063dSJacob Faibussowitsch   PetscCall(PetscMalloc3(2 * dof, &ctx.uLR, dof, &ctx.flux, dof, &ctx.speeds));
818c4762a1bSJed Brown 
819c4762a1bSJed Brown   /* Create a vector to store the solution and to save the initial state */
8209566063dSJacob Faibussowitsch   PetscCall(DMCreateGlobalVector(da, &X));
8219566063dSJacob Faibussowitsch   PetscCall(VecDuplicate(X, &X0));
8229566063dSJacob Faibussowitsch   PetscCall(VecDuplicate(X, &R));
823c4762a1bSJed Brown 
824c4762a1bSJed Brown   /* create index for slow parts and fast parts,
825c4762a1bSJed Brown      count_slow + count_fast = Mx, counts_slow*hs = 0.5, counts_fast*hf = 0.5 */
826c4762a1bSJed Brown   count_slow = Mx / (1.0 + ctx.hratio / 3.0);
82708401ef6SPierre Jolivet   PetscCheck(count_slow % 2 == 0, PETSC_COMM_WORLD, PETSC_ERR_USER, "Please adjust grid size Mx (-da_grid_x) and hratio (-hratio) so that Mx/(1+hartio/3) is even");
828c4762a1bSJed Brown   count_fast = Mx - count_slow;
829c4762a1bSJed Brown   ctx.sf     = count_slow / 2;
830c4762a1bSJed Brown   ctx.fs     = ctx.sf + count_fast;
8319566063dSJacob Faibussowitsch   PetscCall(PetscMalloc1(xm * dof, &index_slow));
8329566063dSJacob Faibussowitsch   PetscCall(PetscMalloc1(xm * dof, &index_fast));
8339566063dSJacob Faibussowitsch   PetscCall(PetscMalloc1(6 * dof, &index_slowbuffer));
834c4762a1bSJed Brown   if (((AdvectCtx *)ctx.physics2.user)->a > 0) {
835c4762a1bSJed Brown     ctx.lsbwidth = 2;
836c4762a1bSJed Brown     ctx.rsbwidth = 4;
837c4762a1bSJed Brown   } else {
838c4762a1bSJed Brown     ctx.lsbwidth = 4;
839c4762a1bSJed Brown     ctx.rsbwidth = 2;
840c4762a1bSJed Brown   }
841c4762a1bSJed Brown   for (i = xs; i < xs + xm; i++) {
842c4762a1bSJed Brown     if (i < ctx.sf - ctx.lsbwidth || i > ctx.fs + ctx.rsbwidth - 1)
843c4762a1bSJed Brown       for (k = 0; k < dof; k++) index_slow[islow++] = i * dof + k;
844c4762a1bSJed Brown     else if ((i >= ctx.sf - ctx.lsbwidth && i < ctx.sf) || (i > ctx.fs - 1 && i <= ctx.fs + ctx.rsbwidth - 1))
845c4762a1bSJed Brown       for (k = 0; k < dof; k++) index_slowbuffer[islowbuffer++] = i * dof + k;
846c4762a1bSJed Brown     else
847c4762a1bSJed Brown       for (k = 0; k < dof; k++) index_fast[ifast++] = i * dof + k;
848c4762a1bSJed Brown   }
8499566063dSJacob Faibussowitsch   PetscCall(ISCreateGeneral(PETSC_COMM_WORLD, islow, index_slow, PETSC_COPY_VALUES, &ctx.iss));
8509566063dSJacob Faibussowitsch   PetscCall(ISCreateGeneral(PETSC_COMM_WORLD, ifast, index_fast, PETSC_COPY_VALUES, &ctx.isf));
8519566063dSJacob Faibussowitsch   PetscCall(ISCreateGeneral(PETSC_COMM_WORLD, islowbuffer, index_slowbuffer, PETSC_COPY_VALUES, &ctx.issb));
852c4762a1bSJed Brown 
853c4762a1bSJed Brown   /* Create a time-stepping object */
8549566063dSJacob Faibussowitsch   PetscCall(TSCreate(comm, &ts));
8559566063dSJacob Faibussowitsch   PetscCall(TSSetDM(ts, da));
8569566063dSJacob Faibussowitsch   PetscCall(TSSetRHSFunction(ts, R, FVRHSFunction_2WaySplit, &ctx));
8579566063dSJacob Faibussowitsch   PetscCall(TSRHSSplitSetIS(ts, "slow", ctx.iss));
8589566063dSJacob Faibussowitsch   PetscCall(TSRHSSplitSetIS(ts, "slowbuffer", ctx.issb));
8599566063dSJacob Faibussowitsch   PetscCall(TSRHSSplitSetIS(ts, "fast", ctx.isf));
8609566063dSJacob Faibussowitsch   PetscCall(TSRHSSplitSetRHSFunction(ts, "slow", NULL, FVRHSFunctionslow_2WaySplit, &ctx));
8619566063dSJacob Faibussowitsch   PetscCall(TSRHSSplitSetRHSFunction(ts, "fast", NULL, FVRHSFunctionfast_2WaySplit, &ctx));
8629566063dSJacob Faibussowitsch   PetscCall(TSRHSSplitSetRHSFunction(ts, "slowbuffer", NULL, FVRHSFunctionslowbuffer_2WaySplit, &ctx));
863c4762a1bSJed Brown 
8649566063dSJacob Faibussowitsch   PetscCall(TSSetType(ts, TSSSP));
8659566063dSJacob Faibussowitsch   /*PetscCall(TSSetType(ts,TSMPRK));*/
8669566063dSJacob Faibussowitsch   PetscCall(TSSetMaxTime(ts, 10));
8679566063dSJacob Faibussowitsch   PetscCall(TSSetExactFinalTime(ts, TS_EXACTFINALTIME_STEPOVER));
868c4762a1bSJed Brown 
869c4762a1bSJed Brown   /* Compute initial conditions and starting time step */
8709566063dSJacob Faibussowitsch   PetscCall(FVSample_2WaySplit(&ctx, da, 0, X0));
8719566063dSJacob Faibussowitsch   PetscCall(FVRHSFunction_2WaySplit(ts, 0, X0, X, (void *)&ctx)); /* Initial function evaluation, only used to determine max speed */
8729566063dSJacob Faibussowitsch   PetscCall(VecCopy(X0, X));                                      /* The function value was not used so we set X=X0 again */
8739566063dSJacob Faibussowitsch   PetscCall(TSSetTimeStep(ts, ctx.cfl / ctx.cfl_idt));
8749566063dSJacob Faibussowitsch   PetscCall(TSSetFromOptions(ts)); /* Take runtime options */
8759566063dSJacob Faibussowitsch   PetscCall(SolutionStatsView(da, X, PETSC_VIEWER_STDOUT_WORLD));
876c4762a1bSJed Brown   {
877c4762a1bSJed Brown     PetscInt           steps;
878c4762a1bSJed Brown     PetscScalar        mass_initial, mass_final, mass_difference, mass_differenceg;
879c4762a1bSJed Brown     const PetscScalar *ptr_X, *ptr_X0;
880c4762a1bSJed Brown     const PetscReal    hs = (ctx.xmax - ctx.xmin) * 3.0 / 4.0 / count_slow;
881c4762a1bSJed Brown     const PetscReal    hf = (ctx.xmax - ctx.xmin) / 4.0 / count_fast;
882c4762a1bSJed Brown 
8839566063dSJacob Faibussowitsch     PetscCall(TSSolve(ts, X));
8849566063dSJacob Faibussowitsch     PetscCall(TSGetSolveTime(ts, &ptime));
8859566063dSJacob Faibussowitsch     PetscCall(TSGetStepNumber(ts, &steps));
886c4762a1bSJed Brown     /* calculate the total mass at initial time and final time */
887c4762a1bSJed Brown     mass_initial = 0.0;
888c4762a1bSJed Brown     mass_final   = 0.0;
8899566063dSJacob Faibussowitsch     PetscCall(DMDAVecGetArrayRead(da, X0, (void *)&ptr_X0));
8909566063dSJacob Faibussowitsch     PetscCall(DMDAVecGetArrayRead(da, X, (void *)&ptr_X));
891c4762a1bSJed Brown     for (i = xs; i < xs + xm; i++) {
892c4762a1bSJed Brown       if (i < ctx.sf || i > ctx.fs - 1) {
893c4762a1bSJed Brown         for (k = 0; k < dof; k++) {
894c4762a1bSJed Brown           mass_initial = mass_initial + hs * ptr_X0[i * dof + k];
895c4762a1bSJed Brown           mass_final   = mass_final + hs * ptr_X[i * dof + k];
896c4762a1bSJed Brown         }
897c4762a1bSJed Brown       } else {
898c4762a1bSJed Brown         for (k = 0; k < dof; k++) {
899c4762a1bSJed Brown           mass_initial = mass_initial + hf * ptr_X0[i * dof + k];
900c4762a1bSJed Brown           mass_final   = mass_final + hf * ptr_X[i * dof + k];
901c4762a1bSJed Brown         }
902c4762a1bSJed Brown       }
903c4762a1bSJed Brown     }
9049566063dSJacob Faibussowitsch     PetscCall(DMDAVecRestoreArrayRead(da, X0, (void *)&ptr_X0));
9059566063dSJacob Faibussowitsch     PetscCall(DMDAVecRestoreArrayRead(da, X, (void *)&ptr_X));
906c4762a1bSJed Brown     mass_difference = mass_final - mass_initial;
907462c564dSBarry Smith     PetscCallMPI(MPIU_Allreduce(&mass_difference, &mass_differenceg, 1, MPIU_SCALAR, MPIU_SUM, comm));
9089566063dSJacob Faibussowitsch     PetscCall(PetscPrintf(comm, "Mass difference %g\n", (double)mass_differenceg));
90963a3b9bcSJacob Faibussowitsch     PetscCall(PetscPrintf(comm, "Final time %g, steps %" PetscInt_FMT "\n", (double)ptime, steps));
910300f1712SStefano Zampini     PetscCall(PetscPrintf(comm, "Maximum allowable stepsize according to CFL %g\n", (double)(1 / ctx.cfl_idt)));
911c4762a1bSJed Brown     if (ctx.exact) {
912c4762a1bSJed Brown       PetscReal nrm1 = 0;
9139566063dSJacob Faibussowitsch       PetscCall(SolutionErrorNorms_2WaySplit(&ctx, da, ptime, X, &nrm1));
9149566063dSJacob Faibussowitsch       PetscCall(PetscPrintf(comm, "Error ||x-x_e||_1 %g\n", (double)nrm1));
915c4762a1bSJed Brown     }
916c4762a1bSJed Brown     if (ctx.simulation) {
917c4762a1bSJed Brown       PetscReal          nrm1 = 0;
918c4762a1bSJed Brown       PetscViewer        fd;
919c4762a1bSJed Brown       char               filename[PETSC_MAX_PATH_LEN] = "binaryoutput";
920c4762a1bSJed Brown       Vec                XR;
921c4762a1bSJed Brown       PetscBool          flg;
922c4762a1bSJed Brown       const PetscScalar *ptr_XR;
9239566063dSJacob Faibussowitsch       PetscCall(PetscOptionsGetString(NULL, NULL, "-f", filename, sizeof(filename), &flg));
9243c633725SBarry Smith       PetscCheck(flg, PETSC_COMM_WORLD, PETSC_ERR_USER, "Must indicate binary file with the -f option");
9259566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryOpen(PETSC_COMM_WORLD, filename, FILE_MODE_READ, &fd));
9269566063dSJacob Faibussowitsch       PetscCall(VecDuplicate(X0, &XR));
9279566063dSJacob Faibussowitsch       PetscCall(VecLoad(XR, fd));
9289566063dSJacob Faibussowitsch       PetscCall(PetscViewerDestroy(&fd));
9299566063dSJacob Faibussowitsch       PetscCall(VecGetArrayRead(X, &ptr_X));
9309566063dSJacob Faibussowitsch       PetscCall(VecGetArrayRead(XR, &ptr_XR));
931c4762a1bSJed Brown       for (i = xs; i < xs + xm; i++) {
932c4762a1bSJed Brown         if (i < ctx.sf || i > ctx.fs - 1)
933c4762a1bSJed Brown           for (k = 0; k < dof; k++) nrm1 = nrm1 + hs * PetscAbs(ptr_X[i * dof + k] - ptr_XR[i * dof + k]);
934c4762a1bSJed Brown         else
935c4762a1bSJed Brown           for (k = 0; k < dof; k++) nrm1 = nrm1 + hf * PetscAbs(ptr_X[i * dof + k] - ptr_XR[i * dof + k]);
936c4762a1bSJed Brown       }
9379566063dSJacob Faibussowitsch       PetscCall(VecRestoreArrayRead(X, &ptr_X));
9389566063dSJacob Faibussowitsch       PetscCall(VecRestoreArrayRead(XR, &ptr_XR));
9399566063dSJacob Faibussowitsch       PetscCall(PetscPrintf(comm, "Error ||x-x_e||_1 %g\n", (double)nrm1));
9409566063dSJacob Faibussowitsch       PetscCall(VecDestroy(&XR));
941c4762a1bSJed Brown     }
942c4762a1bSJed Brown   }
943c4762a1bSJed Brown 
9449566063dSJacob Faibussowitsch   PetscCall(SolutionStatsView(da, X, PETSC_VIEWER_STDOUT_WORLD));
9459566063dSJacob Faibussowitsch   if (draw & 0x1) PetscCall(VecView(X0, PETSC_VIEWER_DRAW_WORLD));
9469566063dSJacob Faibussowitsch   if (draw & 0x2) PetscCall(VecView(X, PETSC_VIEWER_DRAW_WORLD));
947c4762a1bSJed Brown   if (draw & 0x4) {
948c4762a1bSJed Brown     Vec Y;
9499566063dSJacob Faibussowitsch     PetscCall(VecDuplicate(X, &Y));
9509566063dSJacob Faibussowitsch     PetscCall(FVSample_2WaySplit(&ctx, da, ptime, Y));
9519566063dSJacob Faibussowitsch     PetscCall(VecAYPX(Y, -1, X));
9529566063dSJacob Faibussowitsch     PetscCall(VecView(Y, PETSC_VIEWER_DRAW_WORLD));
9539566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&Y));
954c4762a1bSJed Brown   }
955c4762a1bSJed Brown 
956c4762a1bSJed Brown   if (view_final) {
957c4762a1bSJed Brown     PetscViewer viewer;
9589566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIOpen(PETSC_COMM_WORLD, final_fname, &viewer));
9599566063dSJacob Faibussowitsch     PetscCall(PetscViewerPushFormat(viewer, PETSC_VIEWER_ASCII_MATLAB));
9609566063dSJacob Faibussowitsch     PetscCall(VecView(X, viewer));
9619566063dSJacob Faibussowitsch     PetscCall(PetscViewerPopFormat(viewer));
9629566063dSJacob Faibussowitsch     PetscCall(PetscViewerDestroy(&viewer));
963c4762a1bSJed Brown   }
964c4762a1bSJed Brown 
965c4762a1bSJed Brown   /* Clean up */
9669566063dSJacob Faibussowitsch   PetscCall((*ctx.physics2.destroy)(ctx.physics2.user));
9679566063dSJacob Faibussowitsch   for (i = 0; i < ctx.physics2.dof; i++) PetscCall(PetscFree(ctx.physics2.fieldname[i]));
9689566063dSJacob Faibussowitsch   PetscCall(PetscFree4(ctx.R, ctx.Rinv, ctx.cjmpLR, ctx.cslope));
9699566063dSJacob Faibussowitsch   PetscCall(PetscFree3(ctx.uLR, ctx.flux, ctx.speeds));
9709566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&X));
9719566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&X0));
9729566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&R));
9739566063dSJacob Faibussowitsch   PetscCall(DMDestroy(&da));
9749566063dSJacob Faibussowitsch   PetscCall(TSDestroy(&ts));
9759566063dSJacob Faibussowitsch   PetscCall(ISDestroy(&ctx.iss));
9769566063dSJacob Faibussowitsch   PetscCall(ISDestroy(&ctx.isf));
9779566063dSJacob Faibussowitsch   PetscCall(ISDestroy(&ctx.issb));
9789566063dSJacob Faibussowitsch   PetscCall(PetscFree(index_slow));
9799566063dSJacob Faibussowitsch   PetscCall(PetscFree(index_fast));
9809566063dSJacob Faibussowitsch   PetscCall(PetscFree(index_slowbuffer));
9819566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListDestroy(&limiters));
9829566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListDestroy(&physics));
9839566063dSJacob Faibussowitsch   PetscCall(PetscFinalize());
984b122ec5aSJacob Faibussowitsch   return 0;
985c4762a1bSJed Brown }
986c4762a1bSJed Brown 
987c4762a1bSJed Brown /*TEST
988c4762a1bSJed Brown 
989c4762a1bSJed Brown     build:
990f56ea12dSJed Brown       requires: !complex
991c4762a1bSJed Brown       depends: finitevolume1d.c
992c4762a1bSJed Brown 
993c4762a1bSJed Brown     test:
994c4762a1bSJed Brown       suffix: 1
995*188af4bfSBarry Smith       args: -da_grid_x 60 -initial 7 -xmin -1 -xmax 1 -hratio 2 -limit mc -ts_time_step 0.025 -ts_max_steps 24 -ts_type mprk -ts_mprk_type 2a22
996c4762a1bSJed Brown 
997c4762a1bSJed Brown     test:
998c4762a1bSJed Brown       suffix: 2
999*188af4bfSBarry Smith       args: -da_grid_x 60 -initial 7 -xmin -1 -xmax 1 -hratio 2 -limit mc -ts_time_step 0.025 -ts_max_steps 24 -ts_type mprk -ts_mprk_type 2a22 -ts_use_splitrhsfunction 0
1000c4762a1bSJed Brown       output_file: output/ex6_1.out
1001c4762a1bSJed Brown 
1002c4762a1bSJed Brown TEST*/
1003