xref: /petsc/src/ts/impls/implicit/irk/irk.c (revision 1cc06b555e92f8ec64db10330b8bbd830e5bc876)
1d2567f34SHong Zhang /*
2d2567f34SHong Zhang   Code for timestepping with implicit Runge-Kutta method
3d2567f34SHong Zhang 
4d2567f34SHong Zhang   Notes:
5d2567f34SHong Zhang   The general system is written as
6d2567f34SHong Zhang 
7d2567f34SHong Zhang   F(t,U,Udot) = 0
8d2567f34SHong Zhang 
9d2567f34SHong Zhang */
10d2567f34SHong Zhang #include <petsc/private/tsimpl.h> /*I   "petscts.h"   I*/
11d2567f34SHong Zhang #include <petscdm.h>
12d2567f34SHong Zhang #include <petscdt.h>
13d2567f34SHong Zhang 
14d2567f34SHong Zhang static TSIRKType         TSIRKDefault = TSIRKGAUSS;
15d2567f34SHong Zhang static PetscBool         TSIRKRegisterAllCalled;
16d2567f34SHong Zhang static PetscBool         TSIRKPackageInitialized;
17d2567f34SHong Zhang static PetscFunctionList TSIRKList;
18d2567f34SHong Zhang 
19d2567f34SHong Zhang struct _IRKTableau {
20d2567f34SHong Zhang   PetscReal   *A, *b, *c;
21d2567f34SHong Zhang   PetscScalar *A_inv, *A_inv_rowsum, *I_s;
22d2567f34SHong Zhang   PetscReal   *binterp; /* Dense output formula */
23d2567f34SHong Zhang };
24d2567f34SHong Zhang 
25d2567f34SHong Zhang typedef struct _IRKTableau *IRKTableau;
26d2567f34SHong Zhang 
27d2567f34SHong Zhang typedef struct {
28d2567f34SHong Zhang   char        *method_name;
29d2567f34SHong Zhang   PetscInt     order;   /* Classical approximation order of the method */
30d2567f34SHong Zhang   PetscInt     nstages; /* Number of stages */
31d2567f34SHong Zhang   PetscBool    stiffly_accurate;
32d2567f34SHong Zhang   PetscInt     pinterp; /* Interpolation order */
33d2567f34SHong Zhang   IRKTableau   tableau;
34d2567f34SHong Zhang   Vec          U0;    /* Backup vector */
35d2567f34SHong Zhang   Vec          Z;     /* Combined stage vector */
36d2567f34SHong Zhang   Vec         *Y;     /* States computed during the step */
37d2567f34SHong Zhang   Vec          Ydot;  /* Work vector holding time derivatives during residual evaluation */
38d2567f34SHong Zhang   Vec          U;     /* U is used to compute Ydot = shift(Y-U) */
39d2567f34SHong Zhang   Vec         *YdotI; /* Work vectors to hold the residual evaluation */
40d2567f34SHong Zhang   Mat          TJ;    /* KAIJ matrix for the Jacobian of the combined system */
41d2567f34SHong Zhang   PetscScalar *work;  /* Scalar work */
42d2567f34SHong Zhang   TSStepStatus status;
43d2567f34SHong Zhang   PetscBool    rebuild_completion;
44d2567f34SHong Zhang   PetscReal    ccfl;
45d2567f34SHong Zhang } TS_IRK;
46d2567f34SHong Zhang 
47d2567f34SHong Zhang /*@C
48bcf0153eSBarry Smith    TSIRKTableauCreate - create the tableau for `TSIRK` and provide the entries
49d2567f34SHong Zhang 
50d2567f34SHong Zhang    Not Collective
51d2567f34SHong Zhang 
52d2567f34SHong Zhang    Input Parameters:
53d2567f34SHong Zhang +  ts - timestepping context
54d2567f34SHong Zhang .  nstages - number of stages, this is the dimension of the matrices below
55d2567f34SHong Zhang .  A - stage coefficients (dimension nstages*nstages, row-major)
56d2567f34SHong Zhang .  b - step completion table (dimension nstages)
57d2567f34SHong Zhang .  c - abscissa (dimension nstages)
5802024617SSatish Balay .  binterp - coefficients of the interpolation formula (dimension nstages)
59d2567f34SHong Zhang .  A_inv - inverse of A (dimension nstages*nstages, row-major)
60d2567f34SHong Zhang .  A_inv_rowsum - row sum of the inverse of A (dimension nstages)
6102024617SSatish Balay -  I_s - identity matrix (dimension nstages*nstages)
62d2567f34SHong Zhang 
63d2567f34SHong Zhang    Level: advanced
64d2567f34SHong Zhang 
65*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRK`, `TSIRKRegister()`
66d2567f34SHong Zhang @*/
67d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKTableauCreate(TS ts, PetscInt nstages, const PetscReal *A, const PetscReal *b, const PetscReal *c, const PetscReal *binterp, const PetscScalar *A_inv, const PetscScalar *A_inv_rowsum, const PetscScalar *I_s)
68d71ae5a4SJacob Faibussowitsch {
69d2567f34SHong Zhang   TS_IRK    *irk = (TS_IRK *)ts->data;
70d2567f34SHong Zhang   IRKTableau tab = irk->tableau;
71d2567f34SHong Zhang 
72d2567f34SHong Zhang   PetscFunctionBegin;
73d2567f34SHong Zhang   irk->order = nstages;
749566063dSJacob Faibussowitsch   PetscCall(PetscMalloc3(PetscSqr(nstages), &tab->A, PetscSqr(nstages), &tab->A_inv, PetscSqr(nstages), &tab->I_s));
759566063dSJacob Faibussowitsch   PetscCall(PetscMalloc4(nstages, &tab->b, nstages, &tab->c, nstages, &tab->binterp, nstages, &tab->A_inv_rowsum));
769566063dSJacob Faibussowitsch   PetscCall(PetscArraycpy(tab->A, A, PetscSqr(nstages)));
779566063dSJacob Faibussowitsch   PetscCall(PetscArraycpy(tab->b, b, nstages));
789566063dSJacob Faibussowitsch   PetscCall(PetscArraycpy(tab->c, c, nstages));
79d2567f34SHong Zhang   /* optional coefficient arrays */
801baa6e33SBarry Smith   if (binterp) PetscCall(PetscArraycpy(tab->binterp, binterp, nstages));
811baa6e33SBarry Smith   if (A_inv) PetscCall(PetscArraycpy(tab->A_inv, A_inv, PetscSqr(nstages)));
821baa6e33SBarry Smith   if (A_inv_rowsum) PetscCall(PetscArraycpy(tab->A_inv_rowsum, A_inv_rowsum, nstages));
831baa6e33SBarry Smith   if (I_s) PetscCall(PetscArraycpy(tab->I_s, I_s, PetscSqr(nstages)));
843ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
85d2567f34SHong Zhang }
86d2567f34SHong Zhang 
87d2567f34SHong Zhang /* Arrays should be freed with PetscFree3(A,b,c) */
88d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKCreate_Gauss(TS ts)
89d71ae5a4SJacob Faibussowitsch {
90d2567f34SHong Zhang   PetscInt     nstages;
91d2567f34SHong Zhang   PetscReal   *gauss_A_real, *gauss_b, *b, *gauss_c;
92d2567f34SHong Zhang   PetscScalar *gauss_A, *gauss_A_inv, *gauss_A_inv_rowsum, *I_s;
93d2567f34SHong Zhang   PetscScalar *G0, *G1;
94d2567f34SHong Zhang   PetscInt     i, j;
95d2567f34SHong Zhang   Mat          G0mat, G1mat, Amat;
96d2567f34SHong Zhang 
97d2567f34SHong Zhang   PetscFunctionBegin;
989566063dSJacob Faibussowitsch   PetscCall(TSIRKGetNumStages(ts, &nstages));
999566063dSJacob Faibussowitsch   PetscCall(PetscMalloc3(PetscSqr(nstages), &gauss_A_real, nstages, &gauss_b, nstages, &gauss_c));
1009566063dSJacob Faibussowitsch   PetscCall(PetscMalloc4(PetscSqr(nstages), &gauss_A, PetscSqr(nstages), &gauss_A_inv, nstages, &gauss_A_inv_rowsum, PetscSqr(nstages), &I_s));
1019566063dSJacob Faibussowitsch   PetscCall(PetscMalloc3(nstages, &b, PetscSqr(nstages), &G0, PetscSqr(nstages), &G1));
1029566063dSJacob Faibussowitsch   PetscCall(PetscDTGaussQuadrature(nstages, 0., 1., gauss_c, b));
103d2567f34SHong Zhang   for (i = 0; i < nstages; i++) gauss_b[i] = b[i]; /* copy to possibly-complex array */
104d2567f34SHong Zhang 
105d2567f34SHong Zhang   /* A^T = G0^{-1} G1 */
106d2567f34SHong Zhang   for (i = 0; i < nstages; i++) {
107d2567f34SHong Zhang     for (j = 0; j < nstages; j++) {
108d2567f34SHong Zhang       G0[i * nstages + j] = PetscPowRealInt(gauss_c[i], j);
109d2567f34SHong Zhang       G1[i * nstages + j] = PetscPowRealInt(gauss_c[i], j + 1) / (j + 1);
110d2567f34SHong Zhang     }
111d2567f34SHong Zhang   }
112d2567f34SHong Zhang   /* The arrays above are row-aligned, but we create dense matrices as the transpose */
1139566063dSJacob Faibussowitsch   PetscCall(MatCreateSeqDense(PETSC_COMM_SELF, nstages, nstages, G0, &G0mat));
1149566063dSJacob Faibussowitsch   PetscCall(MatCreateSeqDense(PETSC_COMM_SELF, nstages, nstages, G1, &G1mat));
1159566063dSJacob Faibussowitsch   PetscCall(MatCreateSeqDense(PETSC_COMM_SELF, nstages, nstages, gauss_A, &Amat));
1169566063dSJacob Faibussowitsch   PetscCall(MatLUFactor(G0mat, NULL, NULL, NULL));
1179566063dSJacob Faibussowitsch   PetscCall(MatMatSolve(G0mat, G1mat, Amat));
1189566063dSJacob Faibussowitsch   PetscCall(MatTranspose(Amat, MAT_INPLACE_MATRIX, &Amat));
119d2567f34SHong Zhang   for (i = 0; i < nstages; i++)
1209371c9d4SSatish Balay     for (j = 0; j < nstages; j++) gauss_A_real[i * nstages + j] = PetscRealPart(gauss_A[i * nstages + j]);
121d2567f34SHong Zhang 
1229566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&G0mat));
1239566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&G1mat));
1249566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&Amat));
1259566063dSJacob Faibussowitsch   PetscCall(PetscFree3(b, G0, G1));
126d2567f34SHong Zhang 
127d2567f34SHong Zhang   { /* Invert A */
128d2567f34SHong Zhang     /* PETSc does not provide a routine to calculate the inverse of a general matrix.
129d2567f34SHong Zhang      * To get the inverse of A, we form a sequential BAIJ matrix from it, consisting of a single block with block size
130d2567f34SHong Zhang      * equal to the dimension of A, and then use MatInvertBlockDiagonal(). */
131d2567f34SHong Zhang     Mat                A_baij;
132d2567f34SHong Zhang     PetscInt           idxm[1] = {0}, idxn[1] = {0};
133d2567f34SHong Zhang     const PetscScalar *A_inv;
134d2567f34SHong Zhang 
1359566063dSJacob Faibussowitsch     PetscCall(MatCreateSeqBAIJ(PETSC_COMM_SELF, nstages, nstages, nstages, 1, NULL, &A_baij));
1369566063dSJacob Faibussowitsch     PetscCall(MatSetOption(A_baij, MAT_ROW_ORIENTED, PETSC_FALSE));
1379566063dSJacob Faibussowitsch     PetscCall(MatSetValuesBlocked(A_baij, 1, idxm, 1, idxn, gauss_A, INSERT_VALUES));
1389566063dSJacob Faibussowitsch     PetscCall(MatAssemblyBegin(A_baij, MAT_FINAL_ASSEMBLY));
1399566063dSJacob Faibussowitsch     PetscCall(MatAssemblyEnd(A_baij, MAT_FINAL_ASSEMBLY));
1409566063dSJacob Faibussowitsch     PetscCall(MatInvertBlockDiagonal(A_baij, &A_inv));
1419566063dSJacob Faibussowitsch     PetscCall(PetscMemcpy(gauss_A_inv, A_inv, nstages * nstages * sizeof(PetscScalar)));
1429566063dSJacob Faibussowitsch     PetscCall(MatDestroy(&A_baij));
143d2567f34SHong Zhang   }
144d2567f34SHong Zhang 
145d2567f34SHong Zhang   /* Compute row sums A_inv_rowsum and identity I_s */
146d2567f34SHong Zhang   for (i = 0; i < nstages; i++) {
147d2567f34SHong Zhang     gauss_A_inv_rowsum[i] = 0;
148d2567f34SHong Zhang     for (j = 0; j < nstages; j++) {
149d2567f34SHong Zhang       gauss_A_inv_rowsum[i] += gauss_A_inv[i + nstages * j];
150d2567f34SHong Zhang       I_s[i + nstages * j] = 1. * (i == j);
151d2567f34SHong Zhang     }
152d2567f34SHong Zhang   }
1539566063dSJacob Faibussowitsch   PetscCall(TSIRKTableauCreate(ts, nstages, gauss_A_real, gauss_b, gauss_c, NULL, gauss_A_inv, gauss_A_inv_rowsum, I_s));
1549566063dSJacob Faibussowitsch   PetscCall(PetscFree3(gauss_A_real, gauss_b, gauss_c));
1559566063dSJacob Faibussowitsch   PetscCall(PetscFree4(gauss_A, gauss_A_inv, gauss_A_inv_rowsum, I_s));
1563ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
157d2567f34SHong Zhang }
158d2567f34SHong Zhang 
159d2567f34SHong Zhang /*@C
160bcf0153eSBarry Smith    TSIRKRegister -  adds a `TSIRK` implementation
161d2567f34SHong Zhang 
162d2567f34SHong Zhang    Not Collective
163d2567f34SHong Zhang 
164d2567f34SHong Zhang    Input Parameters:
165d2567f34SHong Zhang +  sname - name of user-defined IRK scheme
166d2567f34SHong Zhang -  function - function to create method context
167d2567f34SHong Zhang 
168bcf0153eSBarry Smith    Level: advanced
169bcf0153eSBarry Smith 
170bcf0153eSBarry Smith    Note:
171bcf0153eSBarry Smith    `TSIRKRegister()` may be called multiple times to add several user-defined families.
172d2567f34SHong Zhang 
173d2567f34SHong Zhang    Sample usage:
174d2567f34SHong Zhang .vb
175d2567f34SHong Zhang    TSIRKRegister("my_scheme", MySchemeCreate);
176d2567f34SHong Zhang .ve
177d2567f34SHong Zhang 
178d2567f34SHong Zhang    Then, your scheme can be chosen with the procedural interface via
179d2567f34SHong Zhang $     TSIRKSetType(ts, "my_scheme")
180d2567f34SHong Zhang    or at runtime via the option
181d2567f34SHong Zhang $     -ts_irk_type my_scheme
182d2567f34SHong Zhang 
183*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRK`, `TSIRKRegisterAll()`
184d2567f34SHong Zhang @*/
185d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKRegister(const char sname[], PetscErrorCode (*function)(TS))
186d71ae5a4SJacob Faibussowitsch {
187d2567f34SHong Zhang   PetscFunctionBegin;
1889566063dSJacob Faibussowitsch   PetscCall(TSIRKInitializePackage());
1899566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListAdd(&TSIRKList, sname, function));
1903ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
191d2567f34SHong Zhang }
192d2567f34SHong Zhang 
193d2567f34SHong Zhang /*@C
194bcf0153eSBarry Smith   TSIRKRegisterAll - Registers all of the implicit Runge-Kutta methods in `TSIRK`
195d2567f34SHong Zhang 
196d2567f34SHong Zhang   Not Collective, but should be called by all processes which will need the schemes to be registered
197d2567f34SHong Zhang 
198d2567f34SHong Zhang   Level: advanced
199d2567f34SHong Zhang 
200*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRK`, `TSIRKRegisterDestroy()`
201d2567f34SHong Zhang @*/
202d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKRegisterAll(void)
203d71ae5a4SJacob Faibussowitsch {
204d2567f34SHong Zhang   PetscFunctionBegin;
2053ba16761SJacob Faibussowitsch   if (TSIRKRegisterAllCalled) PetscFunctionReturn(PETSC_SUCCESS);
206d2567f34SHong Zhang   TSIRKRegisterAllCalled = PETSC_TRUE;
207d2567f34SHong Zhang 
2089566063dSJacob Faibussowitsch   PetscCall(TSIRKRegister(TSIRKGAUSS, TSIRKCreate_Gauss));
2093ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
210d2567f34SHong Zhang }
211d2567f34SHong Zhang 
212d2567f34SHong Zhang /*@C
213bcf0153eSBarry Smith    TSIRKRegisterDestroy - Frees the list of schemes that were registered by `TSIRKRegister()`.
214d2567f34SHong Zhang 
215d2567f34SHong Zhang    Not Collective
216d2567f34SHong Zhang 
217d2567f34SHong Zhang    Level: advanced
218d2567f34SHong Zhang 
219*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRK`, `TSIRKRegister()`, `TSIRKRegisterAll()`
220d2567f34SHong Zhang @*/
221d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKRegisterDestroy(void)
222d71ae5a4SJacob Faibussowitsch {
223d2567f34SHong Zhang   PetscFunctionBegin;
224d2567f34SHong Zhang   TSIRKRegisterAllCalled = PETSC_FALSE;
2253ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
226d2567f34SHong Zhang }
227d2567f34SHong Zhang 
228d2567f34SHong Zhang /*@C
229bcf0153eSBarry Smith   TSIRKInitializePackage - This function initializes everything in the `TSIRK` package. It is called
230bcf0153eSBarry Smith   from `TSInitializePackage()`.
231d2567f34SHong Zhang 
232d2567f34SHong Zhang   Level: developer
233d2567f34SHong Zhang 
234*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRK`, `PetscInitialize()`, `TSIRKFinalizePackage()`, `TSInitializePackage()`
235d2567f34SHong Zhang @*/
236d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKInitializePackage(void)
237d71ae5a4SJacob Faibussowitsch {
238d2567f34SHong Zhang   PetscFunctionBegin;
2393ba16761SJacob Faibussowitsch   if (TSIRKPackageInitialized) PetscFunctionReturn(PETSC_SUCCESS);
240d2567f34SHong Zhang   TSIRKPackageInitialized = PETSC_TRUE;
2419566063dSJacob Faibussowitsch   PetscCall(TSIRKRegisterAll());
2429566063dSJacob Faibussowitsch   PetscCall(PetscRegisterFinalize(TSIRKFinalizePackage));
2433ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
244d2567f34SHong Zhang }
245d2567f34SHong Zhang 
246d2567f34SHong Zhang /*@C
247bcf0153eSBarry Smith   TSIRKFinalizePackage - This function destroys everything in the `TSIRK` package. It is
248bcf0153eSBarry Smith   called from `PetscFinalize()`.
249d2567f34SHong Zhang 
250d2567f34SHong Zhang   Level: developer
251d2567f34SHong Zhang 
252*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRK`, `PetscFinalize()`, `TSIRKFinalizePackage()`, `TSInitializePackage()`
253d2567f34SHong Zhang @*/
254d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKFinalizePackage(void)
255d71ae5a4SJacob Faibussowitsch {
256d2567f34SHong Zhang   PetscFunctionBegin;
2579566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListDestroy(&TSIRKList));
258d2567f34SHong Zhang   TSIRKPackageInitialized = PETSC_FALSE;
2593ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
260d2567f34SHong Zhang }
261d2567f34SHong Zhang 
262d2567f34SHong Zhang /*
263d2567f34SHong Zhang  This function can be called before or after ts->vec_sol has been updated.
264d2567f34SHong Zhang */
265d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSEvaluateStep_IRK(TS ts, PetscInt order, Vec U, PetscBool *done)
266d71ae5a4SJacob Faibussowitsch {
267d2567f34SHong Zhang   TS_IRK      *irk   = (TS_IRK *)ts->data;
268d2567f34SHong Zhang   IRKTableau   tab   = irk->tableau;
269d2567f34SHong Zhang   Vec         *YdotI = irk->YdotI;
270d2567f34SHong Zhang   PetscScalar *w     = irk->work;
271d2567f34SHong Zhang   PetscReal    h;
272d2567f34SHong Zhang   PetscInt     j;
273d2567f34SHong Zhang 
274d2567f34SHong Zhang   PetscFunctionBegin;
275d2567f34SHong Zhang   switch (irk->status) {
276d2567f34SHong Zhang   case TS_STEP_INCOMPLETE:
277d71ae5a4SJacob Faibussowitsch   case TS_STEP_PENDING:
278d71ae5a4SJacob Faibussowitsch     h = ts->time_step;
279d71ae5a4SJacob Faibussowitsch     break;
280d71ae5a4SJacob Faibussowitsch   case TS_STEP_COMPLETE:
281d71ae5a4SJacob Faibussowitsch     h = ts->ptime - ts->ptime_prev;
282d71ae5a4SJacob Faibussowitsch     break;
283d71ae5a4SJacob Faibussowitsch   default:
284d71ae5a4SJacob Faibussowitsch     SETERRQ(PetscObjectComm((PetscObject)ts), PETSC_ERR_PLIB, "Invalid TSStepStatus");
285d2567f34SHong Zhang   }
286d2567f34SHong Zhang 
2879566063dSJacob Faibussowitsch   PetscCall(VecCopy(ts->vec_sol, U));
288d2567f34SHong Zhang   for (j = 0; j < irk->nstages; j++) w[j] = h * tab->b[j];
2899566063dSJacob Faibussowitsch   PetscCall(VecMAXPY(U, irk->nstages, w, YdotI));
2903ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
291d2567f34SHong Zhang }
292d2567f34SHong Zhang 
293d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSRollBack_IRK(TS ts)
294d71ae5a4SJacob Faibussowitsch {
295d2567f34SHong Zhang   TS_IRK *irk = (TS_IRK *)ts->data;
296d2567f34SHong Zhang 
297d2567f34SHong Zhang   PetscFunctionBegin;
2989566063dSJacob Faibussowitsch   PetscCall(VecCopy(irk->U0, ts->vec_sol));
2993ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
300d2567f34SHong Zhang }
301d2567f34SHong Zhang 
302d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSStep_IRK(TS ts)
303d71ae5a4SJacob Faibussowitsch {
304d2567f34SHong Zhang   TS_IRK        *irk   = (TS_IRK *)ts->data;
305d2567f34SHong Zhang   IRKTableau     tab   = irk->tableau;
306d2567f34SHong Zhang   PetscScalar   *A_inv = tab->A_inv, *A_inv_rowsum = tab->A_inv_rowsum;
307d2567f34SHong Zhang   const PetscInt nstages = irk->nstages;
308d2567f34SHong Zhang   SNES           snes;
309d2567f34SHong Zhang   PetscInt       i, j, its, lits, bs;
310d2567f34SHong Zhang   TSAdapt        adapt;
311d2567f34SHong Zhang   PetscInt       rejections     = 0;
312d2567f34SHong Zhang   PetscBool      accept         = PETSC_TRUE;
313d2567f34SHong Zhang   PetscReal      next_time_step = ts->time_step;
314d2567f34SHong Zhang 
315d2567f34SHong Zhang   PetscFunctionBegin;
31648a46eb9SPierre Jolivet   if (!ts->steprollback) PetscCall(VecCopy(ts->vec_sol, irk->U0));
3179566063dSJacob Faibussowitsch   PetscCall(VecGetBlockSize(ts->vec_sol, &bs));
31848a46eb9SPierre Jolivet   for (i = 0; i < nstages; i++) PetscCall(VecStrideScatter(ts->vec_sol, i * bs, irk->Z, INSERT_VALUES));
319d2567f34SHong Zhang 
320d2567f34SHong Zhang   irk->status = TS_STEP_INCOMPLETE;
321d2567f34SHong Zhang   while (!ts->reason && irk->status != TS_STEP_COMPLETE) {
3229566063dSJacob Faibussowitsch     PetscCall(VecCopy(ts->vec_sol, irk->U));
3239566063dSJacob Faibussowitsch     PetscCall(TSGetSNES(ts, &snes));
3249566063dSJacob Faibussowitsch     PetscCall(SNESSolve(snes, NULL, irk->Z));
3259566063dSJacob Faibussowitsch     PetscCall(SNESGetIterationNumber(snes, &its));
3269566063dSJacob Faibussowitsch     PetscCall(SNESGetLinearSolveIterations(snes, &lits));
3279371c9d4SSatish Balay     ts->snes_its += its;
3289371c9d4SSatish Balay     ts->ksp_its += lits;
3299566063dSJacob Faibussowitsch     PetscCall(VecStrideGatherAll(irk->Z, irk->Y, INSERT_VALUES));
330d2567f34SHong Zhang     for (i = 0; i < nstages; i++) {
3319566063dSJacob Faibussowitsch       PetscCall(VecZeroEntries(irk->YdotI[i]));
33248a46eb9SPierre Jolivet       for (j = 0; j < nstages; j++) PetscCall(VecAXPY(irk->YdotI[i], A_inv[i + j * nstages] / ts->time_step, irk->Y[j]));
3339566063dSJacob Faibussowitsch       PetscCall(VecAXPY(irk->YdotI[i], -A_inv_rowsum[i] / ts->time_step, irk->U));
334d2567f34SHong Zhang     }
335d2567f34SHong Zhang     irk->status = TS_STEP_INCOMPLETE;
3369566063dSJacob Faibussowitsch     PetscCall(TSEvaluateStep_IRK(ts, irk->order, ts->vec_sol, NULL));
337d2567f34SHong Zhang     irk->status = TS_STEP_PENDING;
3389566063dSJacob Faibussowitsch     PetscCall(TSGetAdapt(ts, &adapt));
3399566063dSJacob Faibussowitsch     PetscCall(TSAdaptChoose(adapt, ts, ts->time_step, NULL, &next_time_step, &accept));
340d2567f34SHong Zhang     irk->status = accept ? TS_STEP_COMPLETE : TS_STEP_INCOMPLETE;
341d2567f34SHong Zhang     if (!accept) {
3429566063dSJacob Faibussowitsch       PetscCall(TSRollBack_IRK(ts));
343d2567f34SHong Zhang       ts->time_step = next_time_step;
344d2567f34SHong Zhang       goto reject_step;
345d2567f34SHong Zhang     }
346d2567f34SHong Zhang 
347d2567f34SHong Zhang     ts->ptime += ts->time_step;
348d2567f34SHong Zhang     ts->time_step = next_time_step;
349d2567f34SHong Zhang     break;
350d2567f34SHong Zhang   reject_step:
3519371c9d4SSatish Balay     ts->reject++;
3529371c9d4SSatish Balay     accept = PETSC_FALSE;
353d2567f34SHong Zhang     if (!ts->reason && ++rejections > ts->max_reject && ts->max_reject >= 0) {
354d2567f34SHong Zhang       ts->reason = TS_DIVERGED_STEP_REJECTED;
35563a3b9bcSJacob Faibussowitsch       PetscCall(PetscInfo(ts, "Step=%" PetscInt_FMT ", step rejections %" PetscInt_FMT " greater than current TS allowed, stopping solve\n", ts->steps, rejections));
356d2567f34SHong Zhang     }
357d2567f34SHong Zhang   }
3583ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
359d2567f34SHong Zhang }
360d2567f34SHong Zhang 
361d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSInterpolate_IRK(TS ts, PetscReal itime, Vec U)
362d71ae5a4SJacob Faibussowitsch {
363d2567f34SHong Zhang   TS_IRK          *irk     = (TS_IRK *)ts->data;
364d2567f34SHong Zhang   PetscInt         nstages = irk->nstages, pinterp = irk->pinterp, i, j;
365d2567f34SHong Zhang   PetscReal        h;
366d2567f34SHong Zhang   PetscReal        tt, t;
367d2567f34SHong Zhang   PetscScalar     *bt;
368d2567f34SHong Zhang   const PetscReal *B = irk->tableau->binterp;
369d2567f34SHong Zhang 
370d2567f34SHong Zhang   PetscFunctionBegin;
3713c633725SBarry Smith   PetscCheck(B, PetscObjectComm((PetscObject)ts), PETSC_ERR_SUP, "TSIRK %s does not have an interpolation formula", irk->method_name);
372d2567f34SHong Zhang   switch (irk->status) {
373d2567f34SHong Zhang   case TS_STEP_INCOMPLETE:
374d2567f34SHong Zhang   case TS_STEP_PENDING:
375d2567f34SHong Zhang     h = ts->time_step;
376d2567f34SHong Zhang     t = (itime - ts->ptime) / h;
377d2567f34SHong Zhang     break;
378d2567f34SHong Zhang   case TS_STEP_COMPLETE:
379d2567f34SHong Zhang     h = ts->ptime - ts->ptime_prev;
380d2567f34SHong Zhang     t = (itime - ts->ptime) / h + 1; /* In the interval [0,1] */
381d2567f34SHong Zhang     break;
382d71ae5a4SJacob Faibussowitsch   default:
383d71ae5a4SJacob Faibussowitsch     SETERRQ(PetscObjectComm((PetscObject)ts), PETSC_ERR_PLIB, "Invalid TSStepStatus");
384d2567f34SHong Zhang   }
3859566063dSJacob Faibussowitsch   PetscCall(PetscMalloc1(nstages, &bt));
386d2567f34SHong Zhang   for (i = 0; i < nstages; i++) bt[i] = 0;
387d2567f34SHong Zhang   for (j = 0, tt = t; j < pinterp; j++, tt *= t) {
388ad540459SPierre Jolivet     for (i = 0; i < nstages; i++) bt[i] += h * B[i * pinterp + j] * tt;
389d2567f34SHong Zhang   }
3909566063dSJacob Faibussowitsch   PetscCall(VecMAXPY(U, nstages, bt, irk->YdotI));
3913ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
392d2567f34SHong Zhang }
393d2567f34SHong Zhang 
394d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKTableauReset(TS ts)
395d71ae5a4SJacob Faibussowitsch {
396d2567f34SHong Zhang   TS_IRK    *irk = (TS_IRK *)ts->data;
397d2567f34SHong Zhang   IRKTableau tab = irk->tableau;
398d2567f34SHong Zhang 
399d2567f34SHong Zhang   PetscFunctionBegin;
4003ba16761SJacob Faibussowitsch   if (!tab) PetscFunctionReturn(PETSC_SUCCESS);
4019566063dSJacob Faibussowitsch   PetscCall(PetscFree3(tab->A, tab->A_inv, tab->I_s));
4029566063dSJacob Faibussowitsch   PetscCall(PetscFree4(tab->b, tab->c, tab->binterp, tab->A_inv_rowsum));
4033ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
404d2567f34SHong Zhang }
405d2567f34SHong Zhang 
406d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSReset_IRK(TS ts)
407d71ae5a4SJacob Faibussowitsch {
408d2567f34SHong Zhang   TS_IRK *irk = (TS_IRK *)ts->data;
409d2567f34SHong Zhang 
410d2567f34SHong Zhang   PetscFunctionBegin;
4119566063dSJacob Faibussowitsch   PetscCall(TSIRKTableauReset(ts));
4121baa6e33SBarry Smith   if (irk->tableau) PetscCall(PetscFree(irk->tableau));
4131baa6e33SBarry Smith   if (irk->method_name) PetscCall(PetscFree(irk->method_name));
4141baa6e33SBarry Smith   if (irk->work) PetscCall(PetscFree(irk->work));
4159566063dSJacob Faibussowitsch   PetscCall(VecDestroyVecs(irk->nstages, &irk->Y));
4169566063dSJacob Faibussowitsch   PetscCall(VecDestroyVecs(irk->nstages, &irk->YdotI));
4179566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&irk->Ydot));
4189566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&irk->Z));
4199566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&irk->U));
4209566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&irk->U0));
4219566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&irk->TJ));
4223ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
423d2567f34SHong Zhang }
424d2567f34SHong Zhang 
425d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKGetVecs(TS ts, DM dm, Vec *U)
426d71ae5a4SJacob Faibussowitsch {
427d2567f34SHong Zhang   TS_IRK *irk = (TS_IRK *)ts->data;
428d2567f34SHong Zhang 
429d2567f34SHong Zhang   PetscFunctionBegin;
430d2567f34SHong Zhang   if (U) {
431d2567f34SHong Zhang     if (dm && dm != ts->dm) {
4329566063dSJacob Faibussowitsch       PetscCall(DMGetNamedGlobalVector(dm, "TSIRK_U", U));
433d2567f34SHong Zhang     } else *U = irk->U;
434d2567f34SHong Zhang   }
4353ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
436d2567f34SHong Zhang }
437d2567f34SHong Zhang 
438d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKRestoreVecs(TS ts, DM dm, Vec *U)
439d71ae5a4SJacob Faibussowitsch {
440d2567f34SHong Zhang   PetscFunctionBegin;
441d2567f34SHong Zhang   if (U) {
44248a46eb9SPierre Jolivet     if (dm && dm != ts->dm) PetscCall(DMRestoreNamedGlobalVector(dm, "TSIRK_U", U));
443d2567f34SHong Zhang   }
4443ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
445d2567f34SHong Zhang }
446d2567f34SHong Zhang 
447d2567f34SHong Zhang /*
448d2567f34SHong Zhang   This defines the nonlinear equations that is to be solved with SNES
449d2567f34SHong Zhang     G[e\otimes t + C*dt, Z, Zdot] = 0
450d2567f34SHong Zhang     Zdot = (In \otimes S)*Z - (In \otimes Se) U
451d2567f34SHong Zhang   where S = 1/(dt*A)
452d2567f34SHong Zhang */
453d71ae5a4SJacob Faibussowitsch static PetscErrorCode SNESTSFormFunction_IRK(SNES snes, Vec ZC, Vec FC, TS ts)
454d71ae5a4SJacob Faibussowitsch {
455d2567f34SHong Zhang   TS_IRK            *irk     = (TS_IRK *)ts->data;
456d2567f34SHong Zhang   IRKTableau         tab     = irk->tableau;
457d2567f34SHong Zhang   const PetscInt     nstages = irk->nstages;
458d2567f34SHong Zhang   const PetscReal   *c       = tab->c;
459d2567f34SHong Zhang   const PetscScalar *A_inv = tab->A_inv, *A_inv_rowsum = tab->A_inv_rowsum;
460d2567f34SHong Zhang   DM                 dm, dmsave;
461d2567f34SHong Zhang   Vec                U, *YdotI = irk->YdotI, Ydot = irk->Ydot, *Y = irk->Y;
462d2567f34SHong Zhang   PetscReal          h = ts->time_step;
463d2567f34SHong Zhang   PetscInt           i, j;
464d2567f34SHong Zhang 
465d2567f34SHong Zhang   PetscFunctionBegin;
4669566063dSJacob Faibussowitsch   PetscCall(SNESGetDM(snes, &dm));
4679566063dSJacob Faibussowitsch   PetscCall(TSIRKGetVecs(ts, dm, &U));
4689566063dSJacob Faibussowitsch   PetscCall(VecStrideGatherAll(ZC, Y, INSERT_VALUES));
469d2567f34SHong Zhang   dmsave = ts->dm;
470d2567f34SHong Zhang   ts->dm = dm;
471d2567f34SHong Zhang   for (i = 0; i < nstages; i++) {
4729566063dSJacob Faibussowitsch     PetscCall(VecZeroEntries(Ydot));
47348a46eb9SPierre Jolivet     for (j = 0; j < nstages; j++) PetscCall(VecAXPY(Ydot, A_inv[j * nstages + i] / h, Y[j]));
4749566063dSJacob Faibussowitsch     PetscCall(VecAXPY(Ydot, -A_inv_rowsum[i] / h, U)); /* Ydot = (S \otimes In)*Z - (Se \otimes In) U */
4759566063dSJacob Faibussowitsch     PetscCall(TSComputeIFunction(ts, ts->ptime + ts->time_step * c[i], Y[i], Ydot, YdotI[i], PETSC_FALSE));
476d2567f34SHong Zhang   }
4779566063dSJacob Faibussowitsch   PetscCall(VecStrideScatterAll(YdotI, FC, INSERT_VALUES));
478d2567f34SHong Zhang   ts->dm = dmsave;
4799566063dSJacob Faibussowitsch   PetscCall(TSIRKRestoreVecs(ts, dm, &U));
4803ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
481d2567f34SHong Zhang }
482d2567f34SHong Zhang 
483d2567f34SHong Zhang /*
484d2567f34SHong Zhang    For explicit ODE, the Jacobian is
485d2567f34SHong Zhang      JC = I_n \otimes S - J \otimes I_s
486d2567f34SHong Zhang    For DAE, the Jacobian is
487d2567f34SHong Zhang      JC = M_n \otimes S - J \otimes I_s
488d2567f34SHong Zhang */
489d71ae5a4SJacob Faibussowitsch static PetscErrorCode SNESTSFormJacobian_IRK(SNES snes, Vec ZC, Mat JC, Mat JCpre, TS ts)
490d71ae5a4SJacob Faibussowitsch {
491d2567f34SHong Zhang   TS_IRK          *irk     = (TS_IRK *)ts->data;
492d2567f34SHong Zhang   IRKTableau       tab     = irk->tableau;
493d2567f34SHong Zhang   const PetscInt   nstages = irk->nstages;
494d2567f34SHong Zhang   const PetscReal *c       = tab->c;
495d2567f34SHong Zhang   DM               dm, dmsave;
496d2567f34SHong Zhang   Vec             *Y = irk->Y, Ydot = irk->Ydot;
497d2567f34SHong Zhang   Mat              J;
498d2567f34SHong Zhang   PetscScalar     *S;
499d2567f34SHong Zhang   PetscInt         i, j, bs;
500d2567f34SHong Zhang 
501d2567f34SHong Zhang   PetscFunctionBegin;
5029566063dSJacob Faibussowitsch   PetscCall(SNESGetDM(snes, &dm));
503d2567f34SHong Zhang   /* irk->Ydot has already been computed in SNESTSFormFunction_IRK (SNES guarantees this) */
504d2567f34SHong Zhang   dmsave = ts->dm;
505d2567f34SHong Zhang   ts->dm = dm;
5069566063dSJacob Faibussowitsch   PetscCall(VecGetBlockSize(Y[nstages - 1], &bs));
507d2567f34SHong Zhang   if (ts->equation_type <= TS_EQ_ODE_EXPLICIT) { /* Support explicit formulas only */
5089566063dSJacob Faibussowitsch     PetscCall(VecStrideGather(ZC, (nstages - 1) * bs, Y[nstages - 1], INSERT_VALUES));
5099566063dSJacob Faibussowitsch     PetscCall(MatKAIJGetAIJ(JC, &J));
5109566063dSJacob Faibussowitsch     PetscCall(TSComputeIJacobian(ts, ts->ptime + ts->time_step * c[nstages - 1], Y[nstages - 1], Ydot, 0, J, J, PETSC_FALSE));
5119566063dSJacob Faibussowitsch     PetscCall(MatKAIJGetS(JC, NULL, NULL, &S));
512d2567f34SHong Zhang     for (i = 0; i < nstages; i++)
5139371c9d4SSatish Balay       for (j = 0; j < nstages; j++) S[i + nstages * j] = tab->A_inv[i + nstages * j] / ts->time_step;
5149566063dSJacob Faibussowitsch     PetscCall(MatKAIJRestoreS(JC, &S));
51598921bdaSJacob Faibussowitsch   } else SETERRQ(PetscObjectComm((PetscObject)ts), PETSC_ERR_SUP, "TSIRK %s does not support implicit formula", irk->method_name); /* TODO: need the mass matrix for DAE  */
516d2567f34SHong Zhang   ts->dm = dmsave;
5173ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
518d2567f34SHong Zhang }
519d2567f34SHong Zhang 
520d71ae5a4SJacob Faibussowitsch static PetscErrorCode DMCoarsenHook_TSIRK(DM fine, DM coarse, void *ctx)
521d71ae5a4SJacob Faibussowitsch {
522d2567f34SHong Zhang   PetscFunctionBegin;
5233ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
524d2567f34SHong Zhang }
525d2567f34SHong Zhang 
526d71ae5a4SJacob Faibussowitsch static PetscErrorCode DMRestrictHook_TSIRK(DM fine, Mat restrct, Vec rscale, Mat inject, DM coarse, void *ctx)
527d71ae5a4SJacob Faibussowitsch {
528d2567f34SHong Zhang   TS  ts = (TS)ctx;
529d2567f34SHong Zhang   Vec U, U_c;
530d2567f34SHong Zhang 
531d2567f34SHong Zhang   PetscFunctionBegin;
5329566063dSJacob Faibussowitsch   PetscCall(TSIRKGetVecs(ts, fine, &U));
5339566063dSJacob Faibussowitsch   PetscCall(TSIRKGetVecs(ts, coarse, &U_c));
5349566063dSJacob Faibussowitsch   PetscCall(MatRestrict(restrct, U, U_c));
5359566063dSJacob Faibussowitsch   PetscCall(VecPointwiseMult(U_c, rscale, U_c));
5369566063dSJacob Faibussowitsch   PetscCall(TSIRKRestoreVecs(ts, fine, &U));
5379566063dSJacob Faibussowitsch   PetscCall(TSIRKRestoreVecs(ts, coarse, &U_c));
5383ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
539d2567f34SHong Zhang }
540d2567f34SHong Zhang 
541d71ae5a4SJacob Faibussowitsch static PetscErrorCode DMSubDomainHook_TSIRK(DM dm, DM subdm, void *ctx)
542d71ae5a4SJacob Faibussowitsch {
543d2567f34SHong Zhang   PetscFunctionBegin;
5443ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
545d2567f34SHong Zhang }
546d2567f34SHong Zhang 
547d71ae5a4SJacob Faibussowitsch static PetscErrorCode DMSubDomainRestrictHook_TSIRK(DM dm, VecScatter gscat, VecScatter lscat, DM subdm, void *ctx)
548d71ae5a4SJacob Faibussowitsch {
549d2567f34SHong Zhang   TS  ts = (TS)ctx;
550d2567f34SHong Zhang   Vec U, U_c;
551d2567f34SHong Zhang 
552d2567f34SHong Zhang   PetscFunctionBegin;
5539566063dSJacob Faibussowitsch   PetscCall(TSIRKGetVecs(ts, dm, &U));
5549566063dSJacob Faibussowitsch   PetscCall(TSIRKGetVecs(ts, subdm, &U_c));
555d2567f34SHong Zhang 
5569566063dSJacob Faibussowitsch   PetscCall(VecScatterBegin(gscat, U, U_c, INSERT_VALUES, SCATTER_FORWARD));
5579566063dSJacob Faibussowitsch   PetscCall(VecScatterEnd(gscat, U, U_c, INSERT_VALUES, SCATTER_FORWARD));
558d2567f34SHong Zhang 
5599566063dSJacob Faibussowitsch   PetscCall(TSIRKRestoreVecs(ts, dm, &U));
5609566063dSJacob Faibussowitsch   PetscCall(TSIRKRestoreVecs(ts, subdm, &U_c));
5613ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
562d2567f34SHong Zhang }
563d2567f34SHong Zhang 
564d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSSetUp_IRK(TS ts)
565d71ae5a4SJacob Faibussowitsch {
566d2567f34SHong Zhang   TS_IRK        *irk = (TS_IRK *)ts->data;
567d2567f34SHong Zhang   IRKTableau     tab = irk->tableau;
568d2567f34SHong Zhang   DM             dm;
569d2567f34SHong Zhang   Mat            J;
570d2567f34SHong Zhang   Vec            R;
571d2567f34SHong Zhang   const PetscInt nstages = irk->nstages;
572d2567f34SHong Zhang   PetscInt       vsize, bs;
573d2567f34SHong Zhang 
574d2567f34SHong Zhang   PetscFunctionBegin;
57548a46eb9SPierre Jolivet   if (!irk->work) PetscCall(PetscMalloc1(irk->nstages, &irk->work));
57648a46eb9SPierre Jolivet   if (!irk->Y) PetscCall(VecDuplicateVecs(ts->vec_sol, irk->nstages, &irk->Y));
57748a46eb9SPierre Jolivet   if (!irk->YdotI) PetscCall(VecDuplicateVecs(ts->vec_sol, irk->nstages, &irk->YdotI));
57848a46eb9SPierre Jolivet   if (!irk->Ydot) PetscCall(VecDuplicate(ts->vec_sol, &irk->Ydot));
57948a46eb9SPierre Jolivet   if (!irk->U) PetscCall(VecDuplicate(ts->vec_sol, &irk->U));
58048a46eb9SPierre Jolivet   if (!irk->U0) PetscCall(VecDuplicate(ts->vec_sol, &irk->U0));
581d2567f34SHong Zhang   if (!irk->Z) {
5829566063dSJacob Faibussowitsch     PetscCall(VecCreate(PetscObjectComm((PetscObject)ts->vec_sol), &irk->Z));
5839566063dSJacob Faibussowitsch     PetscCall(VecGetSize(ts->vec_sol, &vsize));
5849566063dSJacob Faibussowitsch     PetscCall(VecSetSizes(irk->Z, PETSC_DECIDE, vsize * irk->nstages));
5859566063dSJacob Faibussowitsch     PetscCall(VecGetBlockSize(ts->vec_sol, &bs));
5869566063dSJacob Faibussowitsch     PetscCall(VecSetBlockSize(irk->Z, irk->nstages * bs));
5879566063dSJacob Faibussowitsch     PetscCall(VecSetFromOptions(irk->Z));
588d2567f34SHong Zhang   }
5899566063dSJacob Faibussowitsch   PetscCall(TSGetDM(ts, &dm));
5909566063dSJacob Faibussowitsch   PetscCall(DMCoarsenHookAdd(dm, DMCoarsenHook_TSIRK, DMRestrictHook_TSIRK, ts));
5919566063dSJacob Faibussowitsch   PetscCall(DMSubDomainHookAdd(dm, DMSubDomainHook_TSIRK, DMSubDomainRestrictHook_TSIRK, ts));
592d2567f34SHong Zhang 
5939566063dSJacob Faibussowitsch   PetscCall(TSGetSNES(ts, &ts->snes));
5949566063dSJacob Faibussowitsch   PetscCall(VecDuplicate(irk->Z, &R));
5959566063dSJacob Faibussowitsch   PetscCall(SNESSetFunction(ts->snes, R, SNESTSFormFunction, ts));
5969566063dSJacob Faibussowitsch   PetscCall(TSGetIJacobian(ts, &J, NULL, NULL, NULL));
597d2567f34SHong Zhang   if (!irk->TJ) {
598d2567f34SHong Zhang     /* Create the KAIJ matrix for solving the stages */
5999566063dSJacob Faibussowitsch     PetscCall(MatCreateKAIJ(J, nstages, nstages, tab->A_inv, tab->I_s, &irk->TJ));
600d2567f34SHong Zhang   }
6019566063dSJacob Faibussowitsch   PetscCall(SNESSetJacobian(ts->snes, irk->TJ, irk->TJ, SNESTSFormJacobian, ts));
6029566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&R));
6033ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
604d2567f34SHong Zhang }
605d2567f34SHong Zhang 
606d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSSetFromOptions_IRK(TS ts, PetscOptionItems *PetscOptionsObject)
607d71ae5a4SJacob Faibussowitsch {
608d2567f34SHong Zhang   TS_IRK *irk        = (TS_IRK *)ts->data;
609d2567f34SHong Zhang   char    tname[256] = TSIRKGAUSS;
610d2567f34SHong Zhang 
611d2567f34SHong Zhang   PetscFunctionBegin;
612d0609cedSBarry Smith   PetscOptionsHeadBegin(PetscOptionsObject, "IRK ODE solver options");
613d2567f34SHong Zhang   {
614d2567f34SHong Zhang     PetscBool flg1, flg2;
6159566063dSJacob Faibussowitsch     PetscCall(PetscOptionsInt("-ts_irk_nstages", "Stages of the IRK method", "TSIRKSetNumStages", irk->nstages, &irk->nstages, &flg1));
6169566063dSJacob Faibussowitsch     PetscCall(PetscOptionsFList("-ts_irk_type", "Type of IRK method", "TSIRKSetType", TSIRKList, irk->method_name[0] ? irk->method_name : tname, tname, sizeof(tname), &flg2));
617d2567f34SHong Zhang     if (flg1 || flg2 || !irk->method_name[0]) { /* Create the method tableau after nstages or method is set */
6189566063dSJacob Faibussowitsch       PetscCall(TSIRKSetType(ts, tname));
619d2567f34SHong Zhang     }
620d2567f34SHong Zhang   }
621d0609cedSBarry Smith   PetscOptionsHeadEnd();
6223ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
623d2567f34SHong Zhang }
624d2567f34SHong Zhang 
625d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSView_IRK(TS ts, PetscViewer viewer)
626d71ae5a4SJacob Faibussowitsch {
627d2567f34SHong Zhang   TS_IRK   *irk = (TS_IRK *)ts->data;
628d2567f34SHong Zhang   PetscBool iascii;
629d2567f34SHong Zhang 
630d2567f34SHong Zhang   PetscFunctionBegin;
6319566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
632d2567f34SHong Zhang   if (iascii) {
633d2567f34SHong Zhang     IRKTableau tab = irk->tableau;
634d2567f34SHong Zhang     TSIRKType  irktype;
635d2567f34SHong Zhang     char       buf[512];
636d2567f34SHong Zhang 
6379566063dSJacob Faibussowitsch     PetscCall(TSIRKGetType(ts, &irktype));
6389566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPrintf(viewer, "  IRK type %s\n", irktype));
6399566063dSJacob Faibussowitsch     PetscCall(PetscFormatRealArray(buf, sizeof(buf), "% 8.6f", irk->nstages, tab->c));
6409566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPrintf(viewer, "  Abscissa       c = %s\n", buf));
6419566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPrintf(viewer, "Stiffly accurate: %s\n", irk->stiffly_accurate ? "yes" : "no"));
6429566063dSJacob Faibussowitsch     PetscCall(PetscFormatRealArray(buf, sizeof(buf), "% 8.6f", PetscSqr(irk->nstages), tab->A));
6439566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPrintf(viewer, "  A coefficients       A = %s\n", buf));
644d2567f34SHong Zhang   }
6453ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
646d2567f34SHong Zhang }
647d2567f34SHong Zhang 
648d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSLoad_IRK(TS ts, PetscViewer viewer)
649d71ae5a4SJacob Faibussowitsch {
650d2567f34SHong Zhang   SNES    snes;
651d2567f34SHong Zhang   TSAdapt adapt;
652d2567f34SHong Zhang 
653d2567f34SHong Zhang   PetscFunctionBegin;
6549566063dSJacob Faibussowitsch   PetscCall(TSGetAdapt(ts, &adapt));
6559566063dSJacob Faibussowitsch   PetscCall(TSAdaptLoad(adapt, viewer));
6569566063dSJacob Faibussowitsch   PetscCall(TSGetSNES(ts, &snes));
6579566063dSJacob Faibussowitsch   PetscCall(SNESLoad(snes, viewer));
658d2567f34SHong Zhang   /* function and Jacobian context for SNES when used with TS is always ts object */
6599566063dSJacob Faibussowitsch   PetscCall(SNESSetFunction(snes, NULL, NULL, ts));
6609566063dSJacob Faibussowitsch   PetscCall(SNESSetJacobian(snes, NULL, NULL, NULL, ts));
6613ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
662d2567f34SHong Zhang }
663d2567f34SHong Zhang 
664d2567f34SHong Zhang /*@C
665bcf0153eSBarry Smith   TSIRKSetType - Set the type of `TSIRK` scheme to use
666d2567f34SHong Zhang 
66720f4b53cSBarry Smith   Logically Collective
668d2567f34SHong Zhang 
66902024617SSatish Balay   Input Parameters:
670d2567f34SHong Zhang +  ts - timestepping context
671bcf0153eSBarry Smith -  irktype - type of `TSIRK` scheme
672d2567f34SHong Zhang 
673bcf0153eSBarry Smith   Options Database Key:
67467b8a455SSatish Balay .  -ts_irk_type <gauss> - set irk type
675d2567f34SHong Zhang 
676d2567f34SHong Zhang   Level: intermediate
677d2567f34SHong Zhang 
678*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRKGetType()`, `TSIRK`, `TSIRKType`, `TSIRKGAUSS`
679d2567f34SHong Zhang @*/
680d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKSetType(TS ts, TSIRKType irktype)
681d71ae5a4SJacob Faibussowitsch {
682d2567f34SHong Zhang   PetscFunctionBegin;
683d2567f34SHong Zhang   PetscValidHeaderSpecific(ts, TS_CLASSID, 1);
684d2567f34SHong Zhang   PetscValidCharPointer(irktype, 2);
685cac4c232SBarry Smith   PetscTryMethod(ts, "TSIRKSetType_C", (TS, TSIRKType), (ts, irktype));
6863ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
687d2567f34SHong Zhang }
688d2567f34SHong Zhang 
689d2567f34SHong Zhang /*@C
690bcf0153eSBarry Smith   TSIRKGetType - Get the type of `TSIRK` IMEX scheme being used
691d2567f34SHong Zhang 
69220f4b53cSBarry Smith   Logically Collective
693d2567f34SHong Zhang 
694d2567f34SHong Zhang   Input Parameter:
695d2567f34SHong Zhang .  ts - timestepping context
696d2567f34SHong Zhang 
697d2567f34SHong Zhang   Output Parameter:
698bcf0153eSBarry Smith .  irktype - type of `TSIRK` IMEX scheme
699d2567f34SHong Zhang 
700d2567f34SHong Zhang   Level: intermediate
701d2567f34SHong Zhang 
702*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRKGetType()`, `TSIRK`, `TSIRKType`, `TSIRKGAUSS`
703d2567f34SHong Zhang @*/
704d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKGetType(TS ts, TSIRKType *irktype)
705d71ae5a4SJacob Faibussowitsch {
706d2567f34SHong Zhang   PetscFunctionBegin;
707d2567f34SHong Zhang   PetscValidHeaderSpecific(ts, TS_CLASSID, 1);
708cac4c232SBarry Smith   PetscUseMethod(ts, "TSIRKGetType_C", (TS, TSIRKType *), (ts, irktype));
7093ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
710d2567f34SHong Zhang }
711d2567f34SHong Zhang 
712d2567f34SHong Zhang /*@C
713bcf0153eSBarry Smith   TSIRKSetNumStages - Set the number of stages of `TSIRK` scheme to use
714d2567f34SHong Zhang 
71520f4b53cSBarry Smith   Logically Collective
716d2567f34SHong Zhang 
71702024617SSatish Balay   Input Parameters:
718d2567f34SHong Zhang +  ts - timestepping context
719bcf0153eSBarry Smith -  nstages - number of stages of `TSIRK` scheme
720d2567f34SHong Zhang 
721bcf0153eSBarry Smith   Options Database Key:
72267b8a455SSatish Balay .  -ts_irk_nstages <int> - set number of stages
723d2567f34SHong Zhang 
724d2567f34SHong Zhang   Level: intermediate
725d2567f34SHong Zhang 
726*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRKGetNumStages()`, `TSIRK`
727d2567f34SHong Zhang @*/
728d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKSetNumStages(TS ts, PetscInt nstages)
729d71ae5a4SJacob Faibussowitsch {
730d2567f34SHong Zhang   PetscFunctionBegin;
731d2567f34SHong Zhang   PetscValidHeaderSpecific(ts, TS_CLASSID, 1);
732cac4c232SBarry Smith   PetscTryMethod(ts, "TSIRKSetNumStages_C", (TS, PetscInt), (ts, nstages));
7333ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
734d2567f34SHong Zhang }
735d2567f34SHong Zhang 
736d2567f34SHong Zhang /*@C
737bcf0153eSBarry Smith   TSIRKGetNumStages - Get the number of stages of `TSIRK` scheme
738d2567f34SHong Zhang 
73920f4b53cSBarry Smith   Logically Collective
740d2567f34SHong Zhang 
74102024617SSatish Balay   Input Parameters:
742d2567f34SHong Zhang +  ts - timestepping context
743bcf0153eSBarry Smith -  nstages - number of stages of `TSIRK` scheme
744d2567f34SHong Zhang 
745d2567f34SHong Zhang   Level: intermediate
746d2567f34SHong Zhang 
747*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSIRKSetNumStages()`, `TSIRK`
748d2567f34SHong Zhang @*/
749d71ae5a4SJacob Faibussowitsch PetscErrorCode TSIRKGetNumStages(TS ts, PetscInt *nstages)
750d71ae5a4SJacob Faibussowitsch {
751d2567f34SHong Zhang   PetscFunctionBegin;
752d2567f34SHong Zhang   PetscValidHeaderSpecific(ts, TS_CLASSID, 1);
753d2567f34SHong Zhang   PetscValidIntPointer(nstages, 2);
754cac4c232SBarry Smith   PetscTryMethod(ts, "TSIRKGetNumStages_C", (TS, PetscInt *), (ts, nstages));
7553ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
756d2567f34SHong Zhang }
757d2567f34SHong Zhang 
758d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKGetType_IRK(TS ts, TSIRKType *irktype)
759d71ae5a4SJacob Faibussowitsch {
760d2567f34SHong Zhang   TS_IRK *irk = (TS_IRK *)ts->data;
761d2567f34SHong Zhang 
762d2567f34SHong Zhang   PetscFunctionBegin;
763d2567f34SHong Zhang   *irktype = irk->method_name;
7643ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
765d2567f34SHong Zhang }
766d2567f34SHong Zhang 
767d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKSetType_IRK(TS ts, TSIRKType irktype)
768d71ae5a4SJacob Faibussowitsch {
769d2567f34SHong Zhang   TS_IRK *irk = (TS_IRK *)ts->data;
7705f80ce2aSJacob Faibussowitsch   PetscErrorCode (*irkcreate)(TS);
771d2567f34SHong Zhang 
772d2567f34SHong Zhang   PetscFunctionBegin;
773d2567f34SHong Zhang   if (irk->method_name) {
7749566063dSJacob Faibussowitsch     PetscCall(PetscFree(irk->method_name));
7759566063dSJacob Faibussowitsch     PetscCall(TSIRKTableauReset(ts));
776d2567f34SHong Zhang   }
7779566063dSJacob Faibussowitsch   PetscCall(PetscFunctionListFind(TSIRKList, irktype, &irkcreate));
7783c633725SBarry Smith   PetscCheck(irkcreate, PETSC_COMM_SELF, PETSC_ERR_ARG_UNKNOWN_TYPE, "Unknown TSIRK type \"%s\" given", irktype);
7799566063dSJacob Faibussowitsch   PetscCall((*irkcreate)(ts));
7809566063dSJacob Faibussowitsch   PetscCall(PetscStrallocpy(irktype, &irk->method_name));
7813ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
782d2567f34SHong Zhang }
783d2567f34SHong Zhang 
784d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKSetNumStages_IRK(TS ts, PetscInt nstages)
785d71ae5a4SJacob Faibussowitsch {
786d2567f34SHong Zhang   TS_IRK *irk = (TS_IRK *)ts->data;
787d2567f34SHong Zhang 
788d2567f34SHong Zhang   PetscFunctionBegin;
78963a3b9bcSJacob Faibussowitsch   PetscCheck(nstages > 0, PETSC_COMM_SELF, PETSC_ERR_ARG_OUTOFRANGE, "input argument, %" PetscInt_FMT ", out of range", nstages);
790d2567f34SHong Zhang   irk->nstages = nstages;
7913ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
792d2567f34SHong Zhang }
793d2567f34SHong Zhang 
794d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSIRKGetNumStages_IRK(TS ts, PetscInt *nstages)
795d71ae5a4SJacob Faibussowitsch {
796d2567f34SHong Zhang   TS_IRK *irk = (TS_IRK *)ts->data;
797d2567f34SHong Zhang 
798d2567f34SHong Zhang   PetscFunctionBegin;
799d2567f34SHong Zhang   PetscValidIntPointer(nstages, 2);
800d2567f34SHong Zhang   *nstages = irk->nstages;
8013ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
802d2567f34SHong Zhang }
803d2567f34SHong Zhang 
804d71ae5a4SJacob Faibussowitsch static PetscErrorCode TSDestroy_IRK(TS ts)
805d71ae5a4SJacob Faibussowitsch {
806d2567f34SHong Zhang   PetscFunctionBegin;
8079566063dSJacob Faibussowitsch   PetscCall(TSReset_IRK(ts));
808d2567f34SHong Zhang   if (ts->dm) {
8099566063dSJacob Faibussowitsch     PetscCall(DMCoarsenHookRemove(ts->dm, DMCoarsenHook_TSIRK, DMRestrictHook_TSIRK, ts));
8109566063dSJacob Faibussowitsch     PetscCall(DMSubDomainHookRemove(ts->dm, DMSubDomainHook_TSIRK, DMSubDomainRestrictHook_TSIRK, ts));
811d2567f34SHong Zhang   }
8129566063dSJacob Faibussowitsch   PetscCall(PetscFree(ts->data));
8139566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKSetType_C", NULL));
8149566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKGetType_C", NULL));
8159566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKSetNumStages_C", NULL));
8169566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKGetNumStages_C", NULL));
8173ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
818d2567f34SHong Zhang }
819d2567f34SHong Zhang 
820d2567f34SHong Zhang /*MC
821d2567f34SHong Zhang       TSIRK - ODE and DAE solver using Implicit Runge-Kutta schemes
822d2567f34SHong Zhang 
823d2567f34SHong Zhang   Level: beginner
824d2567f34SHong Zhang 
825bcf0153eSBarry Smith   Notes:
826bcf0153eSBarry Smith   `TSIRK` uses the sparse Kronecker product matrix implementation of `MATKAIJ` to achieve good arithmetic intensity.
827d2567f34SHong Zhang 
828bcf0153eSBarry Smith   Gauss-Legrendre methods are currently supported. These are A-stable symplectic methods with an arbitrary number of stages. The order of accuracy is 2s
829bcf0153eSBarry Smith   when using s stages. The default method uses three stages and thus has an order of six. The number of stages (thus order) can be set with
830bcf0153eSBarry Smith   -ts_irk_nstages or `TSIRKSetNumStages()`.
831bcf0153eSBarry Smith 
832*1cc06b55SBarry Smith .seealso: [](ch_ts), `TSCreate()`, `TS`, `TSSetType()`, `TSIRKSetType()`, `TSIRKGetType()`, `TSIRKGAUSS`, `TSIRKRegister()`, `TSIRKSetNumStages()`, `TSType`
833d2567f34SHong Zhang M*/
834d71ae5a4SJacob Faibussowitsch PETSC_EXTERN PetscErrorCode TSCreate_IRK(TS ts)
835d71ae5a4SJacob Faibussowitsch {
836d2567f34SHong Zhang   TS_IRK *irk;
837d2567f34SHong Zhang 
838d2567f34SHong Zhang   PetscFunctionBegin;
8399566063dSJacob Faibussowitsch   PetscCall(TSIRKInitializePackage());
840d2567f34SHong Zhang 
841d2567f34SHong Zhang   ts->ops->reset          = TSReset_IRK;
842d2567f34SHong Zhang   ts->ops->destroy        = TSDestroy_IRK;
843d2567f34SHong Zhang   ts->ops->view           = TSView_IRK;
844d2567f34SHong Zhang   ts->ops->load           = TSLoad_IRK;
845d2567f34SHong Zhang   ts->ops->setup          = TSSetUp_IRK;
846d2567f34SHong Zhang   ts->ops->step           = TSStep_IRK;
847d2567f34SHong Zhang   ts->ops->interpolate    = TSInterpolate_IRK;
848d2567f34SHong Zhang   ts->ops->evaluatestep   = TSEvaluateStep_IRK;
849d2567f34SHong Zhang   ts->ops->rollback       = TSRollBack_IRK;
850d2567f34SHong Zhang   ts->ops->setfromoptions = TSSetFromOptions_IRK;
851d2567f34SHong Zhang   ts->ops->snesfunction   = SNESTSFormFunction_IRK;
852d2567f34SHong Zhang   ts->ops->snesjacobian   = SNESTSFormJacobian_IRK;
853d2567f34SHong Zhang 
854d2567f34SHong Zhang   ts->usessnes = PETSC_TRUE;
855d2567f34SHong Zhang 
8564dfa11a4SJacob Faibussowitsch   PetscCall(PetscNew(&irk));
857d2567f34SHong Zhang   ts->data = (void *)irk;
858d2567f34SHong Zhang 
8599566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKSetType_C", TSIRKSetType_IRK));
8609566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKGetType_C", TSIRKGetType_IRK));
8619566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKSetNumStages_C", TSIRKSetNumStages_IRK));
8629566063dSJacob Faibussowitsch   PetscCall(PetscObjectComposeFunction((PetscObject)ts, "TSIRKGetNumStages_C", TSIRKGetNumStages_IRK));
863d2567f34SHong Zhang   /* 3-stage IRK_Gauss is the default */
8649566063dSJacob Faibussowitsch   PetscCall(PetscNew(&irk->tableau));
865d2567f34SHong Zhang   irk->nstages = 3;
8669566063dSJacob Faibussowitsch   PetscCall(TSIRKSetType(ts, TSIRKDefault));
8673ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
868d2567f34SHong Zhang }
869