xref: /petsc/src/tao/bound/impls/bqnls/bqnls.c (revision d5ae23803a94ca2b9c66785d81449705f0cc1e9e)
16b591159SAlp Dener #include <../src/tao/bound/impls/bqnk/bqnk.h>
26b591159SAlp Dener 
36b591159SAlp Dener static PetscErrorCode TaoBQNLSComputeHessian(Tao tao)
46b591159SAlp Dener {
56b591159SAlp Dener   TAO_BNK        *bnk = (TAO_BNK *)tao->data;
66b591159SAlp Dener   TAO_BQNK       *bqnk = (TAO_BQNK*)bnk->ctx;
76b591159SAlp Dener   PetscErrorCode ierr;
8*d5ae2380SAlp Dener   PetscReal      gnorm2, delta;
96b591159SAlp Dener 
106b591159SAlp Dener   PetscFunctionBegin;
11*d5ae2380SAlp Dener   gnorm2 = bnk->gnorm*bnk->gnorm;
12*d5ae2380SAlp Dener   if (gnorm2 == 0.0) gnorm2 = PetscPowReal(PETSC_MACHINE_EPSILON, 2.0/3.0);
13*d5ae2380SAlp Dener   if (bnk->f != 0.0) {
14*d5ae2380SAlp Dener     delta = 2.0*PetscAbsScalar(bnk->f) / gnorm2;
15*d5ae2380SAlp Dener   } else {
16*d5ae2380SAlp Dener     delta = 2.0 / gnorm2;
17*d5ae2380SAlp Dener   }
18*d5ae2380SAlp Dener   ierr = MatSymBrdnSetDelta(bqnk->B, delta);CHKERRQ(ierr);
196b591159SAlp Dener   ierr = MatLMVMUpdate(bqnk->B, tao->solution, bnk->unprojected_gradient);CHKERRQ(ierr);
206b591159SAlp Dener   PetscFunctionReturn(0);
216b591159SAlp Dener }
226b591159SAlp Dener 
236b591159SAlp Dener static PetscErrorCode TaoBQNLSComputeStep(Tao tao, PetscBool shift, KSPConvergedReason *ksp_reason, PetscInt *step_type)
246b591159SAlp Dener {
256b591159SAlp Dener   TAO_BNK        *bnk = (TAO_BNK *)tao->data;
266b591159SAlp Dener   TAO_BQNK       *bqnk = (TAO_BQNK*)bnk->ctx;
276b591159SAlp Dener   PetscErrorCode ierr;
286b591159SAlp Dener 
296b591159SAlp Dener   PetscFunctionBegin;
306b591159SAlp Dener   ierr = MatSolve(bqnk->B, tao->gradient, tao->stepdirection);CHKERRQ(ierr);
316b591159SAlp Dener   ierr = VecScale(tao->stepdirection, -1.0);CHKERRQ(ierr);
326b591159SAlp Dener   ierr = TaoBNKBoundStep(tao, bnk->as_type, tao->stepdirection);CHKERRQ(ierr);
336b591159SAlp Dener   *ksp_reason = KSP_CONVERGED_ATOL;
346b591159SAlp Dener   *step_type = BNK_BFGS;
356b591159SAlp Dener   PetscFunctionReturn(0);
366b591159SAlp Dener }
376b591159SAlp Dener 
386b591159SAlp Dener static PetscErrorCode TaoSetFromOptions_BQNLS(PetscOptionItems *PetscOptionsObject,Tao tao)
396b591159SAlp Dener {
406b591159SAlp Dener   TAO_BNK        *bnk = (TAO_BNK *)tao->data;
416b591159SAlp Dener   TAO_BQNK       *bqnk = (TAO_BQNK*)bnk->ctx;
426b591159SAlp Dener   PetscErrorCode ierr;
436b591159SAlp Dener   KSPType        ksp_type;
446b591159SAlp Dener   PetscBool      is_spd;
456b591159SAlp Dener 
466b591159SAlp Dener   PetscFunctionBegin;
476b591159SAlp Dener   ierr = PetscOptionsHead(PetscOptionsObject,"Quasi-Newton-Krylov method for bound constrained optimization");CHKERRQ(ierr);
486b591159SAlp Dener   ierr = PetscOptionsEList("-tao_bqnls_as_type", "active set estimation method", "", BNK_AS, BNK_AS_TYPES, BNK_AS[bnk->as_type], &bnk->as_type, 0);CHKERRQ(ierr);
496b591159SAlp Dener   ierr = PetscOptionsReal("-tao_bqnls_epsilon", "(developer) tolerance used when computing actual and predicted reduction", "", bnk->epsilon, &bnk->epsilon,NULL);CHKERRQ(ierr);
506b591159SAlp Dener   ierr = PetscOptionsReal("-tao_bqnls_as_tol", "(developer) initial tolerance used when estimating actively bounded variables", "", bnk->as_tol, &bnk->as_tol,NULL);CHKERRQ(ierr);
516b591159SAlp Dener   ierr = PetscOptionsReal("-tao_bqnls_as_step", "(developer) step length used when estimating actively bounded variables", "", bnk->as_step, &bnk->as_step,NULL);CHKERRQ(ierr);
526b591159SAlp Dener   ierr = PetscOptionsInt("-tao_bqnls_max_cg_its", "number of BNCG iterations to take for each Newton step", "", bnk->max_cg_its, &bnk->max_cg_its,NULL);CHKERRQ(ierr);
536b591159SAlp Dener   ierr = PetscOptionsTail();CHKERRQ(ierr);
546b591159SAlp Dener   ierr = TaoSetFromOptions(bnk->bncg);CHKERRQ(ierr);
556b591159SAlp Dener   ierr = TaoLineSearchSetFromOptions(tao->linesearch);CHKERRQ(ierr);
566b591159SAlp Dener   ierr = KSPSetFromOptions(tao->ksp);CHKERRQ(ierr);
576b591159SAlp Dener   ierr = KSPGetType(tao->ksp,&ksp_type);CHKERRQ(ierr);
586b591159SAlp Dener   bnk->is_nash = bnk->is_gltr = bnk->is_stcg = PETSC_FALSE;
596b591159SAlp Dener   ierr = MatSetFromOptions(bqnk->B);CHKERRQ(ierr);
606b591159SAlp Dener   ierr = MatGetOption(bqnk->B, MAT_SPD, &is_spd);CHKERRQ(ierr);
616b591159SAlp Dener   if (!is_spd) SETERRQ(PetscObjectComm((PetscObject)tao), PETSC_ERR_ARG_INCOMP, "LMVM matrix must be symmetric positive-definite");
626b591159SAlp Dener   PetscFunctionReturn(0);
636b591159SAlp Dener }
646b591159SAlp Dener 
656b591159SAlp Dener PETSC_EXTERN PetscErrorCode TaoCreate_BQNLS(Tao tao)
666b591159SAlp Dener {
676b591159SAlp Dener   TAO_BNK        *bnk;
686b591159SAlp Dener   TAO_BQNK       *bqnk;
696b591159SAlp Dener   PetscErrorCode ierr;
706b591159SAlp Dener 
716b591159SAlp Dener   PetscFunctionBegin;
726b591159SAlp Dener   ierr = TaoCreate_BQNK(tao);CHKERRQ(ierr);
736b591159SAlp Dener   ierr = KSPSetOptionsPrefix(tao->ksp, "unused");CHKERRQ(ierr);
746b591159SAlp Dener   tao->ops->solve = TaoSolve_BNLS;
756b591159SAlp Dener   tao->ops->setfromoptions = TaoSetFromOptions_BQNLS;
766b591159SAlp Dener 
776b591159SAlp Dener   bnk = (TAO_BNK*)tao->data;
786b591159SAlp Dener   bnk->update_type = BNK_UPDATE_STEP;
796b591159SAlp Dener   bnk->computehessian = TaoBQNLSComputeHessian;
806b591159SAlp Dener   bnk->computestep = TaoBQNLSComputeStep;
816b591159SAlp Dener 
826b591159SAlp Dener   bqnk = (TAO_BQNK*)bnk->ctx;
836b591159SAlp Dener   ierr = MatSetOptionsPrefix(bqnk->B, "tao_bqnls_");CHKERRQ(ierr);
846b591159SAlp Dener   ierr = MatSetType(bqnk->B, MATLMVMBFGS);CHKERRQ(ierr);
856b591159SAlp Dener   PetscFunctionReturn(0);
866b591159SAlp Dener }