Actual source code: ex1.c

  1: static char help[] = "Basic equation for generator stability analysis.\n";

  3: /*F

  5: \begin{eqnarray}
  6:                  \frac{2 H}{\omega_s}\frac{d \omega}{dt} & = & P_m - \frac{EV}{X} \sin(\theta) \\
  7:                  \frac{d \theta}{dt} = \omega - \omega_s
  8: \end{eqnarray}

 10: F*/

 12: /*
 13:    Include "petscts.h" so that we can use TS solvers.  Note that this
 14:    file automatically includes:
 15:      petscsys.h       - base PETSc routines   petscvec.h - vectors
 16:      petscmat.h - matrices
 17:      petscis.h     - index sets            petscksp.h - Krylov subspace methods
 18:      petscviewer.h - viewers               petscpc.h  - preconditioners
 19:      petscksp.h   - linear solvers
 20: */

 22: #include <petscts.h>

 24: typedef struct {
 25:   PetscScalar H, omega_s, E, V, X;
 26:   PetscRandom rand;
 27: } AppCtx;

 29: /*
 30:      Defines the ODE passed to the ODE solver
 31: */
 32: static PetscErrorCode IFunction(TS ts, PetscReal t, Vec U, Vec Udot, Vec F, AppCtx *ctx)
 33: {
 34:   PetscScalar       *f, r;
 35:   const PetscScalar *u, *udot;
 36:   static PetscScalar R = .4;

 38:   PetscFunctionBegin;
 39:   PetscCall(PetscRandomGetValue(ctx->rand, &r));
 40:   if (r > .9) R = .5;
 41:   if (r < .1) R = .4;
 42:   R = .4;
 43:   /*  The next three lines allow us to access the entries of the vectors directly */
 44:   PetscCall(VecGetArrayRead(U, &u));
 45:   PetscCall(VecGetArrayRead(Udot, &udot));
 46:   PetscCall(VecGetArray(F, &f));
 47:   f[0] = 2.0 * ctx->H * udot[0] / ctx->omega_s + ctx->E * ctx->V * PetscSinScalar(u[1]) / ctx->X - R;
 48:   f[1] = udot[1] - u[0] + ctx->omega_s;

 50:   PetscCall(VecRestoreArrayRead(U, &u));
 51:   PetscCall(VecRestoreArrayRead(Udot, &udot));
 52:   PetscCall(VecRestoreArray(F, &f));
 53:   PetscFunctionReturn(PETSC_SUCCESS);
 54: }

 56: /*
 57:      Defines the Jacobian of the ODE passed to the ODE solver. See TSSetIJacobian() for the meaning of a and the Jacobian.
 58: */
 59: static PetscErrorCode IJacobian(TS ts, PetscReal t, Vec U, Vec Udot, PetscReal a, Mat A, Mat B, AppCtx *ctx)
 60: {
 61:   PetscInt           rowcol[] = {0, 1};
 62:   PetscScalar        J[2][2];
 63:   const PetscScalar *u, *udot;

 65:   PetscFunctionBegin;
 66:   PetscCall(VecGetArrayRead(U, &u));
 67:   PetscCall(VecGetArrayRead(Udot, &udot));
 68:   J[0][0] = 2.0 * ctx->H * a / ctx->omega_s;
 69:   J[0][1] = -ctx->E * ctx->V * PetscCosScalar(u[1]) / ctx->X;
 70:   J[1][0] = -1.0;
 71:   J[1][1] = a;
 72:   PetscCall(MatSetValues(B, 2, rowcol, 2, rowcol, &J[0][0], INSERT_VALUES));
 73:   PetscCall(VecRestoreArrayRead(U, &u));
 74:   PetscCall(VecRestoreArrayRead(Udot, &udot));

 76:   PetscCall(MatAssemblyBegin(A, MAT_FINAL_ASSEMBLY));
 77:   PetscCall(MatAssemblyEnd(A, MAT_FINAL_ASSEMBLY));
 78:   if (A != B) {
 79:     PetscCall(MatAssemblyBegin(B, MAT_FINAL_ASSEMBLY));
 80:     PetscCall(MatAssemblyEnd(B, MAT_FINAL_ASSEMBLY));
 81:   }
 82:   PetscFunctionReturn(PETSC_SUCCESS);
 83: }

 85: int main(int argc, char **argv)
 86: {
 87:   TS           ts; /* ODE integrator */
 88:   Vec          U;  /* solution will be stored here */
 89:   Mat          A;  /* Jacobian matrix */
 90:   PetscMPIInt  size;
 91:   PetscInt     n = 2;
 92:   AppCtx       ctx;
 93:   PetscScalar *u;

 95:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
 96:      Initialize program
 97:      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
 98:   PetscFunctionBeginUser;
 99:   PetscCall(PetscInitialize(&argc, &argv, (char *)0, help));
100:   PetscCallMPI(MPI_Comm_size(PETSC_COMM_WORLD, &size));
101:   PetscCheck(size == 1, PETSC_COMM_WORLD, PETSC_ERR_WRONG_MPI_SIZE, "Only for sequential runs");

103:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
104:     Create necessary matrix and vectors
105:     - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
106:   PetscCall(MatCreate(PETSC_COMM_WORLD, &A));
107:   PetscCall(MatSetSizes(A, n, n, PETSC_DETERMINE, PETSC_DETERMINE));
108:   PetscCall(MatSetFromOptions(A));
109:   PetscCall(MatSetUp(A));

111:   PetscCall(MatCreateVecs(A, &U, NULL));

113:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
114:     Set runtime options
115:     - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
116:   PetscOptionsBegin(PETSC_COMM_WORLD, NULL, "Reaction options", "");
117:   {
118:     ctx.omega_s = 1.0;
119:     PetscCall(PetscOptionsScalar("-omega_s", "", "", ctx.omega_s, &ctx.omega_s, NULL));
120:     ctx.H = 1.0;
121:     PetscCall(PetscOptionsScalar("-H", "", "", ctx.H, &ctx.H, NULL));
122:     ctx.E = 1.0;
123:     PetscCall(PetscOptionsScalar("-E", "", "", ctx.E, &ctx.E, NULL));
124:     ctx.V = 1.0;
125:     PetscCall(PetscOptionsScalar("-V", "", "", ctx.V, &ctx.V, NULL));
126:     ctx.X = 1.0;
127:     PetscCall(PetscOptionsScalar("-X", "", "", ctx.X, &ctx.X, NULL));

129:     PetscCall(VecGetArray(U, &u));
130:     u[0] = 1;
131:     u[1] = .7;
132:     PetscCall(VecRestoreArray(U, &u));
133:     PetscCall(PetscOptionsGetVec(NULL, NULL, "-initial", U, NULL));
134:   }
135:   PetscOptionsEnd();

137:   PetscCall(PetscRandomCreate(PETSC_COMM_WORLD, &ctx.rand));
138:   PetscCall(PetscRandomSetFromOptions(ctx.rand));

140:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
141:      Create timestepping solver context
142:      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
143:   PetscCall(TSCreate(PETSC_COMM_WORLD, &ts));
144:   PetscCall(TSSetProblemType(ts, TS_NONLINEAR));
145:   PetscCall(TSSetType(ts, TSROSW));
146:   PetscCall(TSSetIFunction(ts, NULL, (TSIFunction)IFunction, &ctx));
147:   PetscCall(TSSetIJacobian(ts, A, A, (TSIJacobian)IJacobian, &ctx));

149:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
150:      Set initial conditions
151:    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
152:   PetscCall(TSSetSolution(ts, U));

154:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
155:      Set solver options
156:    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
157:   PetscCall(TSSetMaxTime(ts, 2000.0));
158:   PetscCall(TSSetExactFinalTime(ts, TS_EXACTFINALTIME_MATCHSTEP));
159:   PetscCall(TSSetTimeStep(ts, .001));
160:   PetscCall(TSSetFromOptions(ts));

162:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
163:      Solve nonlinear system
164:      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
165:   PetscCall(TSSolve(ts, U));

167:   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
168:      Free work space.  All PETSc objects should be destroyed when they are no longer needed.
169:    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
170:   PetscCall(MatDestroy(&A));
171:   PetscCall(VecDestroy(&U));
172:   PetscCall(TSDestroy(&ts));
173:   PetscCall(PetscRandomDestroy(&ctx.rand));
174:   PetscCall(PetscFinalize());
175:   return 0;
176: }

178: /*TEST

180:    build:
181:      requires: !complex !single

183:    test:
184:       args: -ksp_gmres_cgs_refinement_type refine_always -snes_type newtonls -ts_max_steps 10

186:    test:
187:       suffix: 2
188:       args: -ts_max_steps 10

190: TEST*/