Actual source code: ex24.c
1: /*$Id: ex24.c,v 1.11 2001/04/10 19:36:37 bsmith Exp $*/
3: static char help[] = "Tests CG, MINRES and SYMMLQ on symmetric matrices with SBAIJ format. The preconditioner ICC only works on sequential SBAIJ format. nn";
5: #include petscsles.h
8: int main(int argc,char **args)
9: {
10: Mat C;
11: Scalar v,none = -1.0;
12: int i,j,I,J,ierr,Istart,Iend,N,m = 4,n = 4,rank,size,its,k;
13: double err_norm,res_norm;
14: Vec x,b,u,u_tmp;
15: PetscRandom r;
16: SLES sles;
17: PC pc;
18: KSP ksp;
20: PetscInitialize(&argc,&args,(char *)0,help);
21: MPI_Comm_rank(PETSC_COMM_WORLD,&rank);
22: MPI_Comm_size(PETSC_COMM_WORLD,&size);
23: PetscOptionsGetInt(PETSC_NULL,"-m",&m,PETSC_NULL);
24: PetscOptionsGetInt(PETSC_NULL,"-n",&n,PETSC_NULL);
25: N = m*n;
28: /* Generate matrix */
29: MatCreate(PETSC_COMM_WORLD,PETSC_DECIDE,PETSC_DECIDE,N,N,&C);
30: MatSetFromOptions(C);
31: MatGetOwnershipRange(C,&Istart,&Iend);
32: for (I=Istart; I<Iend; I++) {
33: v = -1.0; i = I/n; j = I - i*n;
34: if (i>0) {J = I - n; MatSetValues(C,1,&I,1,&J,&v,ADD_VALUES);}
35: if (i<m-1) {J = I + n; MatSetValues(C,1,&I,1,&J,&v,ADD_VALUES);}
36: if (j>0) {J = I - 1; MatSetValues(C,1,&I,1,&J,&v,ADD_VALUES);}
37: if (j<n-1) {J = I + 1; MatSetValues(C,1,&I,1,&J,&v,ADD_VALUES);}
38: v = 4.0; MatSetValues(C,1,&I,1,&I,&v,ADD_VALUES);
39: }
40: MatAssemblyBegin(C,MAT_FINAL_ASSEMBLY);
41: MatAssemblyEnd(C,MAT_FINAL_ASSEMBLY);
43: /* a shift can make C indefinite. Preconditioners LU, ILU (for BAIJ format) and ICC may fail */
44: /* MatShift(&alpha, C); */
45: /* MatView(C,PETSC_VIEWER_STDOUT_WORLD); */
47: /* Setup and solve for system */
48:
49: /* Create vectors. */
50: VecCreate(PETSC_COMM_WORLD,PETSC_DECIDE,N,&x);
51: VecSetFromOptions(x);
52: VecDuplicate(x,&b);
53: VecDuplicate(x,&u);
54: VecDuplicate(x,&u_tmp);
56: /* Set exact solution u; then compute right-hand-side vector b. */
57: PetscRandomCreate(PETSC_COMM_SELF,RANDOM_DEFAULT,&r);
58: VecSetRandom(r,u);
59: PetscRandomDestroy(r);
60:
61: MatMult(C,u,b);
63: for (k=0; k<3; k++){
64: if (k == 0){ /* CG */
65: SLESCreate(PETSC_COMM_WORLD,&sles);
66: SLESSetOperators(sles,C,C,DIFFERENT_NONZERO_PATTERN);
67: SLESGetKSP(sles,&ksp);
68: PetscPrintf(PETSC_COMM_WORLD,"n CG: n");
69: KSPSetType(ksp,KSPCG);
70: } else if (k == 1){ /* MINRES */
71: SLESCreate(PETSC_COMM_WORLD,&sles);
72: SLESSetOperators(sles,C,C,DIFFERENT_NONZERO_PATTERN);
73: SLESGetKSP(sles,&ksp);
74: PetscPrintf(PETSC_COMM_WORLD,"n MINRES: n");
75: KSPSetType(ksp,KSPMINRES);
76: } else { /* SYMMLQ */
77: SLESCreate(PETSC_COMM_WORLD,&sles);
78: SLESSetOperators(sles,C,C,DIFFERENT_NONZERO_PATTERN);
79: SLESGetKSP(sles,&ksp);
80: PetscPrintf(PETSC_COMM_WORLD,"n SYMMLQ: n");
81: KSPSetType(ksp,KSPSYMMLQ);
82: }
84: SLESGetPC(sles,&pc);
85: /* PCSetType(pc,PCICC); */
86: PCSetType(pc,PCJACOBI);
87: KSPSetTolerances(ksp,1.e-7,PETSC_DEFAULT,PETSC_DEFAULT,PETSC_DEFAULT);
89: /*
90: Set runtime options, e.g.,
91: -ksp_type <type> -pc_type <type> -ksp_monitor -ksp_rtol <rtol>
92: These options will override those specified above as long as
93: SLESSetFromOptions() is called _after_ any other customization
94: routines.
95: */
96: SLESSetFromOptions(sles);
98: /* Solve linear system; */
99: SLESSolve(sles,b,x,&its);
100:
101: /* Check error */
102: VecCopy(u,u_tmp);
103: VecAXPY(&none,x,u_tmp);
104: VecNorm(u_tmp,NORM_2,&err_norm);
105: MatMult(C,x,u_tmp);
106: VecAXPY(&none,b,u_tmp);
107: VecNorm(u_tmp,NORM_2,&res_norm);
108:
109: PetscPrintf(PETSC_COMM_WORLD,"Number of iterations = %3dn",its);
110: PetscPrintf(PETSC_COMM_WORLD,"Residual norm %A;",res_norm);
111: PetscPrintf(PETSC_COMM_WORLD," Error norm %A.n",err_norm);
113: SLESDestroy(sles);
114: }
115:
116: /*
117: Free work space. All PETSc objects should be destroyed when they
118: are no longer needed.
119: */
120: VecDestroy(b);
121: VecDestroy(u);
122: VecDestroy(x);
123: VecDestroy(u_tmp);
124: MatDestroy(C);
126: PetscFinalize();
127: return 0;
128: }