1aaa7dc30SBarry Smith #include <../src/tao/unconstrained/impls/neldermead/neldermead.h> 2aaa7dc30SBarry Smith #include <petscvec.h> 3a7e14dcfSSatish Balay 440244768SBarry Smith 540244768SBarry Smith /*------------------------------------------------------------*/ 640244768SBarry Smith static PetscErrorCode NelderMeadSort(TAO_NelderMead *nm) 740244768SBarry Smith { 840244768SBarry Smith PetscReal *values = nm->f_values; 940244768SBarry Smith PetscInt *indices = nm->indices; 1040244768SBarry Smith PetscInt dim = nm->N+1; 1140244768SBarry Smith PetscInt i,j,index; 1240244768SBarry Smith PetscReal val; 1340244768SBarry Smith 1440244768SBarry Smith PetscFunctionBegin; 1540244768SBarry Smith for (i=1;i<dim;i++) { 1640244768SBarry Smith index = indices[i]; 1740244768SBarry Smith val = values[index]; 1840244768SBarry Smith for (j=i-1; j>=0 && values[indices[j]] > val; j--) { 1940244768SBarry Smith indices[j+1] = indices[j]; 2040244768SBarry Smith } 2140244768SBarry Smith indices[j+1] = index; 2240244768SBarry Smith } 2340244768SBarry Smith PetscFunctionReturn(0); 2440244768SBarry Smith } 2540244768SBarry Smith 2640244768SBarry Smith 2740244768SBarry Smith /*------------------------------------------------------------*/ 2840244768SBarry Smith static PetscErrorCode NelderMeadReplace(TAO_NelderMead *nm, PetscInt index, Vec Xmu, PetscReal f) 2940244768SBarry Smith { 3040244768SBarry Smith PetscErrorCode ierr; 3140244768SBarry Smith 3240244768SBarry Smith PetscFunctionBegin; 3340244768SBarry Smith /* Add new vector's fraction of average */ 3440244768SBarry Smith ierr = VecAXPY(nm->Xbar,nm->oneOverN,Xmu);CHKERRQ(ierr); 3540244768SBarry Smith ierr = VecCopy(Xmu,nm->simplex[index]);CHKERRQ(ierr); 3640244768SBarry Smith nm->f_values[index] = f; 3740244768SBarry Smith 3840244768SBarry Smith ierr = NelderMeadSort(nm);CHKERRQ(ierr); 3940244768SBarry Smith 4040244768SBarry Smith /* Subtract last vector from average */ 4140244768SBarry Smith ierr = VecAXPY(nm->Xbar,-nm->oneOverN,nm->simplex[nm->indices[nm->N]]);CHKERRQ(ierr); 4240244768SBarry Smith PetscFunctionReturn(0); 4340244768SBarry Smith } 4440244768SBarry Smith 45a7e14dcfSSatish Balay /* ---------------------------------------------------------- */ 46441846f8SBarry Smith static PetscErrorCode TaoSetUp_NM(Tao tao) 47a7e14dcfSSatish Balay { 48a7e14dcfSSatish Balay PetscErrorCode ierr; 49a7e14dcfSSatish Balay TAO_NelderMead *nm = (TAO_NelderMead *)tao->data; 5053506e15SBarry Smith PetscInt n; 51a7e14dcfSSatish Balay 52a7e14dcfSSatish Balay PetscFunctionBegin; 5353506e15SBarry Smith ierr = VecGetSize(tao->solution,&n);CHKERRQ(ierr); 5453506e15SBarry Smith nm->N = n; 5553506e15SBarry Smith nm->oneOverN = 1.0/n; 56a7e14dcfSSatish Balay ierr = VecDuplicateVecs(tao->solution,nm->N+1,&nm->simplex);CHKERRQ(ierr); 57854ce69bSBarry Smith ierr = PetscMalloc1(nm->N+1,&nm->f_values);CHKERRQ(ierr); 58854ce69bSBarry Smith ierr = PetscMalloc1(nm->N+1,&nm->indices);CHKERRQ(ierr); 59a7e14dcfSSatish Balay ierr = VecDuplicate(tao->solution,&nm->Xbar);CHKERRQ(ierr); 60a7e14dcfSSatish Balay ierr = VecDuplicate(tao->solution,&nm->Xmur);CHKERRQ(ierr); 61a7e14dcfSSatish Balay ierr = VecDuplicate(tao->solution,&nm->Xmue);CHKERRQ(ierr); 62a7e14dcfSSatish Balay ierr = VecDuplicate(tao->solution,&nm->Xmuc);CHKERRQ(ierr); 63a7e14dcfSSatish Balay 64*83c8fe1dSLisandro Dalcin tao->gradient=NULL; 65a7e14dcfSSatish Balay tao->step=0; 66a7e14dcfSSatish Balay PetscFunctionReturn(0); 67a7e14dcfSSatish Balay } 68a7e14dcfSSatish Balay 69a7e14dcfSSatish Balay /* ---------------------------------------------------------- */ 7040244768SBarry Smith static PetscErrorCode TaoDestroy_NM(Tao tao) 71a7e14dcfSSatish Balay { 72a7e14dcfSSatish Balay TAO_NelderMead *nm = (TAO_NelderMead*)tao->data; 73a7e14dcfSSatish Balay PetscErrorCode ierr; 7453506e15SBarry Smith 75a7e14dcfSSatish Balay PetscFunctionBegin; 76a7e14dcfSSatish Balay if (tao->setupcalled) { 77a7e14dcfSSatish Balay ierr = VecDestroyVecs(nm->N+1,&nm->simplex);CHKERRQ(ierr); 78a7e14dcfSSatish Balay ierr = VecDestroy(&nm->Xmuc);CHKERRQ(ierr); 79a7e14dcfSSatish Balay ierr = VecDestroy(&nm->Xmue);CHKERRQ(ierr); 80a7e14dcfSSatish Balay ierr = VecDestroy(&nm->Xmur);CHKERRQ(ierr); 81a7e14dcfSSatish Balay ierr = VecDestroy(&nm->Xbar);CHKERRQ(ierr); 82a7e14dcfSSatish Balay } 83a7e14dcfSSatish Balay ierr = PetscFree(nm->indices);CHKERRQ(ierr); 84a7e14dcfSSatish Balay ierr = PetscFree(nm->f_values);CHKERRQ(ierr); 85302440fdSBarry Smith ierr = PetscFree(tao->data);CHKERRQ(ierr); 86a7e14dcfSSatish Balay PetscFunctionReturn(0); 87a7e14dcfSSatish Balay } 88a7e14dcfSSatish Balay 89a7e14dcfSSatish Balay /*------------------------------------------------------------*/ 9040244768SBarry Smith static PetscErrorCode TaoSetFromOptions_NM(PetscOptionItems *PetscOptionsObject,Tao tao) 91a7e14dcfSSatish Balay { 92a7e14dcfSSatish Balay TAO_NelderMead *nm = (TAO_NelderMead*)tao->data; 93a7e14dcfSSatish Balay PetscErrorCode ierr; 94a7e14dcfSSatish Balay 95a7e14dcfSSatish Balay PetscFunctionBegin; 961a1499c8SBarry Smith ierr = PetscOptionsHead(PetscOptionsObject,"Nelder-Mead options");CHKERRQ(ierr); 9794ae4db5SBarry Smith ierr = PetscOptionsReal("-tao_nm_lamda","initial step length","",nm->lamda,&nm->lamda,NULL); CHKERRQ(ierr); 9894ae4db5SBarry Smith ierr = PetscOptionsReal("-tao_nm_mu","mu","",nm->mu_oc,&nm->mu_oc,NULL);CHKERRQ(ierr); 99a7e14dcfSSatish Balay nm->mu_ic = -nm->mu_oc; 100a7e14dcfSSatish Balay nm->mu_r = nm->mu_oc*2.0; 101a7e14dcfSSatish Balay nm->mu_e = nm->mu_oc*4.0; 102a7e14dcfSSatish Balay ierr = PetscOptionsTail();CHKERRQ(ierr); 103a7e14dcfSSatish Balay PetscFunctionReturn(0); 104a7e14dcfSSatish Balay } 105a7e14dcfSSatish Balay 106a7e14dcfSSatish Balay /*------------------------------------------------------------*/ 10740244768SBarry Smith static PetscErrorCode TaoView_NM(Tao tao,PetscViewer viewer) 108a7e14dcfSSatish Balay { 109a7e14dcfSSatish Balay TAO_NelderMead *nm = (TAO_NelderMead*)tao->data; 110a7e14dcfSSatish Balay PetscBool isascii; 111a7e14dcfSSatish Balay PetscErrorCode ierr; 112a7e14dcfSSatish Balay 113a7e14dcfSSatish Balay PetscFunctionBegin; 114a7e14dcfSSatish Balay ierr = PetscObjectTypeCompare((PetscObject)viewer,PETSCVIEWERASCII,&isascii);CHKERRQ(ierr); 115a7e14dcfSSatish Balay if (isascii) { 116a7e14dcfSSatish Balay ierr = PetscViewerASCIIPushTab(viewer);CHKERRQ(ierr); 117a7e14dcfSSatish Balay ierr = PetscViewerASCIIPrintf(viewer,"expansions: %D\n",nm->nexpand);CHKERRQ(ierr); 118a7e14dcfSSatish Balay ierr = PetscViewerASCIIPrintf(viewer,"reflections: %D\n",nm->nreflect);CHKERRQ(ierr); 119a7e14dcfSSatish Balay ierr = PetscViewerASCIIPrintf(viewer,"inside contractions: %D\n",nm->nincontract);CHKERRQ(ierr); 120a7e14dcfSSatish Balay ierr = PetscViewerASCIIPrintf(viewer,"outside contractionss: %D\n",nm->noutcontract);CHKERRQ(ierr); 121a7e14dcfSSatish Balay ierr = PetscViewerASCIIPrintf(viewer,"Shrink steps: %D\n",nm->nshrink);CHKERRQ(ierr); 122a7e14dcfSSatish Balay ierr = PetscViewerASCIIPopTab(viewer);CHKERRQ(ierr); 123a7e14dcfSSatish Balay } 124a7e14dcfSSatish Balay PetscFunctionReturn(0); 125a7e14dcfSSatish Balay } 126a7e14dcfSSatish Balay 127a7e14dcfSSatish Balay /*------------------------------------------------------------*/ 12840244768SBarry Smith static PetscErrorCode TaoSolve_NM(Tao tao) 129a7e14dcfSSatish Balay { 130a7e14dcfSSatish Balay PetscErrorCode ierr; 131a7e14dcfSSatish Balay TAO_NelderMead *nm = (TAO_NelderMead*)tao->data; 132a7e14dcfSSatish Balay PetscReal *x; 1338931d482SJason Sarich PetscInt i; 134a7e14dcfSSatish Balay Vec Xmur=nm->Xmur, Xmue=nm->Xmue, Xmuc=nm->Xmuc, Xbar=nm->Xbar; 135a7e14dcfSSatish Balay PetscReal fr,fe,fc; 136a7e14dcfSSatish Balay PetscInt shrink; 137a7e14dcfSSatish Balay PetscInt low,high; 138a7e14dcfSSatish Balay 139a7e14dcfSSatish Balay PetscFunctionBegin; 140a7e14dcfSSatish Balay nm->nshrink = 0; 141a7e14dcfSSatish Balay nm->nreflect = 0; 142a7e14dcfSSatish Balay nm->nincontract = 0; 143a7e14dcfSSatish Balay nm->noutcontract = 0; 144a7e14dcfSSatish Balay nm->nexpand = 0; 145a7e14dcfSSatish Balay 146a7e14dcfSSatish Balay if (tao->XL || tao->XU || tao->ops->computebounds) { 1479dcef436SStefano Zampini ierr = PetscInfo(tao,"WARNING: Variable bounds have been set but will be ignored by NelderMead algorithm\n");CHKERRQ(ierr); 148a7e14dcfSSatish Balay } 149a7e14dcfSSatish Balay 150a7e14dcfSSatish Balay ierr = VecCopy(tao->solution,nm->simplex[0]);CHKERRQ(ierr); 151a7e14dcfSSatish Balay ierr = TaoComputeObjective(tao,nm->simplex[0],&nm->f_values[0]);CHKERRQ(ierr); 152a7e14dcfSSatish Balay nm->indices[0]=0; 153a7e14dcfSSatish Balay for (i=1;i<nm->N+1;i++){ 154a7e14dcfSSatish Balay ierr = VecCopy(tao->solution,nm->simplex[i]);CHKERRQ(ierr); 155a7e14dcfSSatish Balay ierr = VecGetOwnershipRange(nm->simplex[i],&low,&high);CHKERRQ(ierr); 156a7e14dcfSSatish Balay if (i-1 >= low && i-1 < high) { 157a7e14dcfSSatish Balay ierr = VecGetArray(nm->simplex[i],&x);CHKERRQ(ierr); 158a7e14dcfSSatish Balay x[i-1-low] += nm->lamda; 159a7e14dcfSSatish Balay ierr = VecRestoreArray(nm->simplex[i],&x);CHKERRQ(ierr); 160a7e14dcfSSatish Balay } 161a7e14dcfSSatish Balay 162a7e14dcfSSatish Balay ierr = TaoComputeObjective(tao,nm->simplex[i],&nm->f_values[i]);CHKERRQ(ierr); 163a7e14dcfSSatish Balay nm->indices[i] = i; 164a7e14dcfSSatish Balay } 165a7e14dcfSSatish Balay 166a7e14dcfSSatish Balay /* Xbar = (Sum of all simplex vectors - worst vector)/N */ 167a7e14dcfSSatish Balay ierr = NelderMeadSort(nm);CHKERRQ(ierr); 168a7e14dcfSSatish Balay ierr = VecSet(Xbar,0.0);CHKERRQ(ierr); 169a7e14dcfSSatish Balay for (i=0;i<nm->N;i++) { 170302440fdSBarry Smith ierr = VecAXPY(Xbar,1.0,nm->simplex[nm->indices[i]]);CHKERRQ(ierr); 171a7e14dcfSSatish Balay } 172302440fdSBarry Smith ierr = VecScale(Xbar,nm->oneOverN);CHKERRQ(ierr); 1733ecd9318SAlp Dener tao->reason = TAO_CONTINUE_ITERATING; 174a7e14dcfSSatish Balay while (1) { 175e1e80dc8SAlp Dener /* Call general purpose update function */ 176e1e80dc8SAlp Dener if (tao->ops->update) { 1778fcddce6SStefano Zampini ierr = (*tao->ops->update)(tao, tao->niter, tao->user_update);CHKERRQ(ierr); 178e1e80dc8SAlp Dener } 179d643b47cSToby Isaac ++tao->niter; 180a7e14dcfSSatish Balay shrink = 0; 181a7e14dcfSSatish Balay ierr = VecCopy(nm->simplex[nm->indices[0]],tao->solution);CHKERRQ(ierr); 1823ecd9318SAlp Dener ierr = TaoLogConvergenceHistory(tao, nm->f_values[nm->indices[0]], nm->f_values[nm->indices[nm->N]]-nm->f_values[nm->indices[0]], 0.0, tao->ksp_its);CHKERRQ(ierr); 1833ecd9318SAlp Dener ierr = TaoMonitor(tao,tao->niter, nm->f_values[nm->indices[0]], nm->f_values[nm->indices[nm->N]]-nm->f_values[nm->indices[0]], 0.0, 1.0);CHKERRQ(ierr); 1843ecd9318SAlp Dener ierr = (*tao->ops->convergencetest)(tao,tao->cnvP);CHKERRQ(ierr); 1853ecd9318SAlp Dener if (tao->reason != TAO_CONTINUE_ITERATING) break; 186a7e14dcfSSatish Balay 187a7e14dcfSSatish Balay /* x(mu) = (1 + mu)Xbar - mu*X_N+1 */ 188a7e14dcfSSatish Balay ierr = VecAXPBYPCZ(Xmur,1+nm->mu_r,-nm->mu_r,0,Xbar,nm->simplex[nm->indices[nm->N]]);CHKERRQ(ierr); 189a7e14dcfSSatish Balay ierr = TaoComputeObjective(tao,Xmur,&fr);CHKERRQ(ierr); 190a7e14dcfSSatish Balay 191a7e14dcfSSatish Balay if (nm->f_values[nm->indices[0]] <= fr && fr < nm->f_values[nm->indices[nm->N-1]]) { 192a7e14dcfSSatish Balay /* reflect */ 193a7e14dcfSSatish Balay nm->nreflect++; 194a7e14dcfSSatish Balay ierr = PetscInfo(0,"Reflect\n");CHKERRQ(ierr); 195a7e14dcfSSatish Balay ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmur,fr);CHKERRQ(ierr); 19653506e15SBarry Smith } else if (fr < nm->f_values[nm->indices[0]]) { 197a7e14dcfSSatish Balay /* expand */ 198a7e14dcfSSatish Balay nm->nexpand++; 199a7e14dcfSSatish Balay ierr = PetscInfo(0,"Expand\n");CHKERRQ(ierr); 200a7e14dcfSSatish Balay ierr = VecAXPBYPCZ(Xmue,1+nm->mu_e,-nm->mu_e,0,Xbar,nm->simplex[nm->indices[nm->N]]);CHKERRQ(ierr); 201a7e14dcfSSatish Balay ierr = TaoComputeObjective(tao,Xmue,&fe);CHKERRQ(ierr); 202a7e14dcfSSatish Balay if (fe < fr) { 203a7e14dcfSSatish Balay ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmue,fe);CHKERRQ(ierr); 204a7e14dcfSSatish Balay } else { 205a7e14dcfSSatish Balay ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmur,fr);CHKERRQ(ierr); 206a7e14dcfSSatish Balay } 207a7e14dcfSSatish Balay } else if (nm->f_values[nm->indices[nm->N-1]] <= fr && fr < nm->f_values[nm->indices[nm->N]]) { 208a7e14dcfSSatish Balay /* outside contraction */ 209a7e14dcfSSatish Balay nm->noutcontract++; 210a7e14dcfSSatish Balay ierr = PetscInfo(0,"Outside Contraction\n");CHKERRQ(ierr); 211a7e14dcfSSatish Balay ierr = VecAXPBYPCZ(Xmuc,1+nm->mu_oc,-nm->mu_oc,0,Xbar,nm->simplex[nm->indices[nm->N]]);CHKERRQ(ierr); 212a7e14dcfSSatish Balay 213a7e14dcfSSatish Balay ierr = TaoComputeObjective(tao,Xmuc,&fc);CHKERRQ(ierr); 214a7e14dcfSSatish Balay if (fc <= fr) { 215a7e14dcfSSatish Balay ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmuc,fc);CHKERRQ(ierr); 21653506e15SBarry Smith } else shrink=1; 217a7e14dcfSSatish Balay } else { 218a7e14dcfSSatish Balay /* inside contraction */ 219a7e14dcfSSatish Balay nm->nincontract++; 220a7e14dcfSSatish Balay ierr = PetscInfo(0,"Inside Contraction\n");CHKERRQ(ierr); 221a7e14dcfSSatish Balay ierr = VecAXPBYPCZ(Xmuc,1+nm->mu_ic,-nm->mu_ic,0,Xbar,nm->simplex[nm->indices[nm->N]]);CHKERRQ(ierr); 222a7e14dcfSSatish Balay ierr = TaoComputeObjective(tao,Xmuc,&fc);CHKERRQ(ierr); 223a7e14dcfSSatish Balay if (fc < nm->f_values[nm->indices[nm->N]]) { 224a7e14dcfSSatish Balay ierr = NelderMeadReplace(nm,nm->indices[nm->N],Xmuc,fc);CHKERRQ(ierr); 22553506e15SBarry Smith } else shrink = 1; 226a7e14dcfSSatish Balay } 227a7e14dcfSSatish Balay 228a7e14dcfSSatish Balay if (shrink) { 229a7e14dcfSSatish Balay nm->nshrink++; 230a7e14dcfSSatish Balay ierr = PetscInfo(0,"Shrink\n");CHKERRQ(ierr); 231a7e14dcfSSatish Balay 232a7e14dcfSSatish Balay for (i=1;i<nm->N+1;i++) { 233302440fdSBarry Smith ierr = VecAXPBY(nm->simplex[nm->indices[i]],1.5,-0.5,nm->simplex[nm->indices[0]]);CHKERRQ(ierr); 23453506e15SBarry Smith ierr = TaoComputeObjective(tao,nm->simplex[nm->indices[i]], &nm->f_values[nm->indices[i]]);CHKERRQ(ierr); 235a7e14dcfSSatish Balay } 236a7e14dcfSSatish Balay ierr = VecAXPBY(Xbar,1.5*nm->oneOverN,-0.5,nm->simplex[nm->indices[0]]);CHKERRQ(ierr); 237a7e14dcfSSatish Balay 238a7e14dcfSSatish Balay /* Add last vector's fraction of average */ 239a7e14dcfSSatish Balay ierr = VecAXPY(Xbar,nm->oneOverN,nm->simplex[nm->indices[nm->N]]);CHKERRQ(ierr); 240302440fdSBarry Smith ierr = NelderMeadSort(nm);CHKERRQ(ierr); 241a7e14dcfSSatish Balay /* Subtract new last vector from average */ 242a7e14dcfSSatish Balay ierr = VecAXPY(Xbar,-nm->oneOverN,nm->simplex[nm->indices[nm->N]]);CHKERRQ(ierr); 243a7e14dcfSSatish Balay } 244a7e14dcfSSatish Balay } 245a7e14dcfSSatish Balay PetscFunctionReturn(0); 246a7e14dcfSSatish Balay } 247a7e14dcfSSatish Balay 248a7e14dcfSSatish Balay /* ---------------------------------------------------------- */ 2491eb8069cSJason Sarich /*MC 2501eb8069cSJason Sarich TAONM - Nelder-Mead solver for derivative free, unconstrained minimization 2511eb8069cSJason Sarich 2521eb8069cSJason Sarich Options Database Keys: 2531eb8069cSJason Sarich + -tao_nm_lamda - initial step length 254a2b725a8SWilliam Gropp - -tao_nm_mu - expansion/contraction factor 2551eb8069cSJason Sarich 2561eb8069cSJason Sarich Level: beginner 2571eb8069cSJason Sarich M*/ 2581eb8069cSJason Sarich 259728e0ed0SBarry Smith PETSC_EXTERN PetscErrorCode TaoCreate_NM(Tao tao) 260a7e14dcfSSatish Balay { 261a7e14dcfSSatish Balay TAO_NelderMead *nm; 262a7e14dcfSSatish Balay PetscErrorCode ierr; 263a7e14dcfSSatish Balay 264a7e14dcfSSatish Balay PetscFunctionBegin; 2653c9e27cfSGeoffrey Irving ierr = PetscNewLog(tao,&nm);CHKERRQ(ierr); 266a7e14dcfSSatish Balay tao->data = (void*)nm; 267a7e14dcfSSatish Balay 268a7e14dcfSSatish Balay tao->ops->setup = TaoSetUp_NM; 269a7e14dcfSSatish Balay tao->ops->solve = TaoSolve_NM; 270a7e14dcfSSatish Balay tao->ops->view = TaoView_NM; 271a7e14dcfSSatish Balay tao->ops->setfromoptions = TaoSetFromOptions_NM; 272a7e14dcfSSatish Balay tao->ops->destroy = TaoDestroy_NM; 273a7e14dcfSSatish Balay 2746552cf8aSJason Sarich /* Override default settings (unless already changed) */ 2756552cf8aSJason Sarich if (!tao->max_it_changed) tao->max_it = 2000; 2766552cf8aSJason Sarich if (!tao->max_funcs_changed) tao->max_funcs = 4000; 277a7e14dcfSSatish Balay 278*83c8fe1dSLisandro Dalcin nm->simplex = NULL; 279a7e14dcfSSatish Balay nm->lamda = 1; 280a7e14dcfSSatish Balay 281a7e14dcfSSatish Balay nm->mu_ic = -0.5; 282a7e14dcfSSatish Balay nm->mu_oc = 0.5; 283a7e14dcfSSatish Balay nm->mu_r = 1.0; 284a7e14dcfSSatish Balay nm->mu_e = 2.0; 285a7e14dcfSSatish Balay 286a7e14dcfSSatish Balay PetscFunctionReturn(0); 287a7e14dcfSSatish Balay } 288a7e14dcfSSatish Balay 289