Actual source code: ex1.c
1: static char help[] = "Test file for the PCFactorSetShiftType()\n";
2: /*
3: * Test file for the PCFactorSetShiftType() routine or -pc_factor_shift_type POSITIVE_DEFINITE option.
4: * The test matrix is the example from Kershaw's paper [J.Comp.Phys 1978]
5: * of a positive definite matrix for which ILU(0) will give a negative pivot.
6: * This means that the CG method will break down; the Manteuffel shift
7: * [Math. Comp. 1980] repairs this.
8: *
9: * Run the executable twice:
10: * 1/ without options: the iterative method diverges because of an
11: * indefinite preconditioner
12: * 2/ with -pc_factor_shift_type POSITIVE_DEFINITE option (or comment in the PCFactorSetShiftType() line below):
13: * the method will now successfully converge.
14: *
15: * Contributed by Victor Eijkhout 2003.
16: */
18: #include <petscksp.h>
20: int main(int argc, char **argv)
21: {
22: KSP solver;
23: PC prec;
24: Mat A, M;
25: Vec X, B, D;
26: MPI_Comm comm;
27: PetscScalar v;
28: KSPConvergedReason reason;
29: PetscInt i, j, its;
31: PetscFunctionBeginUser;
32: PetscCall(PetscInitialize(&argc, &argv, 0, help));
33: comm = MPI_COMM_SELF;
35: /*
36: * Construct the Kershaw matrix
37: * and a suitable rhs / initial guess
38: */
39: PetscCall(MatCreateSeqAIJ(comm, 4, 4, 4, 0, &A));
40: PetscCall(VecCreateSeq(comm, 4, &B));
41: PetscCall(VecDuplicate(B, &X));
42: for (i = 0; i < 4; i++) {
43: v = 3;
44: PetscCall(MatSetValues(A, 1, &i, 1, &i, &v, INSERT_VALUES));
45: v = 1;
46: PetscCall(VecSetValues(B, 1, &i, &v, INSERT_VALUES));
47: PetscCall(VecSetValues(X, 1, &i, &v, INSERT_VALUES));
48: }
50: i = 0;
51: v = 0;
52: PetscCall(VecSetValues(X, 1, &i, &v, INSERT_VALUES));
54: for (i = 0; i < 3; i++) {
55: v = -2;
56: j = i + 1;
57: PetscCall(MatSetValues(A, 1, &i, 1, &j, &v, INSERT_VALUES));
58: PetscCall(MatSetValues(A, 1, &j, 1, &i, &v, INSERT_VALUES));
59: }
60: i = 0;
61: j = 3;
62: v = 2;
64: PetscCall(MatSetValues(A, 1, &i, 1, &j, &v, INSERT_VALUES));
65: PetscCall(MatSetValues(A, 1, &j, 1, &i, &v, INSERT_VALUES));
66: PetscCall(MatAssemblyBegin(A, MAT_FINAL_ASSEMBLY));
67: PetscCall(MatAssemblyEnd(A, MAT_FINAL_ASSEMBLY));
68: PetscCall(VecAssemblyBegin(B));
69: PetscCall(VecAssemblyEnd(B));
71: /*
72: * A Conjugate Gradient method
73: * with ILU(0) preconditioning
74: */
75: PetscCall(KSPCreate(comm, &solver));
76: PetscCall(KSPSetOperators(solver, A, A));
78: PetscCall(KSPSetType(solver, KSPCG));
79: PetscCall(KSPSetInitialGuessNonzero(solver, PETSC_TRUE));
81: /*
82: * ILU preconditioner;
83: * this will break down unless you add the Shift line,
84: * or use the -pc_factor_shift_positive_definite option */
85: PetscCall(KSPGetPC(solver, &prec));
86: PetscCall(PCSetType(prec, PCILU));
87: /* PetscCall(PCFactorSetShiftType(prec,MAT_SHIFT_POSITIVE_DEFINITE)); */
89: PetscCall(KSPSetFromOptions(solver));
90: PetscCall(KSPSetUp(solver));
92: /*
93: * Now that the factorisation is done, show the pivots;
94: * note that the last one is negative. This in itself is not an error,
95: * but it will make the iterative method diverge.
96: */
97: PetscCall(PCFactorGetMatrix(prec, &M));
98: PetscCall(VecDuplicate(B, &D));
99: PetscCall(MatGetDiagonal(M, D));
101: /*
102: * Solve the system;
103: * without the shift this will diverge with
104: * an indefinite preconditioner
105: */
106: PetscCall(KSPSolve(solver, B, X));
107: PetscCall(KSPGetConvergedReason(solver, &reason));
108: if (reason == KSP_DIVERGED_INDEFINITE_PC) {
109: PetscCall(PetscPrintf(PETSC_COMM_WORLD, "\nDivergence because of indefinite preconditioner;\n"));
110: PetscCall(PetscPrintf(PETSC_COMM_WORLD, "Run the executable again but with '-pc_factor_shift_type POSITIVE_DEFINITE' option.\n"));
111: } else if (reason < 0) {
112: PetscCall(PetscPrintf(PETSC_COMM_WORLD, "\nOther kind of divergence: this should not happen.\n"));
113: } else {
114: PetscCall(KSPGetIterationNumber(solver, &its));
115: }
117: PetscCall(VecDestroy(&X));
118: PetscCall(VecDestroy(&B));
119: PetscCall(VecDestroy(&D));
120: PetscCall(MatDestroy(&A));
121: PetscCall(KSPDestroy(&solver));
122: PetscCall(PetscFinalize());
123: return 0;
124: }
126: /*TEST
128: test:
129: args: -pc_factor_shift_type positive_definite
131: TEST*/