xref: /petsc/src/tao/bound/impls/tron/tron.c (revision a7e14dcfba0d07adf6226a919460249440ec94c7)
1*a7e14dcfSSatish Balay #include "tron.h"
2*a7e14dcfSSatish Balay #include "petsc-private/kspimpl.h"
3*a7e14dcfSSatish Balay #include "petsc-private/matimpl.h"
4*a7e14dcfSSatish Balay #include "src/matrix/submatfree.h"
5*a7e14dcfSSatish Balay 
6*a7e14dcfSSatish Balay 
7*a7e14dcfSSatish Balay /* TRON Routines */
8*a7e14dcfSSatish Balay static PetscErrorCode TronGradientProjections(TaoSolver,TAO_TRON*);
9*a7e14dcfSSatish Balay /*------------------------------------------------------------*/
10*a7e14dcfSSatish Balay #undef __FUNCT__
11*a7e14dcfSSatish Balay #define __FUNCT__ "TaoDestroy_TRON"
12*a7e14dcfSSatish Balay static PetscErrorCode TaoDestroy_TRON(TaoSolver tao)
13*a7e14dcfSSatish Balay {
14*a7e14dcfSSatish Balay   TAO_TRON *tron = (TAO_TRON *)tao->data;
15*a7e14dcfSSatish Balay   PetscErrorCode ierr;
16*a7e14dcfSSatish Balay 
17*a7e14dcfSSatish Balay   PetscFunctionBegin;
18*a7e14dcfSSatish Balay 
19*a7e14dcfSSatish Balay   ierr = VecDestroy(&tron->X_New);CHKERRQ(ierr);
20*a7e14dcfSSatish Balay   ierr = VecDestroy(&tron->G_New);CHKERRQ(ierr);
21*a7e14dcfSSatish Balay   ierr = VecDestroy(&tron->Work);CHKERRQ(ierr);
22*a7e14dcfSSatish Balay   ierr = VecDestroy(&tron->DXFree);CHKERRQ(ierr);
23*a7e14dcfSSatish Balay   ierr = VecDestroy(&tron->R); CHKERRQ(ierr);
24*a7e14dcfSSatish Balay   ierr = VecDestroy(&tron->diag); CHKERRQ(ierr);
25*a7e14dcfSSatish Balay   ierr = VecScatterDestroy(&tron->scatter); CHKERRQ(ierr);
26*a7e14dcfSSatish Balay   ierr = ISDestroy(&tron->Free_Local); CHKERRQ(ierr);
27*a7e14dcfSSatish Balay   ierr = MatDestroy(&tron->H_sub); CHKERRQ(ierr);
28*a7e14dcfSSatish Balay   ierr = MatDestroy(&tron->Hpre_sub); CHKERRQ(ierr);
29*a7e14dcfSSatish Balay   ierr = PetscFree(tao->data);
30*a7e14dcfSSatish Balay   tao->data = PETSC_NULL;
31*a7e14dcfSSatish Balay 
32*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
33*a7e14dcfSSatish Balay }
34*a7e14dcfSSatish Balay 
35*a7e14dcfSSatish Balay /*------------------------------------------------------------*/
36*a7e14dcfSSatish Balay #undef __FUNCT__
37*a7e14dcfSSatish Balay #define __FUNCT__ "TaoSetFromOptions_TRON"
38*a7e14dcfSSatish Balay static PetscErrorCode TaoSetFromOptions_TRON(TaoSolver tao)
39*a7e14dcfSSatish Balay {
40*a7e14dcfSSatish Balay   TAO_TRON  *tron = (TAO_TRON *)tao->data;
41*a7e14dcfSSatish Balay   PetscErrorCode        ierr;
42*a7e14dcfSSatish Balay   PetscBool flg;
43*a7e14dcfSSatish Balay 
44*a7e14dcfSSatish Balay   PetscFunctionBegin;
45*a7e14dcfSSatish Balay 
46*a7e14dcfSSatish Balay   ierr = PetscOptionsHead("Newton Trust Region Method for bound constrained optimization");CHKERRQ(ierr);
47*a7e14dcfSSatish Balay 
48*a7e14dcfSSatish Balay   ierr = PetscOptionsInt("-tron_maxgpits","maximum number of gradient projections per TRON iterate","TaoSetMaxGPIts",tron->maxgpits,&tron->maxgpits,&flg);
49*a7e14dcfSSatish Balay   CHKERRQ(ierr);
50*a7e14dcfSSatish Balay 
51*a7e14dcfSSatish Balay   ierr = PetscOptionsTail();CHKERRQ(ierr);
52*a7e14dcfSSatish Balay   ierr = TaoLineSearchSetFromOptions(tao->linesearch);CHKERRQ(ierr);
53*a7e14dcfSSatish Balay   ierr = KSPSetFromOptions(tao->ksp); CHKERRQ(ierr);
54*a7e14dcfSSatish Balay 
55*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
56*a7e14dcfSSatish Balay }
57*a7e14dcfSSatish Balay 
58*a7e14dcfSSatish Balay /*------------------------------------------------------------*/
59*a7e14dcfSSatish Balay #undef __FUNCT__
60*a7e14dcfSSatish Balay #define __FUNCT__ "TaoView_TRON"
61*a7e14dcfSSatish Balay static PetscErrorCode TaoView_TRON(TaoSolver tao, PetscViewer viewer)
62*a7e14dcfSSatish Balay {
63*a7e14dcfSSatish Balay   TAO_TRON  *tron = (TAO_TRON *)tao->data;
64*a7e14dcfSSatish Balay   PetscBool isascii;
65*a7e14dcfSSatish Balay   PetscErrorCode   ierr;
66*a7e14dcfSSatish Balay 
67*a7e14dcfSSatish Balay   PetscFunctionBegin;
68*a7e14dcfSSatish Balay   ierr = PetscObjectTypeCompare((PetscObject)viewer,PETSCVIEWERASCII,&isascii);CHKERRQ(ierr);
69*a7e14dcfSSatish Balay   if (isascii) {
70*a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPushTab(viewer); CHKERRQ(ierr);
71*a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPrintf(viewer,"Total PG its: %D,",tron->total_gp_its);CHKERRQ(ierr);
72*a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPrintf(viewer,"PG tolerance: %G \n",tron->pg_ftol);CHKERRQ(ierr);
73*a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPopTab(viewer); CHKERRQ(ierr);
74*a7e14dcfSSatish Balay   } else {
75*a7e14dcfSSatish Balay     SETERRQ1(((PetscObject)tao)->comm,PETSC_ERR_SUP,"Viewer type %s not supported for TAO TRON",((PetscObject)viewer)->type_name);
76*a7e14dcfSSatish Balay   }
77*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
78*a7e14dcfSSatish Balay }
79*a7e14dcfSSatish Balay 
80*a7e14dcfSSatish Balay 
81*a7e14dcfSSatish Balay /* ---------------------------------------------------------- */
82*a7e14dcfSSatish Balay #undef __FUNCT__
83*a7e14dcfSSatish Balay #define __FUNCT__ "TaoSetup_TRON"
84*a7e14dcfSSatish Balay static PetscErrorCode TaoSetup_TRON(TaoSolver tao)
85*a7e14dcfSSatish Balay {
86*a7e14dcfSSatish Balay   PetscErrorCode ierr;
87*a7e14dcfSSatish Balay   TAO_TRON *tron = (TAO_TRON *)tao->data;
88*a7e14dcfSSatish Balay 
89*a7e14dcfSSatish Balay   PetscFunctionBegin;
90*a7e14dcfSSatish Balay 
91*a7e14dcfSSatish Balay   /* Allocate some arrays */
92*a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution, &tron->diag); CHKERRQ(ierr);
93*a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution, &tron->X_New); CHKERRQ(ierr);
94*a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution, &tron->G_New); CHKERRQ(ierr);
95*a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution, &tron->Work); CHKERRQ(ierr);
96*a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution, &tao->gradient); CHKERRQ(ierr);
97*a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution, &tao->stepdirection); CHKERRQ(ierr);
98*a7e14dcfSSatish Balay   if (!tao->XL) {
99*a7e14dcfSSatish Balay       ierr = VecDuplicate(tao->solution, &tao->XL); CHKERRQ(ierr);
100*a7e14dcfSSatish Balay       ierr = VecSet(tao->XL, TAO_NINFINITY); CHKERRQ(ierr);
101*a7e14dcfSSatish Balay   }
102*a7e14dcfSSatish Balay   if (!tao->XU) {
103*a7e14dcfSSatish Balay       ierr = VecDuplicate(tao->solution, &tao->XU); CHKERRQ(ierr);
104*a7e14dcfSSatish Balay       ierr = VecSet(tao->XU, TAO_INFINITY); CHKERRQ(ierr);
105*a7e14dcfSSatish Balay   }
106*a7e14dcfSSatish Balay 
107*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
108*a7e14dcfSSatish Balay }
109*a7e14dcfSSatish Balay 
110*a7e14dcfSSatish Balay 
111*a7e14dcfSSatish Balay 
112*a7e14dcfSSatish Balay #undef __FUNCT__
113*a7e14dcfSSatish Balay #define __FUNCT__ "TaoSolve_TRON"
114*a7e14dcfSSatish Balay static PetscErrorCode TaoSolve_TRON(TaoSolver tao){
115*a7e14dcfSSatish Balay 
116*a7e14dcfSSatish Balay   TAO_TRON *tron = (TAO_TRON *)tao->data;
117*a7e14dcfSSatish Balay   PetscErrorCode ierr;
118*a7e14dcfSSatish Balay   PetscInt iter=0,its;
119*a7e14dcfSSatish Balay 
120*a7e14dcfSSatish Balay   TaoSolverTerminationReason reason = TAO_CONTINUE_ITERATING;
121*a7e14dcfSSatish Balay   TaoLineSearchTerminationReason ls_reason = TAOLINESEARCH_CONTINUE_ITERATING;
122*a7e14dcfSSatish Balay   PetscReal prered,actred,delta,f,f_new,rhok,gdx,xdiff,stepsize;
123*a7e14dcfSSatish Balay   PetscFunctionBegin;
124*a7e14dcfSSatish Balay 
125*a7e14dcfSSatish Balay   tron->pgstepsize=1.0;
126*a7e14dcfSSatish Balay 
127*a7e14dcfSSatish Balay   tao->trust = tao->trust0;
128*a7e14dcfSSatish Balay   /*   Project the current point onto the feasible set */
129*a7e14dcfSSatish Balay   ierr = TaoComputeVariableBounds(tao); CHKERRQ(ierr);
130*a7e14dcfSSatish Balay   ierr = VecMedian(tao->XL,tao->solution,tao->XU,tao->solution); CHKERRQ(ierr);
131*a7e14dcfSSatish Balay   ierr = TaoLineSearchSetVariableBounds(tao->linesearch,tao->XL,tao->XU); CHKERRQ(ierr);
132*a7e14dcfSSatish Balay 
133*a7e14dcfSSatish Balay 
134*a7e14dcfSSatish Balay   ierr = TaoComputeObjectiveAndGradient(tao,tao->solution,&tron->f,tao->gradient);CHKERRQ(ierr);
135*a7e14dcfSSatish Balay   if (tron->Free_Local) {
136*a7e14dcfSSatish Balay     ierr = ISDestroy(&tron->Free_Local); CHKERRQ(ierr);
137*a7e14dcfSSatish Balay     tron->Free_Local = PETSC_NULL;
138*a7e14dcfSSatish Balay   }
139*a7e14dcfSSatish Balay   ierr = VecWhichBetween(tao->XL,tao->solution,tao->XU,&tron->Free_Local); CHKERRQ(ierr);
140*a7e14dcfSSatish Balay 
141*a7e14dcfSSatish Balay   /* Project the gradient and calculate the norm */
142*a7e14dcfSSatish Balay   ierr = VecBoundGradientProjection(tao->gradient,tao->solution, tao->XL, tao->XU, tao->gradient); CHKERRQ(ierr);
143*a7e14dcfSSatish Balay   ierr = VecNorm(tao->gradient,NORM_2,&tron->gnorm); CHKERRQ(ierr);
144*a7e14dcfSSatish Balay 
145*a7e14dcfSSatish Balay   if (PetscIsInfOrNanReal(tron->f) || PetscIsInfOrNanReal(tron->gnorm)) {
146*a7e14dcfSSatish Balay     SETERRQ(PETSC_COMM_SELF,1, "User provided compute function generated Inf pr NaN");
147*a7e14dcfSSatish Balay   }
148*a7e14dcfSSatish Balay 
149*a7e14dcfSSatish Balay   if (tao->trust <= 0) {
150*a7e14dcfSSatish Balay     tao->trust=PetscMax(tron->gnorm*tron->gnorm,1.0);
151*a7e14dcfSSatish Balay   }
152*a7e14dcfSSatish Balay 
153*a7e14dcfSSatish Balay   tron->stepsize=tao->trust;
154*a7e14dcfSSatish Balay   ierr = TaoMonitor(tao, iter, tron->f, tron->gnorm, 0.0, tron->stepsize, &reason); CHKERRQ(ierr);
155*a7e14dcfSSatish Balay   while (reason==TAO_CONTINUE_ITERATING){
156*a7e14dcfSSatish Balay 
157*a7e14dcfSSatish Balay     ierr = TronGradientProjections(tao,tron); CHKERRQ(ierr);
158*a7e14dcfSSatish Balay     f=tron->f; delta=tao->trust;
159*a7e14dcfSSatish Balay     tron->n_free_last = tron->n_free;
160*a7e14dcfSSatish Balay     ierr = TaoComputeHessian(tao,tao->solution,&tao->hessian, &tao->hessian_pre, &tron->matflag);CHKERRQ(ierr);
161*a7e14dcfSSatish Balay 
162*a7e14dcfSSatish Balay     ierr = ISGetSize(tron->Free_Local, &tron->n_free);  CHKERRQ(ierr);
163*a7e14dcfSSatish Balay 
164*a7e14dcfSSatish Balay     /* If no free variables */
165*a7e14dcfSSatish Balay     if (tron->n_free == 0) {
166*a7e14dcfSSatish Balay       actred=0;
167*a7e14dcfSSatish Balay       PetscInfo(tao,"No free variables in tron iteration.");
168*a7e14dcfSSatish Balay       break;
169*a7e14dcfSSatish Balay 
170*a7e14dcfSSatish Balay     }
171*a7e14dcfSSatish Balay     /* use free_local to mask/submat gradient, hessian, stepdirection */
172*a7e14dcfSSatish Balay     ierr = VecGetSubVec(tao->gradient,tron->Free_Local,tao->subset_type,0.0,&tron->R);
173*a7e14dcfSSatish Balay     CHKERRQ(ierr);
174*a7e14dcfSSatish Balay     ierr = VecGetSubVec(tao->gradient,tron->Free_Local,tao->subset_type,0.0,&tron->DXFree);
175*a7e14dcfSSatish Balay     CHKERRQ(ierr);
176*a7e14dcfSSatish Balay     ierr = VecSet(tron->DXFree,0.0); CHKERRQ(ierr);
177*a7e14dcfSSatish Balay     ierr = VecScale(tron->R, -1.0); CHKERRQ(ierr);
178*a7e14dcfSSatish Balay     ierr = MatGetSubMat(tao->hessian, tron->Free_Local, tron->diag, tao->subset_type, &tron->H_sub); CHKERRQ(ierr);
179*a7e14dcfSSatish Balay     if (tao->hessian == tao->hessian_pre) {
180*a7e14dcfSSatish Balay       ierr = MatDestroy(&tron->Hpre_sub); CHKERRQ(ierr);
181*a7e14dcfSSatish Balay       ierr = PetscObjectReference((PetscObject)(tron->H_sub)); CHKERRQ(ierr);
182*a7e14dcfSSatish Balay       tron->Hpre_sub = tron->H_sub;
183*a7e14dcfSSatish Balay     } else {
184*a7e14dcfSSatish Balay       ierr = MatGetSubMat(tao->hessian_pre, tron->Free_Local, tron->diag, tao->subset_type,&tron->Hpre_sub);
185*a7e14dcfSSatish Balay       CHKERRQ(ierr);
186*a7e14dcfSSatish Balay     }
187*a7e14dcfSSatish Balay     ierr = KSPReset(tao->ksp); CHKERRQ(ierr);
188*a7e14dcfSSatish Balay     ierr = KSPSetOperators(tao->ksp, tron->H_sub, tron->Hpre_sub, tron->matflag); CHKERRQ(ierr);
189*a7e14dcfSSatish Balay     while (1) {
190*a7e14dcfSSatish Balay 
191*a7e14dcfSSatish Balay       /* Approximately solve the reduced linear system */
192*a7e14dcfSSatish Balay       ierr = KSPSTCGSetRadius(tao->ksp,delta); CHKERRQ(ierr);
193*a7e14dcfSSatish Balay       ierr = KSPSolve(tao->ksp, tron->R, tron->DXFree); CHKERRQ(ierr);
194*a7e14dcfSSatish Balay       ierr = KSPGetIterationNumber(tao->ksp,&its); CHKERRQ(ierr);
195*a7e14dcfSSatish Balay       tao->ksp_its+=its;
196*a7e14dcfSSatish Balay       ierr = VecSet(tao->stepdirection,0.0); CHKERRQ(ierr);
197*a7e14dcfSSatish Balay 
198*a7e14dcfSSatish Balay       /* Add dxfree matrix to compute step direction vector */
199*a7e14dcfSSatish Balay       ierr = VecReducedXPY(tao->stepdirection,tron->DXFree,tron->Free_Local); CHKERRQ(ierr);
200*a7e14dcfSSatish Balay       if (0) {
201*a7e14dcfSSatish Balay 	PetscReal rhs,stepnorm;
202*a7e14dcfSSatish Balay 	ierr = VecNorm(tron->R,NORM_2,&rhs); CHKERRQ(ierr);
203*a7e14dcfSSatish Balay 	ierr = VecNorm(tron->DXFree,NORM_2,&stepnorm); CHKERRQ(ierr);
204*a7e14dcfSSatish Balay 	ierr = PetscPrintf(PETSC_COMM_WORLD,"|rhs|=%g\t|s|=%g\n",rhs,stepnorm); CHKERRQ(ierr);
205*a7e14dcfSSatish Balay       }
206*a7e14dcfSSatish Balay 
207*a7e14dcfSSatish Balay 
208*a7e14dcfSSatish Balay       ierr = VecDot(tao->gradient, tao->stepdirection, &gdx); CHKERRQ(ierr);
209*a7e14dcfSSatish Balay       ierr = PetscInfo1(tao,"Expected decrease in function value: %14.12e\n",gdx);
210*a7e14dcfSSatish Balay       CHKERRQ(ierr);
211*a7e14dcfSSatish Balay 
212*a7e14dcfSSatish Balay       ierr = VecCopy(tao->solution, tron->X_New); CHKERRQ(ierr);
213*a7e14dcfSSatish Balay       ierr = VecCopy(tao->gradient, tron->G_New); CHKERRQ(ierr);
214*a7e14dcfSSatish Balay 
215*a7e14dcfSSatish Balay       stepsize=1.0;f_new=f;
216*a7e14dcfSSatish Balay 
217*a7e14dcfSSatish Balay       ierr = TaoLineSearchSetInitialStepLength(tao->linesearch,1.0); CHKERRQ(ierr);
218*a7e14dcfSSatish Balay       ierr = TaoLineSearchApply(tao->linesearch, tron->X_New, &f_new, tron->G_New, tao->stepdirection,
219*a7e14dcfSSatish Balay 				&stepsize,&ls_reason); CHKERRQ(ierr); CHKERRQ(ierr);
220*a7e14dcfSSatish Balay       ierr = TaoAddLineSearchCounts(tao); CHKERRQ(ierr);
221*a7e14dcfSSatish Balay 
222*a7e14dcfSSatish Balay       ierr = MatMult(tao->hessian, tao->stepdirection, tron->Work); CHKERRQ(ierr);
223*a7e14dcfSSatish Balay       ierr = VecAYPX(tron->Work, 0.5, tao->gradient); CHKERRQ(ierr);
224*a7e14dcfSSatish Balay       ierr = VecDot(tao->stepdirection, tron->Work, &prered); CHKERRQ(ierr);
225*a7e14dcfSSatish Balay       actred = f_new - f;
226*a7e14dcfSSatish Balay       if (actred<0) {
227*a7e14dcfSSatish Balay 	rhok=PetscAbs(-actred/prered);
228*a7e14dcfSSatish Balay       } else {
229*a7e14dcfSSatish Balay 	rhok=0.0;
230*a7e14dcfSSatish Balay       }
231*a7e14dcfSSatish Balay 
232*a7e14dcfSSatish Balay       /* Compare actual improvement to the quadratic model */
233*a7e14dcfSSatish Balay       if (rhok > tron->eta1) { /* Accept the point */
234*a7e14dcfSSatish Balay 	/* d = x_new - x */
235*a7e14dcfSSatish Balay 	ierr = VecCopy(tron->X_New, tao->stepdirection); CHKERRQ(ierr);
236*a7e14dcfSSatish Balay 	ierr = VecAXPY(tao->stepdirection, -1.0, tao->solution); CHKERRQ(ierr);
237*a7e14dcfSSatish Balay 
238*a7e14dcfSSatish Balay 	ierr = VecNorm(tao->stepdirection, NORM_2, &xdiff); CHKERRQ(ierr);
239*a7e14dcfSSatish Balay 	xdiff *= stepsize;
240*a7e14dcfSSatish Balay 
241*a7e14dcfSSatish Balay 	/* Adjust trust region size */
242*a7e14dcfSSatish Balay 	if (rhok < tron->eta2 ){
243*a7e14dcfSSatish Balay 	  delta = PetscMin(xdiff,delta)*tron->sigma1;
244*a7e14dcfSSatish Balay 	} else if (rhok > tron->eta4 ){
245*a7e14dcfSSatish Balay 	  delta= PetscMin(xdiff,delta)*tron->sigma3;
246*a7e14dcfSSatish Balay 	} else if (rhok > tron->eta3 ){
247*a7e14dcfSSatish Balay 	  delta=PetscMin(xdiff,delta)*tron->sigma2;
248*a7e14dcfSSatish Balay 	}
249*a7e14dcfSSatish Balay 	ierr = VecBoundGradientProjection(tron->G_New,tron->X_New, tao->XL, tao->XU, tao->gradient); CHKERRQ(ierr);
250*a7e14dcfSSatish Balay 	if (tron->Free_Local) {
251*a7e14dcfSSatish Balay 	  ierr = ISDestroy(&tron->Free_Local); CHKERRQ(ierr);
252*a7e14dcfSSatish Balay 	  tron->Free_Local=PETSC_NULL;
253*a7e14dcfSSatish Balay 	}
254*a7e14dcfSSatish Balay 	ierr = VecWhichBetween(tao->XL, tron->X_New, tao->XU, &tron->Free_Local); CHKERRQ(ierr);
255*a7e14dcfSSatish Balay 	f=f_new;
256*a7e14dcfSSatish Balay 	ierr = VecNorm(tao->gradient,NORM_2,&tron->gnorm); CHKERRQ(ierr);
257*a7e14dcfSSatish Balay 	ierr = VecCopy(tron->X_New, tao->solution); CHKERRQ(ierr);
258*a7e14dcfSSatish Balay 	ierr = VecCopy(tron->G_New, tao->gradient); CHKERRQ(ierr);
259*a7e14dcfSSatish Balay 	break;
260*a7e14dcfSSatish Balay       }
261*a7e14dcfSSatish Balay       else if (delta <= 1e-30) {
262*a7e14dcfSSatish Balay 	break;
263*a7e14dcfSSatish Balay       }
264*a7e14dcfSSatish Balay       else {
265*a7e14dcfSSatish Balay 	delta /= 4.0;
266*a7e14dcfSSatish Balay       }
267*a7e14dcfSSatish Balay     } /* end linear solve loop */
268*a7e14dcfSSatish Balay 
269*a7e14dcfSSatish Balay 
270*a7e14dcfSSatish Balay     tron->f=f; tron->actred=actred; tao->trust=delta;
271*a7e14dcfSSatish Balay     iter++;
272*a7e14dcfSSatish Balay     ierr = TaoMonitor(tao, iter, tron->f, tron->gnorm, 0.0, delta, &reason); CHKERRQ(ierr);
273*a7e14dcfSSatish Balay   }  /* END MAIN LOOP  */
274*a7e14dcfSSatish Balay 
275*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
276*a7e14dcfSSatish Balay }
277*a7e14dcfSSatish Balay 
278*a7e14dcfSSatish Balay 
279*a7e14dcfSSatish Balay #undef __FUNCT__
280*a7e14dcfSSatish Balay #define __FUNCT__ "TronGradientProjections"
281*a7e14dcfSSatish Balay static PetscErrorCode TronGradientProjections(TaoSolver tao,TAO_TRON *tron)
282*a7e14dcfSSatish Balay {
283*a7e14dcfSSatish Balay   PetscErrorCode ierr;
284*a7e14dcfSSatish Balay   PetscInt i;
285*a7e14dcfSSatish Balay   TaoLineSearchTerminationReason ls_reason;
286*a7e14dcfSSatish Balay   PetscReal actred=-1.0,actred_max=0.0;
287*a7e14dcfSSatish Balay   PetscReal f_new;
288*a7e14dcfSSatish Balay   /*
289*a7e14dcfSSatish Balay      The gradient and function value passed into and out of this
290*a7e14dcfSSatish Balay      routine should be current and correct.
291*a7e14dcfSSatish Balay 
292*a7e14dcfSSatish Balay      The free, active, and binding variables should be already identified
293*a7e14dcfSSatish Balay   */
294*a7e14dcfSSatish Balay 
295*a7e14dcfSSatish Balay   PetscFunctionBegin;
296*a7e14dcfSSatish Balay   if (tron->Free_Local) {
297*a7e14dcfSSatish Balay     ierr = ISDestroy(&tron->Free_Local); CHKERRQ(ierr);
298*a7e14dcfSSatish Balay     tron->Free_Local = PETSC_NULL;
299*a7e14dcfSSatish Balay   }
300*a7e14dcfSSatish Balay   ierr = VecWhichBetween(tao->XL,tao->solution,tao->XU,&tron->Free_Local); CHKERRQ(ierr);
301*a7e14dcfSSatish Balay 
302*a7e14dcfSSatish Balay   for (i=0;i<tron->maxgpits;i++){
303*a7e14dcfSSatish Balay 
304*a7e14dcfSSatish Balay     if ( -actred <= (tron->pg_ftol)*actred_max) break;
305*a7e14dcfSSatish Balay 
306*a7e14dcfSSatish Balay     tron->gp_iterates++; tron->total_gp_its++;
307*a7e14dcfSSatish Balay     f_new=tron->f;
308*a7e14dcfSSatish Balay 
309*a7e14dcfSSatish Balay     ierr = VecCopy(tao->gradient, tao->stepdirection); CHKERRQ(ierr);
310*a7e14dcfSSatish Balay     ierr = VecScale(tao->stepdirection, -1.0); CHKERRQ(ierr);
311*a7e14dcfSSatish Balay     ierr = TaoLineSearchSetInitialStepLength(tao->linesearch,tron->pgstepsize); CHKERRQ(ierr);
312*a7e14dcfSSatish Balay     ierr = TaoLineSearchApply(tao->linesearch, tao->solution, &f_new, tao->gradient, tao->stepdirection,
313*a7e14dcfSSatish Balay 			      &tron->pgstepsize, &ls_reason); CHKERRQ(ierr);
314*a7e14dcfSSatish Balay     ierr = TaoAddLineSearchCounts(tao); CHKERRQ(ierr);
315*a7e14dcfSSatish Balay 
316*a7e14dcfSSatish Balay 
317*a7e14dcfSSatish Balay     /* Update the iterate */
318*a7e14dcfSSatish Balay     actred = f_new - tron->f;
319*a7e14dcfSSatish Balay     actred_max = PetscMax(actred_max,-(f_new - tron->f));
320*a7e14dcfSSatish Balay     tron->f = f_new;
321*a7e14dcfSSatish Balay     if (tron->Free_Local) {
322*a7e14dcfSSatish Balay       ierr = ISDestroy(&tron->Free_Local); CHKERRQ(ierr);
323*a7e14dcfSSatish Balay       tron->Free_Local = PETSC_NULL;
324*a7e14dcfSSatish Balay     }
325*a7e14dcfSSatish Balay     ierr = VecWhichBetween(tao->XL,tao->solution,tao->XU,&tron->Free_Local); CHKERRQ(ierr);
326*a7e14dcfSSatish Balay   }
327*a7e14dcfSSatish Balay 
328*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
329*a7e14dcfSSatish Balay }
330*a7e14dcfSSatish Balay 
331*a7e14dcfSSatish Balay #undef __FUNCT__
332*a7e14dcfSSatish Balay #define __FUNCT__ "TaoComputeDual_TRON"
333*a7e14dcfSSatish Balay static PetscErrorCode TaoComputeDual_TRON(TaoSolver tao, Vec DXL, Vec DXU) {
334*a7e14dcfSSatish Balay 
335*a7e14dcfSSatish Balay   TAO_TRON *tron = (TAO_TRON *)tao->data;
336*a7e14dcfSSatish Balay   PetscErrorCode       ierr;
337*a7e14dcfSSatish Balay 
338*a7e14dcfSSatish Balay   PetscFunctionBegin;
339*a7e14dcfSSatish Balay 
340*a7e14dcfSSatish Balay   PetscValidHeaderSpecific(tao,TAOSOLVER_CLASSID,1);
341*a7e14dcfSSatish Balay   PetscValidHeaderSpecific(DXL,VEC_CLASSID,2);
342*a7e14dcfSSatish Balay   PetscValidHeaderSpecific(DXU,VEC_CLASSID,3);
343*a7e14dcfSSatish Balay 
344*a7e14dcfSSatish Balay   if (!tron->Work || !tao->gradient) {
345*a7e14dcfSSatish Balay       SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ORDER,"Dual variables don't exist yet or no longer exist.\n");
346*a7e14dcfSSatish Balay   }
347*a7e14dcfSSatish Balay 
348*a7e14dcfSSatish Balay   ierr = VecBoundGradientProjection(tao->gradient,tao->solution,tao->XL,tao->XU,tron->Work); CHKERRQ(ierr);
349*a7e14dcfSSatish Balay   ierr = VecCopy(tron->Work,DXL); CHKERRQ(ierr);
350*a7e14dcfSSatish Balay   ierr = VecAXPY(DXL,-1.0,tao->gradient); CHKERRQ(ierr);
351*a7e14dcfSSatish Balay   ierr = VecSet(DXU,0.0); CHKERRQ(ierr);
352*a7e14dcfSSatish Balay   ierr = VecPointwiseMax(DXL,DXL,DXU); CHKERRQ(ierr);
353*a7e14dcfSSatish Balay 
354*a7e14dcfSSatish Balay   ierr = VecCopy(tao->gradient,DXU); CHKERRQ(ierr);
355*a7e14dcfSSatish Balay   ierr = VecAXPY(DXU,-1.0,tron->Work); CHKERRQ(ierr);
356*a7e14dcfSSatish Balay   ierr = VecSet(tron->Work,0.0); CHKERRQ(ierr);
357*a7e14dcfSSatish Balay   ierr = VecPointwiseMin(DXU,tron->Work,DXU); CHKERRQ(ierr);
358*a7e14dcfSSatish Balay 
359*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
360*a7e14dcfSSatish Balay }
361*a7e14dcfSSatish Balay 
362*a7e14dcfSSatish Balay /*------------------------------------------------------------*/
363*a7e14dcfSSatish Balay EXTERN_C_BEGIN
364*a7e14dcfSSatish Balay #undef __FUNCT__
365*a7e14dcfSSatish Balay #define __FUNCT__ "TaoCreate_TRON"
366*a7e14dcfSSatish Balay PetscErrorCode TaoCreate_TRON(TaoSolver tao)
367*a7e14dcfSSatish Balay {
368*a7e14dcfSSatish Balay   TAO_TRON *tron;
369*a7e14dcfSSatish Balay   PetscErrorCode   ierr;
370*a7e14dcfSSatish Balay   const char *morethuente_type = TAOLINESEARCH_MT;
371*a7e14dcfSSatish Balay   PetscFunctionBegin;
372*a7e14dcfSSatish Balay 
373*a7e14dcfSSatish Balay   tao->ops->setup = TaoSetup_TRON;
374*a7e14dcfSSatish Balay   tao->ops->solve = TaoSolve_TRON;
375*a7e14dcfSSatish Balay   tao->ops->view = TaoView_TRON;
376*a7e14dcfSSatish Balay   tao->ops->setfromoptions = TaoSetFromOptions_TRON;
377*a7e14dcfSSatish Balay   tao->ops->destroy = TaoDestroy_TRON;
378*a7e14dcfSSatish Balay   tao->ops->computedual = TaoComputeDual_TRON;
379*a7e14dcfSSatish Balay 
380*a7e14dcfSSatish Balay   ierr = PetscNewLog(tao,TAO_TRON,&tron); CHKERRQ(ierr);
381*a7e14dcfSSatish Balay 
382*a7e14dcfSSatish Balay   tao->max_it = 50;
383*a7e14dcfSSatish Balay   tao->fatol = 1e-10;
384*a7e14dcfSSatish Balay   tao->frtol = 1e-10;
385*a7e14dcfSSatish Balay   tao->data = (void*)tron;
386*a7e14dcfSSatish Balay   tao->steptol = 1e-12;
387*a7e14dcfSSatish Balay   tao->trust0       = 1.0;
388*a7e14dcfSSatish Balay 
389*a7e14dcfSSatish Balay   /* Initialize pointers and variables */
390*a7e14dcfSSatish Balay   tron->n            = 0;
391*a7e14dcfSSatish Balay   tron->maxgpits     = 3;
392*a7e14dcfSSatish Balay   tron->pg_ftol      = 0.001;
393*a7e14dcfSSatish Balay 
394*a7e14dcfSSatish Balay   tron->eta1         = 1.0e-4;
395*a7e14dcfSSatish Balay   tron->eta2         = 0.25;
396*a7e14dcfSSatish Balay   tron->eta3         = 0.50;
397*a7e14dcfSSatish Balay   tron->eta4         = 0.90;
398*a7e14dcfSSatish Balay 
399*a7e14dcfSSatish Balay   tron->sigma1       = 0.5;
400*a7e14dcfSSatish Balay   tron->sigma2       = 2.0;
401*a7e14dcfSSatish Balay   tron->sigma3       = 4.0;
402*a7e14dcfSSatish Balay 
403*a7e14dcfSSatish Balay   tron->gp_iterates  = 0; /* Cumulative number */
404*a7e14dcfSSatish Balay   tron->total_gp_its = 0;
405*a7e14dcfSSatish Balay 
406*a7e14dcfSSatish Balay   tron->n_free       = 0;
407*a7e14dcfSSatish Balay 
408*a7e14dcfSSatish Balay   tron->DXFree=PETSC_NULL;
409*a7e14dcfSSatish Balay   tron->R=PETSC_NULL;
410*a7e14dcfSSatish Balay   tron->X_New=PETSC_NULL;
411*a7e14dcfSSatish Balay   tron->G_New=PETSC_NULL;
412*a7e14dcfSSatish Balay   tron->Work=PETSC_NULL;
413*a7e14dcfSSatish Balay   tron->Free_Local=PETSC_NULL;
414*a7e14dcfSSatish Balay   tron->H_sub=PETSC_NULL;
415*a7e14dcfSSatish Balay   tron->Hpre_sub=PETSC_NULL;
416*a7e14dcfSSatish Balay   tao->subset_type = TAO_SUBSET_SUBVEC;
417*a7e14dcfSSatish Balay 
418*a7e14dcfSSatish Balay   ierr = TaoLineSearchCreate(((PetscObject)tao)->comm, &tao->linesearch); CHKERRQ(ierr);
419*a7e14dcfSSatish Balay   ierr = TaoLineSearchSetType(tao->linesearch,morethuente_type); CHKERRQ(ierr);
420*a7e14dcfSSatish Balay   ierr = TaoLineSearchUseTaoSolverRoutines(tao->linesearch,tao); CHKERRQ(ierr);
421*a7e14dcfSSatish Balay 
422*a7e14dcfSSatish Balay   ierr = KSPCreate(((PetscObject)tao)->comm, &tao->ksp); CHKERRQ(ierr);
423*a7e14dcfSSatish Balay   ierr = KSPSetType(tao->ksp,KSPSTCG); CHKERRQ(ierr);
424*a7e14dcfSSatish Balay 
425*a7e14dcfSSatish Balay   PetscFunctionReturn(0);
426*a7e14dcfSSatish Balay }
427*a7e14dcfSSatish Balay EXTERN_C_END
428