xref: /petsc/src/tao/unconstrained/impls/neldermead/neldermead.c (revision 3c9e27cfca911a7d7e3219758be42726e83c4ab2)
1a7e14dcfSSatish Balay #include "neldermead.h"
2a7e14dcfSSatish Balay #include "petscvec.h"
3a7e14dcfSSatish Balay 
4a7e14dcfSSatish Balay static PetscErrorCode NelderMeadSort(TAO_NelderMead *nm);
5a7e14dcfSSatish Balay static PetscErrorCode NelderMeadReplace(TAO_NelderMead *nm, PetscInt index, Vec Xmu, PetscReal f);
6a7e14dcfSSatish Balay /* ---------------------------------------------------------- */
7a7e14dcfSSatish Balay #undef __FUNCT__
8a7e14dcfSSatish Balay #define __FUNCT__ "TaoSetUp_NM"
9a7e14dcfSSatish Balay static PetscErrorCode TaoSetUp_NM(TaoSolver tao)
10a7e14dcfSSatish Balay {
11a7e14dcfSSatish Balay   PetscErrorCode ierr;
12a7e14dcfSSatish Balay   TAO_NelderMead *nm = (TAO_NelderMead *)tao->data;
13a7e14dcfSSatish Balay   PetscInt size;
14a7e14dcfSSatish Balay 
15a7e14dcfSSatish Balay   PetscFunctionBegin;
16a7e14dcfSSatish Balay   ierr = VecGetSize(tao->solution,&size); CHKERRQ(ierr);
17a7e14dcfSSatish Balay   nm->N = size;
18a7e14dcfSSatish Balay   nm->oneOverN = 1.0/size;
19a7e14dcfSSatish Balay   ierr = VecDuplicateVecs(tao->solution,nm->N+1,&nm->simplex); CHKERRQ(ierr);
20a7e14dcfSSatish Balay   ierr = PetscMalloc((nm->N+1)*sizeof(PetscReal),&nm->f_values); CHKERRQ(ierr);
21a7e14dcfSSatish Balay   ierr = PetscMalloc((nm->N+1)*sizeof(PetscInt),&nm->indices); CHKERRQ(ierr);
22a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution,&nm->Xbar); CHKERRQ(ierr);
23a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution,&nm->Xmur); CHKERRQ(ierr);
24a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution,&nm->Xmue); CHKERRQ(ierr);
25a7e14dcfSSatish Balay   ierr = VecDuplicate(tao->solution,&nm->Xmuc); CHKERRQ(ierr);
26a7e14dcfSSatish Balay 
27a7e14dcfSSatish Balay   tao->gradient=0;
28a7e14dcfSSatish Balay   tao->step=0;
29a7e14dcfSSatish Balay 
30a7e14dcfSSatish Balay   PetscFunctionReturn(0);
31a7e14dcfSSatish Balay }
32a7e14dcfSSatish Balay 
33a7e14dcfSSatish Balay /* ---------------------------------------------------------- */
34a7e14dcfSSatish Balay #undef __FUNCT__
35a7e14dcfSSatish Balay #define __FUNCT__ "TaoDestroy_NM"
36a7e14dcfSSatish Balay PetscErrorCode TaoDestroy_NM(TaoSolver tao)
37a7e14dcfSSatish Balay {
38a7e14dcfSSatish Balay   TAO_NelderMead *nm = (TAO_NelderMead*)tao->data;
39a7e14dcfSSatish Balay   PetscErrorCode ierr;
40a7e14dcfSSatish Balay   PetscFunctionBegin;
41a7e14dcfSSatish Balay   if (tao->setupcalled) {
42a7e14dcfSSatish Balay     ierr = VecDestroyVecs(nm->N+1,&nm->simplex); CHKERRQ(ierr);
43a7e14dcfSSatish Balay     ierr = VecDestroy(&nm->Xmuc); CHKERRQ(ierr);
44a7e14dcfSSatish Balay     ierr = VecDestroy(&nm->Xmue); CHKERRQ(ierr);
45a7e14dcfSSatish Balay     ierr = VecDestroy(&nm->Xmur); CHKERRQ(ierr);
46a7e14dcfSSatish Balay     ierr = VecDestroy(&nm->Xbar); CHKERRQ(ierr);
47a7e14dcfSSatish Balay   }
48a7e14dcfSSatish Balay   ierr = PetscFree(nm->indices); CHKERRQ(ierr);
49a7e14dcfSSatish Balay   ierr = PetscFree(nm->f_values); CHKERRQ(ierr);
50a7e14dcfSSatish Balay   ierr = PetscFree(tao->data);
51a7e14dcfSSatish Balay   tao->data = 0;
52a7e14dcfSSatish Balay   PetscFunctionReturn(0);
53a7e14dcfSSatish Balay }
54a7e14dcfSSatish Balay 
55a7e14dcfSSatish Balay /*------------------------------------------------------------*/
56a7e14dcfSSatish Balay #undef __FUNCT__
57a7e14dcfSSatish Balay #define __FUNCT__ "TaoSetFromOptions_NM"
58a7e14dcfSSatish Balay PetscErrorCode TaoSetFromOptions_NM(TaoSolver tao)
59a7e14dcfSSatish Balay {
60a7e14dcfSSatish Balay 
61a7e14dcfSSatish Balay   TAO_NelderMead *nm = (TAO_NelderMead*)tao->data;
62a7e14dcfSSatish Balay   PetscBool flg;
63a7e14dcfSSatish Balay   PetscErrorCode ierr;
64a7e14dcfSSatish Balay 
65a7e14dcfSSatish Balay   PetscFunctionBegin;
66a7e14dcfSSatish Balay   ierr = PetscOptionsHead("Nelder-Mead options"); CHKERRQ(ierr);
67a7e14dcfSSatish Balay 
68a7e14dcfSSatish Balay   ierr = PetscOptionsReal("-tao_nm_lamda","initial step length","",nm->lamda,&nm->lamda,&flg);  CHKERRQ(ierr);
69a7e14dcfSSatish Balay 
70a7e14dcfSSatish Balay   ierr = PetscOptionsReal("-tao_nm_mu","mu","",nm->mu_oc,&nm->mu_oc,&flg); CHKERRQ(ierr);
71a7e14dcfSSatish Balay   nm->mu_ic = -nm->mu_oc;
72a7e14dcfSSatish Balay   nm->mu_r = nm->mu_oc*2.0;
73a7e14dcfSSatish Balay   nm->mu_e = nm->mu_oc*4.0;
74a7e14dcfSSatish Balay 
75a7e14dcfSSatish Balay   ierr = PetscOptionsTail(); CHKERRQ(ierr);
76a7e14dcfSSatish Balay   PetscFunctionReturn(0);
77a7e14dcfSSatish Balay }
78a7e14dcfSSatish Balay 
79a7e14dcfSSatish Balay /*------------------------------------------------------------*/
80a7e14dcfSSatish Balay #undef __FUNCT__
81a7e14dcfSSatish Balay #define __FUNCT__ "TaoView_NM"
82a7e14dcfSSatish Balay PetscErrorCode TaoView_NM(TaoSolver tao,PetscViewer viewer)
83a7e14dcfSSatish Balay {
84a7e14dcfSSatish Balay   TAO_NelderMead *nm = (TAO_NelderMead*)tao->data;
85a7e14dcfSSatish Balay   PetscBool isascii;
86a7e14dcfSSatish Balay   PetscErrorCode ierr;
87a7e14dcfSSatish Balay 
88a7e14dcfSSatish Balay   PetscFunctionBegin;
89a7e14dcfSSatish Balay   ierr = PetscObjectTypeCompare((PetscObject)viewer,PETSCVIEWERASCII,&isascii);CHKERRQ(ierr);
90a7e14dcfSSatish Balay   if (isascii) {
91a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPushTab(viewer); CHKERRQ(ierr);
92a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPrintf(viewer,"expansions: %D\n",nm->nexpand); CHKERRQ(ierr);
93a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPrintf(viewer,"reflections: %D\n",nm->nreflect); CHKERRQ(ierr);
94a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPrintf(viewer,"inside contractions: %D\n",nm->nincontract); CHKERRQ(ierr);
95a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPrintf(viewer,"outside contractionss: %D\n",nm->noutcontract); CHKERRQ(ierr);
96a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPrintf(viewer,"Shrink steps: %D\n",nm->nshrink); CHKERRQ(ierr);
97a7e14dcfSSatish Balay     ierr = PetscViewerASCIIPopTab(viewer); CHKERRQ(ierr);
98a7e14dcfSSatish Balay   } else {
99a7e14dcfSSatish Balay     SETERRQ1(((PetscObject)tao)->comm,PETSC_ERR_SUP,"Viewer type %s not supported for TAO NelderMead",((PetscObject)viewer)->type_name);
100a7e14dcfSSatish Balay   }
101a7e14dcfSSatish Balay 
102a7e14dcfSSatish Balay   PetscFunctionReturn(0);
103a7e14dcfSSatish Balay }
104a7e14dcfSSatish Balay 
105a7e14dcfSSatish Balay /*------------------------------------------------------------*/
106a7e14dcfSSatish Balay #undef __FUNCT__
107a7e14dcfSSatish Balay #define __FUNCT__ "TaoSolve_NM"
108a7e14dcfSSatish Balay PetscErrorCode TaoSolve_NM(TaoSolver tao)
109a7e14dcfSSatish Balay {
110a7e14dcfSSatish Balay   PetscErrorCode ierr;
111a7e14dcfSSatish Balay   TAO_NelderMead *nm = (TAO_NelderMead*)tao->data;
112a7e14dcfSSatish Balay   TaoSolverTerminationReason reason;
113a7e14dcfSSatish Balay   PetscReal *x;
114a7e14dcfSSatish Balay   PetscInt iter=0,i;
115a7e14dcfSSatish Balay   Vec Xmur=nm->Xmur, Xmue=nm->Xmue, Xmuc=nm->Xmuc, Xbar=nm->Xbar;
116a7e14dcfSSatish Balay   PetscReal fr,fe,fc;
117a7e14dcfSSatish Balay   PetscInt shrink;
118a7e14dcfSSatish Balay   PetscInt low,high;
119a7e14dcfSSatish Balay 
120a7e14dcfSSatish Balay 
121a7e14dcfSSatish Balay   PetscFunctionBegin;
122a7e14dcfSSatish Balay   nm->nshrink =      0;
123a7e14dcfSSatish Balay   nm->nreflect =     0;
124a7e14dcfSSatish Balay   nm->nincontract =  0;
125a7e14dcfSSatish Balay   nm->noutcontract = 0;
126a7e14dcfSSatish Balay   nm->nexpand =      0;
127a7e14dcfSSatish Balay 
128a7e14dcfSSatish Balay   if (tao->XL || tao->XU || tao->ops->computebounds) {
129a7e14dcfSSatish Balay     ierr = PetscPrintf(((PetscObject)tao)->comm,"WARNING: Variable bounds have been set but will be ignored by NelderMead algorithm\n"); CHKERRQ(ierr);
130a7e14dcfSSatish Balay   }
131a7e14dcfSSatish Balay 
132a7e14dcfSSatish Balay   ierr = VecCopy(tao->solution,nm->simplex[0]); CHKERRQ(ierr);
133a7e14dcfSSatish Balay   ierr = TaoComputeObjective(tao,nm->simplex[0],&nm->f_values[0]); CHKERRQ(ierr);
134a7e14dcfSSatish Balay   nm->indices[0]=0;
135a7e14dcfSSatish Balay   for (i=1;i<nm->N+1;i++){
136a7e14dcfSSatish Balay     ierr = VecCopy(tao->solution,nm->simplex[i]); CHKERRQ(ierr);
137a7e14dcfSSatish Balay     ierr = VecGetOwnershipRange(nm->simplex[i],&low,&high); CHKERRQ(ierr);
138a7e14dcfSSatish Balay     if (i-1 >= low && i-1 < high) {
139a7e14dcfSSatish Balay       ierr = VecGetArray(nm->simplex[i],&x); CHKERRQ(ierr);
140a7e14dcfSSatish Balay       x[i-1-low] += nm->lamda;
141a7e14dcfSSatish Balay       ierr = VecRestoreArray(nm->simplex[i],&x); CHKERRQ(ierr);
142a7e14dcfSSatish Balay     }
143a7e14dcfSSatish Balay 
144a7e14dcfSSatish Balay     ierr = TaoComputeObjective(tao,nm->simplex[i],&nm->f_values[i]); CHKERRQ(ierr);
145a7e14dcfSSatish Balay     nm->indices[i] = i;
146a7e14dcfSSatish Balay   }
147a7e14dcfSSatish Balay 
148a7e14dcfSSatish Balay   /*  Xbar  = (Sum of all simplex vectors - worst vector)/N */
149a7e14dcfSSatish Balay   ierr = NelderMeadSort(nm); CHKERRQ(ierr);
150a7e14dcfSSatish Balay   ierr = VecSet(Xbar,0.0); CHKERRQ(ierr);
151a7e14dcfSSatish Balay   for (i=0;i<nm->N;i++) {
152a7e14dcfSSatish Balay     ierr = VecAXPY(Xbar,1.0,nm->simplex[nm->indices[i]]);
153a7e14dcfSSatish Balay   }
154a7e14dcfSSatish Balay   ierr = VecScale(Xbar,nm->oneOverN);
155a7e14dcfSSatish Balay   reason = TAO_CONTINUE_ITERATING;
156a7e14dcfSSatish Balay   while (1) {
157a7e14dcfSSatish Balay     shrink = 0;
158a7e14dcfSSatish Balay     ierr = VecCopy(nm->simplex[nm->indices[0]],tao->solution); CHKERRQ(ierr);
159a7e14dcfSSatish Balay     ierr = TaoMonitor(tao,iter++,nm->f_values[nm->indices[0]],nm->f_values[nm->indices[nm->N]]-nm->f_values[nm->indices[0]],0.0,1.0,&reason); CHKERRQ(ierr);
160a7e14dcfSSatish Balay     if (reason != TAO_CONTINUE_ITERATING) break;
161a7e14dcfSSatish Balay 
162a7e14dcfSSatish Balay 
163a7e14dcfSSatish Balay 
164a7e14dcfSSatish Balay     /* x(mu) = (1 + mu)Xbar - mu*X_N+1 */
165a7e14dcfSSatish Balay     ierr = VecAXPBYPCZ(Xmur,1+nm->mu_r,-nm->mu_r,0,Xbar,nm->simplex[nm->indices[nm->N]]); CHKERRQ(ierr);
166a7e14dcfSSatish Balay     ierr = TaoComputeObjective(tao,Xmur,&fr); CHKERRQ(ierr);
167a7e14dcfSSatish Balay 
168a7e14dcfSSatish Balay 
169a7e14dcfSSatish Balay     if (nm->f_values[nm->indices[0]] <= fr && fr < nm->f_values[nm->indices[nm->N-1]]) {
170a7e14dcfSSatish Balay       /*  reflect */
171a7e14dcfSSatish Balay       nm->nreflect++;
172a7e14dcfSSatish Balay       ierr = PetscInfo(0,"Reflect\n"); CHKERRQ(ierr);
173a7e14dcfSSatish Balay       ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmur,fr); CHKERRQ(ierr);
174a7e14dcfSSatish Balay     }
175a7e14dcfSSatish Balay 
176a7e14dcfSSatish Balay     else if (fr < nm->f_values[nm->indices[0]]) {
177a7e14dcfSSatish Balay       /*  expand */
178a7e14dcfSSatish Balay       nm->nexpand++;
179a7e14dcfSSatish Balay       ierr = PetscInfo(0,"Expand\n"); CHKERRQ(ierr);
180a7e14dcfSSatish Balay       ierr = VecAXPBYPCZ(Xmue,1+nm->mu_e,-nm->mu_e,0,Xbar,nm->simplex[nm->indices[nm->N]]); CHKERRQ(ierr);
181a7e14dcfSSatish Balay       ierr = TaoComputeObjective(tao,Xmue,&fe); CHKERRQ(ierr);
182a7e14dcfSSatish Balay       if (fe < fr) {
183a7e14dcfSSatish Balay 	ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmue,fe); CHKERRQ(ierr);
184a7e14dcfSSatish Balay       } else {
185a7e14dcfSSatish Balay 	ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmur,fr); CHKERRQ(ierr);
186a7e14dcfSSatish Balay       }
187a7e14dcfSSatish Balay 
188a7e14dcfSSatish Balay     } else if (nm->f_values[nm->indices[nm->N-1]] <= fr && fr < nm->f_values[nm->indices[nm->N]]) {
189a7e14dcfSSatish Balay       /* outside contraction */
190a7e14dcfSSatish Balay       nm->noutcontract++;
191a7e14dcfSSatish Balay       ierr = PetscInfo(0,"Outside Contraction\n"); CHKERRQ(ierr);
192a7e14dcfSSatish Balay       ierr = VecAXPBYPCZ(Xmuc,1+nm->mu_oc,-nm->mu_oc,0,Xbar,nm->simplex[nm->indices[nm->N]]); CHKERRQ(ierr);
193a7e14dcfSSatish Balay 
194a7e14dcfSSatish Balay       ierr = TaoComputeObjective(tao,Xmuc,&fc); CHKERRQ(ierr);
195a7e14dcfSSatish Balay       if (fc <= fr) {
196a7e14dcfSSatish Balay 	ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmuc,fc); CHKERRQ(ierr);
197a7e14dcfSSatish Balay       }	else
198a7e14dcfSSatish Balay 	shrink=1;
199a7e14dcfSSatish Balay     } else {
200a7e14dcfSSatish Balay       /* inside contraction */
201a7e14dcfSSatish Balay       nm->nincontract++;
202a7e14dcfSSatish Balay       ierr = PetscInfo(0,"Inside Contraction\n"); CHKERRQ(ierr);
203a7e14dcfSSatish Balay       ierr = VecAXPBYPCZ(Xmuc,1+nm->mu_ic,-nm->mu_ic,0,Xbar,nm->simplex[nm->indices[nm->N]]); CHKERRQ(ierr);
204a7e14dcfSSatish Balay       ierr = TaoComputeObjective(tao,Xmuc,&fc); CHKERRQ(ierr);
205a7e14dcfSSatish Balay       if (fc < nm->f_values[nm->indices[nm->N]]) {
206a7e14dcfSSatish Balay 	ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmuc,fc); CHKERRQ(ierr);
207a7e14dcfSSatish Balay       } else
208a7e14dcfSSatish Balay 	shrink = 1;
209a7e14dcfSSatish Balay     }
210a7e14dcfSSatish Balay 
211a7e14dcfSSatish Balay     if (shrink) {
212a7e14dcfSSatish Balay       nm->nshrink++;
213a7e14dcfSSatish Balay       ierr = PetscInfo(0,"Shrink\n"); CHKERRQ(ierr);
214a7e14dcfSSatish Balay 
215a7e14dcfSSatish Balay       for (i=1;i<nm->N+1;i++) {
216a7e14dcfSSatish Balay 	  ierr = VecAXPBY(nm->simplex[nm->indices[i]],1.5,-0.5,nm->simplex[nm->indices[0]]);
217a7e14dcfSSatish Balay 	ierr = TaoComputeObjective(tao,nm->simplex[nm->indices[i]],
218a7e14dcfSSatish Balay 				 &nm->f_values[nm->indices[i]]); CHKERRQ(ierr);
219a7e14dcfSSatish Balay       }
220a7e14dcfSSatish Balay       ierr = VecAXPBY(Xbar,1.5*nm->oneOverN,-0.5,nm->simplex[nm->indices[0]]); CHKERRQ(ierr);
221a7e14dcfSSatish Balay 
222a7e14dcfSSatish Balay       /*  Add last vector's fraction of average */
223a7e14dcfSSatish Balay       ierr = VecAXPY(Xbar,nm->oneOverN,nm->simplex[nm->indices[nm->N]]); CHKERRQ(ierr);
224a7e14dcfSSatish Balay       ierr = NelderMeadSort(nm);
225a7e14dcfSSatish Balay       /*  Subtract new last vector from average */
226a7e14dcfSSatish Balay       ierr = VecAXPY(Xbar,-nm->oneOverN,nm->simplex[nm->indices[nm->N]]); CHKERRQ(ierr);
227a7e14dcfSSatish Balay     }
228a7e14dcfSSatish Balay 
229a7e14dcfSSatish Balay 
230a7e14dcfSSatish Balay   }
231a7e14dcfSSatish Balay   PetscFunctionReturn(0);
232a7e14dcfSSatish Balay }
233a7e14dcfSSatish Balay 
234a7e14dcfSSatish Balay /* ---------------------------------------------------------- */
235a7e14dcfSSatish Balay EXTERN_C_BEGIN
236a7e14dcfSSatish Balay #undef __FUNCT__
237a7e14dcfSSatish Balay #define __FUNCT__ "TaoCreate_NM"
238a7e14dcfSSatish Balay PetscErrorCode TaoCreate_NM(TaoSolver tao)
239a7e14dcfSSatish Balay {
240a7e14dcfSSatish Balay   TAO_NelderMead *nm;
241a7e14dcfSSatish Balay   PetscErrorCode ierr;
242a7e14dcfSSatish Balay 
243a7e14dcfSSatish Balay   PetscFunctionBegin;
244*3c9e27cfSGeoffrey Irving   ierr = PetscNewLog(tao,&nm); CHKERRQ(ierr);
245a7e14dcfSSatish Balay   tao->data = (void*)nm;
246a7e14dcfSSatish Balay 
247a7e14dcfSSatish Balay   tao->ops->setup = TaoSetUp_NM;
248a7e14dcfSSatish Balay   tao->ops->solve = TaoSolve_NM;
249a7e14dcfSSatish Balay   tao->ops->view = TaoView_NM;
250a7e14dcfSSatish Balay   tao->ops->setfromoptions = TaoSetFromOptions_NM;
251a7e14dcfSSatish Balay   tao->ops->destroy = TaoDestroy_NM;
252a7e14dcfSSatish Balay 
253a7e14dcfSSatish Balay   tao->max_it = 2000;
254a7e14dcfSSatish Balay   tao->max_funcs = 4000;
255a7e14dcfSSatish Balay   tao->fatol = 1e-8;
256a7e14dcfSSatish Balay   tao->frtol = 1e-8;
257a7e14dcfSSatish Balay 
258a7e14dcfSSatish Balay   nm->simplex = 0;
259a7e14dcfSSatish Balay   nm->lamda = 1;
260a7e14dcfSSatish Balay 
261a7e14dcfSSatish Balay   nm->mu_ic = -0.5;
262a7e14dcfSSatish Balay   nm->mu_oc = 0.5;
263a7e14dcfSSatish Balay   nm->mu_r = 1.0;
264a7e14dcfSSatish Balay   nm->mu_e = 2.0;
265a7e14dcfSSatish Balay 
266a7e14dcfSSatish Balay   PetscFunctionReturn(0);
267a7e14dcfSSatish Balay }
268a7e14dcfSSatish Balay EXTERN_C_END
269a7e14dcfSSatish Balay 
270a7e14dcfSSatish Balay 
271a7e14dcfSSatish Balay /*------------------------------------------------------------*/
272a7e14dcfSSatish Balay #undef __FUNCT__
273a7e14dcfSSatish Balay #define __FUNCT__ "NelderMeadSort"
274a7e14dcfSSatish Balay PetscErrorCode NelderMeadSort(TAO_NelderMead *nm) {
275a7e14dcfSSatish Balay   PetscReal *values = nm->f_values;
276a7e14dcfSSatish Balay   PetscInt *indices = nm->indices;
277a7e14dcfSSatish Balay   PetscInt dim = nm->N+1;
278a7e14dcfSSatish Balay 
279a7e14dcfSSatish Balay   PetscInt i,j,index;
280a7e14dcfSSatish Balay   PetscReal val;
281a7e14dcfSSatish Balay   PetscFunctionBegin;
282a7e14dcfSSatish Balay   for (i=1;i<dim;i++) {
283a7e14dcfSSatish Balay     index = indices[i];
284a7e14dcfSSatish Balay     val = values[index];
285a7e14dcfSSatish Balay     for (j=i-1; j>=0 && values[indices[j]] > val; j--) {
286a7e14dcfSSatish Balay       indices[j+1] = indices[j];
287a7e14dcfSSatish Balay     }
288a7e14dcfSSatish Balay     indices[j+1] = index;
289a7e14dcfSSatish Balay   }
290a7e14dcfSSatish Balay   PetscFunctionReturn(0);
291a7e14dcfSSatish Balay }
292a7e14dcfSSatish Balay 
293a7e14dcfSSatish Balay 
294a7e14dcfSSatish Balay /*------------------------------------------------------------*/
295a7e14dcfSSatish Balay #undef __FUNCT__
296a7e14dcfSSatish Balay #define __FUNCT__ "NelderMeadReplace"
297a7e14dcfSSatish Balay PetscErrorCode NelderMeadReplace(TAO_NelderMead *nm, PetscInt index, Vec Xmu, PetscReal f)
298a7e14dcfSSatish Balay {
299a7e14dcfSSatish Balay   PetscErrorCode ierr;
300a7e14dcfSSatish Balay   PetscFunctionBegin;
301a7e14dcfSSatish Balay   /*  Add new vector's fraction of average */
302a7e14dcfSSatish Balay   ierr = VecAXPY(nm->Xbar,nm->oneOverN,Xmu); CHKERRQ(ierr);
303a7e14dcfSSatish Balay   ierr = VecCopy(Xmu,nm->simplex[index]); CHKERRQ(ierr);
304a7e14dcfSSatish Balay   nm->f_values[index] = f;
305a7e14dcfSSatish Balay 
306a7e14dcfSSatish Balay   ierr = NelderMeadSort(nm);
307a7e14dcfSSatish Balay 
308a7e14dcfSSatish Balay   /*  Subtract last vector from average */
309a7e14dcfSSatish Balay   ierr = VecAXPY(nm->Xbar,-nm->oneOverN,nm->simplex[nm->indices[nm->N]]); CHKERRQ(ierr);
310a7e14dcfSSatish Balay   PetscFunctionReturn(0);
311a7e14dcfSSatish Balay }
312