xref: /petsc/src/tao/leastsquares/impls/brgn/brgn.c (revision 57d508425293f0bb93f59574d14951d8faac9af8)
1463fc0ecSAlp Dener #include <../src/tao/leastsquares/impls/brgn/brgn.h> /*I "petsctao.h" I*/
2737f463aSAlp Dener 
3c0b7dd19SHansol Suh const char *const TaoBRGNRegularizationTypes[] = {"user", "l2prox", "l2pure", "l1dict", "lm", "TaoBRGNRegularizationType", "TAOBRGN_REGULARIZATION_", NULL};
4a3c390cfSAlp Dener 
5d71ae5a4SJacob Faibussowitsch static PetscErrorCode GNHessianProd(Mat H, Vec in, Vec out)
6d71ae5a4SJacob Faibussowitsch {
70d71dc2bSXiang Huang   TAO_BRGN *gn;
80d71dc2bSXiang Huang 
90d71dc2bSXiang Huang   PetscFunctionBegin;
109566063dSJacob Faibussowitsch   PetscCall(MatShellGetContext(H, &gn));
119566063dSJacob Faibussowitsch   PetscCall(MatMult(gn->subsolver->ls_jac, in, gn->r_work));
129566063dSJacob Faibussowitsch   PetscCall(MatMultTranspose(gn->subsolver->ls_jac, gn->r_work, out));
13a3c390cfSAlp Dener   switch (gn->reg_type) {
14c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_USER:
159566063dSJacob Faibussowitsch     PetscCall(MatMult(gn->Hreg, in, gn->x_work));
169566063dSJacob Faibussowitsch     PetscCall(VecAXPY(out, gn->lambda, gn->x_work));
17a3c390cfSAlp Dener     break;
18c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L2PURE:
19d71ae5a4SJacob Faibussowitsch     PetscCall(VecAXPY(out, gn->lambda, in));
20d71ae5a4SJacob Faibussowitsch     break;
21c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L2PROX:
22d71ae5a4SJacob Faibussowitsch     PetscCall(VecAXPY(out, gn->lambda, in));
23d71ae5a4SJacob Faibussowitsch     break;
24c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L1DICT:
25a3c390cfSAlp Dener     /* out = out + lambda*D'*(diag.*(D*in)) */
26a3c390cfSAlp Dener     if (gn->D) {
279566063dSJacob Faibussowitsch       PetscCall(MatMult(gn->D, in, gn->y)); /* y = D*in */
28a3c390cfSAlp Dener     } else {
299566063dSJacob Faibussowitsch       PetscCall(VecCopy(in, gn->y));
30a3c390cfSAlp Dener     }
319566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->y_work, gn->diag, gn->y)); /* y_work = diag.*(D*in), where diag = epsilon^2 ./ sqrt(x.^2+epsilon^2).^3 */
32a3c390cfSAlp Dener     if (gn->D) {
339566063dSJacob Faibussowitsch       PetscCall(MatMultTranspose(gn->D, gn->y_work, gn->x_work)); /* x_work = D'*(diag.*(D*in)) */
34a3c390cfSAlp Dener     } else {
359566063dSJacob Faibussowitsch       PetscCall(VecCopy(gn->y_work, gn->x_work));
36a3c390cfSAlp Dener     }
379566063dSJacob Faibussowitsch     PetscCall(VecAXPY(out, gn->lambda, gn->x_work));
38a3c390cfSAlp Dener     break;
39c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_LM:
409566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->x_work, gn->damping, in));
419566063dSJacob Faibussowitsch     PetscCall(VecAXPY(out, 1, gn->x_work));
42cd1c4666STristan Konolige     break;
43a3c390cfSAlp Dener   }
443ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
450d71dc2bSXiang Huang }
46d71ae5a4SJacob Faibussowitsch static PetscErrorCode ComputeDamping(TAO_BRGN *gn)
47d71ae5a4SJacob Faibussowitsch {
48cd1c4666STristan Konolige   const PetscScalar *diag_ary;
49cd1c4666STristan Konolige   PetscScalar       *damping_ary;
50cd1c4666STristan Konolige   PetscInt           i, n;
51cd1c4666STristan Konolige 
52cd1c4666STristan Konolige   PetscFunctionBegin;
53cd1c4666STristan Konolige   /* update damping */
549566063dSJacob Faibussowitsch   PetscCall(VecGetArray(gn->damping, &damping_ary));
559566063dSJacob Faibussowitsch   PetscCall(VecGetArrayRead(gn->diag, &diag_ary));
569566063dSJacob Faibussowitsch   PetscCall(VecGetLocalSize(gn->damping, &n));
57ad540459SPierre Jolivet   for (i = 0; i < n; i++) damping_ary[i] = PetscClipInterval(diag_ary[i], PETSC_SQRT_MACHINE_EPSILON, PetscSqrtReal(PETSC_MAX_REAL));
589566063dSJacob Faibussowitsch   PetscCall(VecScale(gn->damping, gn->lambda));
599566063dSJacob Faibussowitsch   PetscCall(VecRestoreArray(gn->damping, &damping_ary));
609566063dSJacob Faibussowitsch   PetscCall(VecRestoreArrayRead(gn->diag, &diag_ary));
613ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
62cd1c4666STristan Konolige }
63c0b7dd19SHansol Suh /*@
64c0b7dd19SHansol Suh   TaoBRGNGetDampingVector - Get the damping vector $\mathrm{diag}(J^T J)$ from a `TAOBRGN` with `TAOBRGN_REGULARIZATION_LM` regularization
65cd1c4666STristan Konolige 
66c0b7dd19SHansol Suh   Collective
67c0b7dd19SHansol Suh 
68c0b7dd19SHansol Suh   Input Parameter:
69c0b7dd19SHansol Suh . tao - a `Tao` of type `TAOBRGN` with `TAOBRGN_REGULARIZATION_LM` regularization
70c0b7dd19SHansol Suh 
71c0b7dd19SHansol Suh   Output Parameter:
72c0b7dd19SHansol Suh . d - the damping vector
73c0b7dd19SHansol Suh 
74c0b7dd19SHansol Suh   Level: developer
75c0b7dd19SHansol Suh 
76c0b7dd19SHansol Suh .seealso: [](ch_tao), `Tao`, `TAOBRGN`, `TaoBRGNRegularzationTypes`
77c0b7dd19SHansol Suh @*/
78d71ae5a4SJacob Faibussowitsch PetscErrorCode TaoBRGNGetDampingVector(Tao tao, Vec *d)
79d71ae5a4SJacob Faibussowitsch {
80c0b7dd19SHansol Suh   PetscFunctionBegin;
81c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
82c0b7dd19SHansol Suh   PetscAssertPointer(d, 2);
83c0b7dd19SHansol Suh   PetscUseMethod((PetscObject)tao, "TaoBRGNGetDampingVector_C", (Tao, Vec *), (tao, d));
84c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
85c0b7dd19SHansol Suh }
86c0b7dd19SHansol Suh 
87c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNGetDampingVector_BRGN(Tao tao, Vec *d)
88c0b7dd19SHansol Suh {
89cd1c4666STristan Konolige   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
90cd1c4666STristan Konolige 
91cd1c4666STristan Konolige   PetscFunctionBegin;
92c0b7dd19SHansol Suh   PetscCheck(gn->reg_type == TAOBRGN_REGULARIZATION_LM, PetscObjectComm((PetscObject)tao), PETSC_ERR_SUP, "Damping vector is only available if regularization type is lm.");
93cd1c4666STristan Konolige   *d = gn->damping;
943ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
95cd1c4666STristan Konolige }
960d71dc2bSXiang Huang 
97d71ae5a4SJacob Faibussowitsch static PetscErrorCode GNObjectiveGradientEval(Tao tao, Vec X, PetscReal *fcn, Vec G, void *ptr)
98d71ae5a4SJacob Faibussowitsch {
990d71dc2bSXiang Huang   TAO_BRGN   *gn = (TAO_BRGN *)ptr;
1008e85b1b3SXiang Huang   PetscInt    K; /* dimension of D*X */
1017cea06e1SXiang Huang   PetscScalar yESum;
102a3c390cfSAlp Dener   PetscReal   f_reg;
1030d71dc2bSXiang Huang 
1040d71dc2bSXiang Huang   PetscFunctionBegin;
1058e85b1b3SXiang Huang   /* compute objective *fcn*/
106a3c390cfSAlp Dener   /* compute first term 0.5*||ls_res||_2^2 */
1079566063dSJacob Faibussowitsch   PetscCall(TaoComputeResidual(tao, X, tao->ls_res));
1089566063dSJacob Faibussowitsch   PetscCall(VecDot(tao->ls_res, tao->ls_res, fcn));
109a3c390cfSAlp Dener   *fcn *= 0.5;
110a3c390cfSAlp Dener   /* compute gradient G */
1119566063dSJacob Faibussowitsch   PetscCall(TaoComputeResidualJacobian(tao, X, tao->ls_jac, tao->ls_jac_pre));
1129566063dSJacob Faibussowitsch   PetscCall(MatMultTranspose(tao->ls_jac, tao->ls_res, G));
113a3c390cfSAlp Dener   /* add the regularization contribution */
114a3c390cfSAlp Dener   switch (gn->reg_type) {
115c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_USER:
1169566063dSJacob Faibussowitsch     PetscCall((*gn->regularizerobjandgrad)(tao, X, &f_reg, gn->x_work, gn->reg_obj_ctx));
117a3c390cfSAlp Dener     *fcn += gn->lambda * f_reg;
1189566063dSJacob Faibussowitsch     PetscCall(VecAXPY(G, gn->lambda, gn->x_work));
119a3c390cfSAlp Dener     break;
120c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L2PURE:
121a1c74439SHansol Suh     /* compute f = f + lambda*0.5*xk'*xk */
1229566063dSJacob Faibussowitsch     PetscCall(VecDot(X, X, &f_reg));
123a1c74439SHansol Suh     *fcn += gn->lambda * 0.5 * f_reg;
124a1c74439SHansol Suh     /* compute G = G + lambda*xk */
1259566063dSJacob Faibussowitsch     PetscCall(VecAXPY(G, gn->lambda, X));
126a1c74439SHansol Suh     break;
127c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L2PROX:
1281fc140a9SXiang Huang     /* compute f = f + lambda*0.5*(xk - xkm1)'*(xk - xkm1) */
1299566063dSJacob Faibussowitsch     PetscCall(VecAXPBYPCZ(gn->x_work, 1.0, -1.0, 0.0, X, gn->x_old));
1309566063dSJacob Faibussowitsch     PetscCall(VecDot(gn->x_work, gn->x_work, &f_reg));
131a3c390cfSAlp Dener     *fcn += gn->lambda * 0.5 * f_reg;
132a3c390cfSAlp Dener     /* compute G = G + lambda*(xk - xkm1) */
1339566063dSJacob Faibussowitsch     PetscCall(VecAXPBYPCZ(G, gn->lambda, -gn->lambda, 1.0, X, gn->x_old));
134a3c390cfSAlp Dener     break;
135c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L1DICT:
136a3c390cfSAlp Dener     /* compute f = f + lambda*sum(sqrt(y.^2+epsilon^2) - epsilon), where y = D*x*/
137a3c390cfSAlp Dener     if (gn->D) {
1389566063dSJacob Faibussowitsch       PetscCall(MatMult(gn->D, X, gn->y)); /* y = D*x */
139a3c390cfSAlp Dener     } else {
1409566063dSJacob Faibussowitsch       PetscCall(VecCopy(X, gn->y));
141a3c390cfSAlp Dener     }
1429566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->y_work, gn->y, gn->y));
1439566063dSJacob Faibussowitsch     PetscCall(VecShift(gn->y_work, gn->epsilon * gn->epsilon));
1449566063dSJacob Faibussowitsch     PetscCall(VecSqrtAbs(gn->y_work)); /* gn->y_work = sqrt(y.^2+epsilon^2) */
1459566063dSJacob Faibussowitsch     PetscCall(VecSum(gn->y_work, &yESum));
1469566063dSJacob Faibussowitsch     PetscCall(VecGetSize(gn->y, &K));
147a3c390cfSAlp Dener     *fcn += gn->lambda * (yESum - K * gn->epsilon);
1487cea06e1SXiang Huang     /* compute G = G + lambda*D'*(y./sqrt(y.^2+epsilon^2)),where y = D*x */
1499566063dSJacob Faibussowitsch     PetscCall(VecPointwiseDivide(gn->y_work, gn->y, gn->y_work)); /* reuse y_work = y./sqrt(y.^2+epsilon^2) */
150a3c390cfSAlp Dener     if (gn->D) {
1519566063dSJacob Faibussowitsch       PetscCall(MatMultTranspose(gn->D, gn->y_work, gn->x_work));
152a3c390cfSAlp Dener     } else {
1539566063dSJacob Faibussowitsch       PetscCall(VecCopy(gn->y_work, gn->x_work));
154a3c390cfSAlp Dener     }
1559566063dSJacob Faibussowitsch     PetscCall(VecAXPY(G, gn->lambda, gn->x_work));
156a3c390cfSAlp Dener     break;
157c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_LM:
158c0b7dd19SHansol Suh     break;
159c0b7dd19SHansol Suh   default:
160c0b7dd19SHansol Suh     break;
161a3c390cfSAlp Dener   }
1623ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
1630d71dc2bSXiang Huang }
1640d71dc2bSXiang Huang 
165d71ae5a4SJacob Faibussowitsch static PetscErrorCode GNComputeHessian(Tao tao, Vec X, Mat H, Mat Hpre, void *ptr)
166d71ae5a4SJacob Faibussowitsch {
1678ac80d48SXiang Huang   TAO_BRGN    *gn = (TAO_BRGN *)ptr;
168cd1c4666STristan Konolige   PetscInt     i, n, cstart, cend;
169cd1c4666STristan Konolige   PetscScalar *cnorms, *diag_ary;
170737f463aSAlp Dener 
171737f463aSAlp Dener   PetscFunctionBegin;
1729566063dSJacob Faibussowitsch   PetscCall(TaoComputeResidualJacobian(tao, X, tao->ls_jac, tao->ls_jac_pre));
173fb842aefSJose E. Roman   if (gn->mat_explicit) PetscCall(MatTransposeMatMult(tao->ls_jac, tao->ls_jac, MAT_REUSE_MATRIX, PETSC_DETERMINE, &gn->H));
1740d71dc2bSXiang Huang 
175a3c390cfSAlp Dener   switch (gn->reg_type) {
176c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_USER:
1779566063dSJacob Faibussowitsch     PetscCall((*gn->regularizerhessian)(tao, X, gn->Hreg, gn->reg_hess_ctx));
1781baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatAXPY(gn->H, 1.0, gn->Hreg, DIFFERENT_NONZERO_PATTERN));
179a3c390cfSAlp Dener     break;
180c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L2PURE:
1811baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatShift(gn->H, gn->lambda));
182a1c74439SHansol Suh     break;
183c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L2PROX:
1841baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatShift(gn->H, gn->lambda));
185a3c390cfSAlp Dener     break;
186c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_L1DICT:
1877cea06e1SXiang Huang     /* calculate and store diagonal matrix as a vector: diag = epsilon^2 ./ sqrt(x.^2+epsilon^2).^3* --> diag = epsilon^2 ./ sqrt(y.^2+epsilon^2).^3,where y = D*x */
188a3c390cfSAlp Dener     if (gn->D) {
1899566063dSJacob Faibussowitsch       PetscCall(MatMult(gn->D, X, gn->y)); /* y = D*x */
190a3c390cfSAlp Dener     } else {
1919566063dSJacob Faibussowitsch       PetscCall(VecCopy(X, gn->y));
192a3c390cfSAlp Dener     }
1939566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->y_work, gn->y, gn->y));
1949566063dSJacob Faibussowitsch     PetscCall(VecShift(gn->y_work, gn->epsilon * gn->epsilon));
1959566063dSJacob Faibussowitsch     PetscCall(VecCopy(gn->y_work, gn->diag));                    /* gn->diag = y.^2+epsilon^2 */
1969566063dSJacob Faibussowitsch     PetscCall(VecSqrtAbs(gn->y_work));                           /* gn->y_work = sqrt(y.^2+epsilon^2) */
1979566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->diag, gn->y_work, gn->diag)); /* gn->diag = sqrt(y.^2+epsilon^2).^3 */
1989566063dSJacob Faibussowitsch     PetscCall(VecReciprocal(gn->diag));
1999566063dSJacob Faibussowitsch     PetscCall(VecScale(gn->diag, gn->epsilon * gn->epsilon));
2001baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatDiagonalSet(gn->H, gn->diag, ADD_VALUES));
201a3c390cfSAlp Dener     break;
202c0b7dd19SHansol Suh   case TAOBRGN_REGULARIZATION_LM:
203cd1c4666STristan Konolige     /* compute diagonal of J^T J */
2049566063dSJacob Faibussowitsch     PetscCall(MatGetSize(gn->parent->ls_jac, NULL, &n));
2059566063dSJacob Faibussowitsch     PetscCall(PetscMalloc1(n, &cnorms));
2069566063dSJacob Faibussowitsch     PetscCall(MatGetColumnNorms(gn->parent->ls_jac, NORM_2, cnorms));
2079566063dSJacob Faibussowitsch     PetscCall(MatGetOwnershipRangeColumn(gn->parent->ls_jac, &cstart, &cend));
2089566063dSJacob Faibussowitsch     PetscCall(VecGetArray(gn->diag, &diag_ary));
209ad540459SPierre Jolivet     for (i = 0; i < cend - cstart; i++) diag_ary[i] = cnorms[cstart + i] * cnorms[cstart + i];
2109566063dSJacob Faibussowitsch     PetscCall(VecRestoreArray(gn->diag, &diag_ary));
2119566063dSJacob Faibussowitsch     PetscCall(PetscFree(cnorms));
2129566063dSJacob Faibussowitsch     PetscCall(ComputeDamping(gn));
2131baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatDiagonalSet(gn->H, gn->damping, ADD_VALUES));
214cd1c4666STristan Konolige     break;
215c0b7dd19SHansol Suh   default:
216c0b7dd19SHansol Suh     break;
217a3c390cfSAlp Dener   }
2183ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
219e1e80dc8SAlp Dener }
220e1e80dc8SAlp Dener 
221d71ae5a4SJacob Faibussowitsch static PetscErrorCode GNHookFunction(Tao tao, PetscInt iter, void *ctx)
222d71ae5a4SJacob Faibussowitsch {
2238fcddce6SStefano Zampini   TAO_BRGN *gn = (TAO_BRGN *)ctx;
224e1e80dc8SAlp Dener 
225e1e80dc8SAlp Dener   PetscFunctionBegin;
226e1e80dc8SAlp Dener   /* Update basic tao information from the subsolver */
227e1e80dc8SAlp Dener   gn->parent->nfuncs      = tao->nfuncs;
228e1e80dc8SAlp Dener   gn->parent->ngrads      = tao->ngrads;
229e1e80dc8SAlp Dener   gn->parent->nfuncgrads  = tao->nfuncgrads;
230e1e80dc8SAlp Dener   gn->parent->nhess       = tao->nhess;
231e1e80dc8SAlp Dener   gn->parent->niter       = tao->niter;
232e1e80dc8SAlp Dener   gn->parent->ksp_its     = tao->ksp_its;
233e1e80dc8SAlp Dener   gn->parent->ksp_tot_its = tao->ksp_tot_its;
234cd1c4666STristan Konolige   gn->parent->fc          = tao->fc;
2359566063dSJacob Faibussowitsch   PetscCall(TaoGetConvergedReason(tao, &gn->parent->reason));
236e1e80dc8SAlp Dener   /* Update the solution vectors */
237e1e80dc8SAlp Dener   if (iter == 0) {
2389566063dSJacob Faibussowitsch     PetscCall(VecSet(gn->x_old, 0.0));
239e1e80dc8SAlp Dener   } else {
2409566063dSJacob Faibussowitsch     PetscCall(VecCopy(tao->solution, gn->x_old));
2419566063dSJacob Faibussowitsch     PetscCall(VecCopy(tao->solution, gn->parent->solution));
242e1e80dc8SAlp Dener   }
243e1e80dc8SAlp Dener   /* Update the gradient */
2449566063dSJacob Faibussowitsch   PetscCall(VecCopy(tao->gradient, gn->parent->gradient));
245cd1c4666STristan Konolige 
246cd1c4666STristan Konolige   /* Update damping parameter for LM */
247c0b7dd19SHansol Suh   if (gn->reg_type == TAOBRGN_REGULARIZATION_LM) {
248cd1c4666STristan Konolige     if (iter > 0) {
249cd1c4666STristan Konolige       if (gn->fc_old > tao->fc) {
250cd1c4666STristan Konolige         gn->lambda = gn->lambda * gn->downhill_lambda_change;
251cd1c4666STristan Konolige       } else {
252cd1c4666STristan Konolige         /* uphill step */
253cd1c4666STristan Konolige         gn->lambda = gn->lambda * gn->uphill_lambda_change;
254cd1c4666STristan Konolige       }
255cd1c4666STristan Konolige     }
256cd1c4666STristan Konolige     gn->fc_old = tao->fc;
257cd1c4666STristan Konolige   }
258cd1c4666STristan Konolige 
259e1e80dc8SAlp Dener   /* Call general purpose update function */
2601baa6e33SBarry Smith   if (gn->parent->ops->update) PetscCall((*gn->parent->ops->update)(gn->parent, gn->parent->niter, gn->parent->user_update));
2613ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
262737f463aSAlp Dener }
263737f463aSAlp Dener 
264c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNGetRegularizationType_BRGN(Tao tao, TaoBRGNRegularizationType *type)
265c0b7dd19SHansol Suh {
266c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
267c0b7dd19SHansol Suh 
268c0b7dd19SHansol Suh   PetscFunctionBegin;
269c0b7dd19SHansol Suh   *type = gn->reg_type;
270c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
271c0b7dd19SHansol Suh }
272c0b7dd19SHansol Suh 
273c0b7dd19SHansol Suh /*@
274c0b7dd19SHansol Suh   TaoBRGNGetRegularizationType - Get the `TaoBRGNRegularizationType` of a `TAOBRGN`
275c0b7dd19SHansol Suh 
276c0b7dd19SHansol Suh   Not collective
277c0b7dd19SHansol Suh 
278c0b7dd19SHansol Suh   Input Parameter:
279c0b7dd19SHansol Suh . tao - a `Tao` of type `TAOBRGN`
280c0b7dd19SHansol Suh 
281c0b7dd19SHansol Suh   Output Parameter:
282c0b7dd19SHansol Suh . type - the `TaoBRGNRegularizationType`
283c0b7dd19SHansol Suh 
284c0b7dd19SHansol Suh   Level: advanced
285c0b7dd19SHansol Suh 
286c0b7dd19SHansol Suh .seealso: [](ch_tao), `Tao`, `TAOBRGN`, `TaoBRGNRegularizationType`, `TaoBRGNSetRegularizationType()`
287c0b7dd19SHansol Suh @*/
288c0b7dd19SHansol Suh PetscErrorCode TaoBRGNGetRegularizationType(Tao tao, TaoBRGNRegularizationType *type)
289c0b7dd19SHansol Suh {
290c0b7dd19SHansol Suh   PetscFunctionBegin;
291c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
292c0b7dd19SHansol Suh   PetscAssertPointer(type, 2);
293c0b7dd19SHansol Suh   PetscUseMethod((PetscObject)tao, "TaoBRGNGetRegularizationType_C", (Tao, TaoBRGNRegularizationType *), (tao, type));
294c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
295c0b7dd19SHansol Suh }
296c0b7dd19SHansol Suh 
297c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNSetRegularizationType_BRGN(Tao tao, TaoBRGNRegularizationType type)
298c0b7dd19SHansol Suh {
299c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
300c0b7dd19SHansol Suh 
301c0b7dd19SHansol Suh   PetscFunctionBegin;
302c0b7dd19SHansol Suh   gn->reg_type = type;
303c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
304c0b7dd19SHansol Suh }
305c0b7dd19SHansol Suh 
306c0b7dd19SHansol Suh /*@
307c0b7dd19SHansol Suh   TaoBRGNSetRegularizationType - Set the `TaoBRGNRegularizationType` of a `TAOBRGN`
308c0b7dd19SHansol Suh 
309c0b7dd19SHansol Suh   Logically collective
310c0b7dd19SHansol Suh 
311c0b7dd19SHansol Suh   Input Parameters:
312c0b7dd19SHansol Suh + tao  - a `Tao` of type `TAOBRGN`
313c0b7dd19SHansol Suh - type - the `TaoBRGNRegularizationType`
314c0b7dd19SHansol Suh 
315c0b7dd19SHansol Suh   Level: advanced
316c0b7dd19SHansol Suh 
317c0b7dd19SHansol Suh .seealso: [](ch_tao), `Tao`, `TAOBRGN`, `TaoBRGNRegularizationType`, `TaoBRGNGetRegularizationType`
318c0b7dd19SHansol Suh @*/
319c0b7dd19SHansol Suh PetscErrorCode TaoBRGNSetRegularizationType(Tao tao, TaoBRGNRegularizationType type)
320c0b7dd19SHansol Suh {
321c0b7dd19SHansol Suh   PetscFunctionBegin;
322c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
323c0b7dd19SHansol Suh   PetscValidLogicalCollectiveEnum(tao, type, 2);
324c0b7dd19SHansol Suh   PetscTryMethod((PetscObject)tao, "TaoBRGNSetRegularizationType_C", (Tao, TaoBRGNRegularizationType), (tao, type));
325c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
326c0b7dd19SHansol Suh }
327c0b7dd19SHansol Suh 
328d71ae5a4SJacob Faibussowitsch static PetscErrorCode TaoSolve_BRGN(Tao tao)
329d71ae5a4SJacob Faibussowitsch {
330737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
331737f463aSAlp Dener 
332737f463aSAlp Dener   PetscFunctionBegin;
3339566063dSJacob Faibussowitsch   PetscCall(TaoSolve(gn->subsolver));
334e1e80dc8SAlp Dener   /* Update basic tao information from the subsolver */
335e1e80dc8SAlp Dener   tao->nfuncs      = gn->subsolver->nfuncs;
336e1e80dc8SAlp Dener   tao->ngrads      = gn->subsolver->ngrads;
337e1e80dc8SAlp Dener   tao->nfuncgrads  = gn->subsolver->nfuncgrads;
338e1e80dc8SAlp Dener   tao->nhess       = gn->subsolver->nhess;
339e1e80dc8SAlp Dener   tao->niter       = gn->subsolver->niter;
340e1e80dc8SAlp Dener   tao->ksp_its     = gn->subsolver->ksp_its;
341e1e80dc8SAlp Dener   tao->ksp_tot_its = gn->subsolver->ksp_tot_its;
3429566063dSJacob Faibussowitsch   PetscCall(TaoGetConvergedReason(gn->subsolver, &tao->reason));
343e1e80dc8SAlp Dener   /* Update vectors */
3449566063dSJacob Faibussowitsch   PetscCall(VecCopy(gn->subsolver->solution, tao->solution));
3459566063dSJacob Faibussowitsch   PetscCall(VecCopy(gn->subsolver->gradient, tao->gradient));
3463ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
347737f463aSAlp Dener }
348737f463aSAlp Dener 
349ce78bad3SBarry Smith static PetscErrorCode TaoSetFromOptions_BRGN(Tao tao, PetscOptionItems PetscOptionsObject)
350d71ae5a4SJacob Faibussowitsch {
351737f463aSAlp Dener   TAO_BRGN     *gn = (TAO_BRGN *)tao->data;
352cd1c4666STristan Konolige   TaoLineSearch ls;
353737f463aSAlp Dener 
354737f463aSAlp Dener   PetscFunctionBegin;
355d0609cedSBarry Smith   PetscOptionsHeadBegin(PetscOptionsObject, "least-squares problems with regularizer: ||f(x)||^2 + lambda*g(x), g(x) = ||xk-xkm1||^2 or ||Dx||_1 or user defined function.");
3569566063dSJacob Faibussowitsch   PetscCall(PetscOptionsBool("-tao_brgn_mat_explicit", "switches the Hessian construction to be an explicit matrix rather than MATSHELL", "", gn->mat_explicit, &gn->mat_explicit, NULL));
3579566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-tao_brgn_regularizer_weight", "regularizer weight (default 1e-4)", "", gn->lambda, &gn->lambda, NULL));
3589566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-tao_brgn_l1_smooth_epsilon", "L1-norm smooth approximation parameter: ||x||_1 = sum(sqrt(x.^2+epsilon^2)-epsilon) (default 1e-6)", "", gn->epsilon, &gn->epsilon, NULL));
3599566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-tao_brgn_lm_downhill_lambda_change", "Factor to decrease trust region by on downhill steps", "", gn->downhill_lambda_change, &gn->downhill_lambda_change, NULL));
3609566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-tao_brgn_lm_uphill_lambda_change", "Factor to increase trust region by on uphill steps", "", gn->uphill_lambda_change, &gn->uphill_lambda_change, NULL));
361c0b7dd19SHansol Suh   PetscCall(PetscOptionsEnum("-tao_brgn_regularization_type", "regularization type", "", TaoBRGNRegularizationTypes, (PetscEnum)gn->reg_type, (PetscEnum *)&gn->reg_type, NULL));
362d0609cedSBarry Smith   PetscOptionsHeadEnd();
363cd1c4666STristan Konolige   /* set unit line search direction as the default when using the lm regularizer */
364c0b7dd19SHansol Suh   if (gn->reg_type == TAOBRGN_REGULARIZATION_LM) {
3659566063dSJacob Faibussowitsch     PetscCall(TaoGetLineSearch(gn->subsolver, &ls));
3669566063dSJacob Faibussowitsch     PetscCall(TaoLineSearchSetType(ls, TAOLINESEARCHUNIT));
367cd1c4666STristan Konolige   }
3689566063dSJacob Faibussowitsch   PetscCall(TaoSetFromOptions(gn->subsolver));
3693ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
370737f463aSAlp Dener }
371737f463aSAlp Dener 
372d71ae5a4SJacob Faibussowitsch static PetscErrorCode TaoView_BRGN(Tao tao, PetscViewer viewer)
373d71ae5a4SJacob Faibussowitsch {
374737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
375c0b7dd19SHansol Suh   PetscBool isascii;
376737f463aSAlp Dener 
377737f463aSAlp Dener   PetscFunctionBegin;
378c0b7dd19SHansol Suh   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &isascii));
379c0b7dd19SHansol Suh   if (isascii) {
380c0b7dd19SHansol Suh     PetscCall(PetscViewerASCIIPushTab(viewer));
381c0b7dd19SHansol Suh     PetscCall(PetscViewerASCIIPrintf(viewer, "Regularizer weight: %g\n", (double)gn->lambda));
382c0b7dd19SHansol Suh     PetscCall(PetscViewerASCIIPrintf(viewer, "BRGN Regularization Type: %s\n", TaoBRGNRegularizationTypes[gn->reg_type]));
383c0b7dd19SHansol Suh     switch (gn->reg_type) {
384c0b7dd19SHansol Suh     case TAOBRGN_REGULARIZATION_L1DICT:
385c0b7dd19SHansol Suh       PetscCall(PetscViewerASCIIPrintf(viewer, "L1 smooth epsilon: %g\n", (double)gn->epsilon));
386c0b7dd19SHansol Suh       break;
387c0b7dd19SHansol Suh     case TAOBRGN_REGULARIZATION_LM:
388c0b7dd19SHansol Suh       PetscCall(PetscViewerASCIIPrintf(viewer, "Downhill trust region decrease factor:: %g\n", (double)gn->downhill_lambda_change));
389c0b7dd19SHansol Suh       PetscCall(PetscViewerASCIIPrintf(viewer, "Uphill trust region increase factor:: %g\n", (double)gn->uphill_lambda_change));
390c0b7dd19SHansol Suh       break;
391c0b7dd19SHansol Suh     case TAOBRGN_REGULARIZATION_L2PROX:
392c0b7dd19SHansol Suh     case TAOBRGN_REGULARIZATION_L2PURE:
393c0b7dd19SHansol Suh     case TAOBRGN_REGULARIZATION_USER:
394c0b7dd19SHansol Suh     default:
395c0b7dd19SHansol Suh       break;
396c0b7dd19SHansol Suh     }
397c0b7dd19SHansol Suh     PetscCall(PetscViewerASCIIPopTab(viewer));
398c0b7dd19SHansol Suh   }
3999566063dSJacob Faibussowitsch   PetscCall(PetscViewerASCIIPushTab(viewer));
4009566063dSJacob Faibussowitsch   PetscCall(TaoView(gn->subsolver, viewer));
4019566063dSJacob Faibussowitsch   PetscCall(PetscViewerASCIIPopTab(viewer));
4023ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
403737f463aSAlp Dener }
404737f463aSAlp Dener 
405d71ae5a4SJacob Faibussowitsch static PetscErrorCode TaoSetUp_BRGN(Tao tao)
406d71ae5a4SJacob Faibussowitsch {
407737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
408737f463aSAlp Dener   PetscBool is_bnls, is_bntr, is_bntl;
4098e85b1b3SXiang Huang   PetscInt  i, n, N, K; /* dict has size K*N*/
410737f463aSAlp Dener 
411737f463aSAlp Dener   PetscFunctionBegin;
4123c859ba3SBarry Smith   PetscCheck(tao->ls_res, PetscObjectComm((PetscObject)tao), PETSC_ERR_ORDER, "TaoSetResidualRoutine() must be called before setup!");
4139566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)gn->subsolver, TAOBNLS, &is_bnls));
4149566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)gn->subsolver, TAOBNTR, &is_bntr));
4159566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)gn->subsolver, TAOBNTL, &is_bntl));
4163c859ba3SBarry Smith   PetscCheck((!is_bnls && !is_bntr && !is_bntl) || tao->ls_jac, PetscObjectComm((PetscObject)tao), PETSC_ERR_ORDER, "TaoSetResidualJacobianRoutine() must be called before setup!");
41748a46eb9SPierre Jolivet   if (!tao->gradient) PetscCall(VecDuplicate(tao->solution, &tao->gradient));
41848a46eb9SPierre Jolivet   if (!gn->x_work) PetscCall(VecDuplicate(tao->solution, &gn->x_work));
41948a46eb9SPierre Jolivet   if (!gn->r_work) PetscCall(VecDuplicate(tao->ls_res, &gn->r_work));
420e1e80dc8SAlp Dener   if (!gn->x_old) {
4219566063dSJacob Faibussowitsch     PetscCall(VecDuplicate(tao->solution, &gn->x_old));
4229566063dSJacob Faibussowitsch     PetscCall(VecSet(gn->x_old, 0.0));
423e1e80dc8SAlp Dener   }
4247cea06e1SXiang Huang 
425c0b7dd19SHansol Suh   if (TAOBRGN_REGULARIZATION_L1DICT == gn->reg_type) {
4262036730cSSajid Ali     if (!gn->y) {
42730eeff36SXiang Huang       if (gn->D) {
4289566063dSJacob Faibussowitsch         PetscCall(MatGetSize(gn->D, &K, &N)); /* Shell matrices still must have sizes defined. K = N for identity matrix, K=N-1 or N for gradient matrix */
4299566063dSJacob Faibussowitsch         PetscCall(MatCreateVecs(gn->D, NULL, &gn->y));
43030eeff36SXiang Huang       } else {
431aaa8cc7dSPierre Jolivet         PetscCall(VecDuplicate(tao->solution, &gn->y)); /* If user does not setup dict matrix, use identity matrix, K=N */
43230eeff36SXiang Huang       }
4339566063dSJacob Faibussowitsch       PetscCall(VecSet(gn->y, 0.0));
4347cea06e1SXiang Huang     }
43548a46eb9SPierre Jolivet     if (!gn->y_work) PetscCall(VecDuplicate(gn->y, &gn->y_work));
4368ac80d48SXiang Huang     if (!gn->diag) {
4379566063dSJacob Faibussowitsch       PetscCall(VecDuplicate(gn->y, &gn->diag));
4389566063dSJacob Faibussowitsch       PetscCall(VecSet(gn->diag, 0.0));
4398ac80d48SXiang Huang     }
44030eeff36SXiang Huang   }
441c0b7dd19SHansol Suh   if (TAOBRGN_REGULARIZATION_LM == gn->reg_type) {
44248a46eb9SPierre Jolivet     if (!gn->diag) PetscCall(MatCreateVecs(tao->ls_jac, &gn->diag, NULL));
44348a46eb9SPierre Jolivet     if (!gn->damping) PetscCall(MatCreateVecs(tao->ls_jac, &gn->damping, NULL));
444cd1c4666STristan Konolige   }
4450d71dc2bSXiang Huang 
446e1e80dc8SAlp Dener   if (!tao->setupcalled) {
447737f463aSAlp Dener     /* Hessian setup */
4485eb5f4d6SAlp Dener     if (gn->mat_explicit) {
4499566063dSJacob Faibussowitsch       PetscCall(TaoComputeResidualJacobian(tao, tao->solution, tao->ls_jac, tao->ls_jac_pre));
450fb842aefSJose E. Roman       PetscCall(MatTransposeMatMult(tao->ls_jac, tao->ls_jac, MAT_INITIAL_MATRIX, PETSC_DETERMINE, &gn->H));
4515eb5f4d6SAlp Dener     } else {
4529566063dSJacob Faibussowitsch       PetscCall(VecGetLocalSize(tao->solution, &n));
4539566063dSJacob Faibussowitsch       PetscCall(VecGetSize(tao->solution, &N));
4549566063dSJacob Faibussowitsch       PetscCall(MatCreate(PetscObjectComm((PetscObject)tao), &gn->H));
4559566063dSJacob Faibussowitsch       PetscCall(MatSetSizes(gn->H, n, n, N, N));
4569566063dSJacob Faibussowitsch       PetscCall(MatSetType(gn->H, MATSHELL));
4579566063dSJacob Faibussowitsch       PetscCall(MatSetOption(gn->H, MAT_SYMMETRIC, PETSC_TRUE));
458*57d50842SBarry Smith       PetscCall(MatShellSetOperation(gn->H, MATOP_MULT, (PetscErrorCodeFn *)GNHessianProd));
4599566063dSJacob Faibussowitsch       PetscCall(MatShellSetContext(gn->H, gn));
4605eb5f4d6SAlp Dener     }
4619566063dSJacob Faibussowitsch     PetscCall(MatSetUp(gn->H));
462a5b23f4aSJose E. Roman     /* Subsolver setup,include initial vector and dictionary D */
4639566063dSJacob Faibussowitsch     PetscCall(TaoSetUpdate(gn->subsolver, GNHookFunction, gn));
4649566063dSJacob Faibussowitsch     PetscCall(TaoSetSolution(gn->subsolver, tao->solution));
4651baa6e33SBarry Smith     if (tao->bounded) PetscCall(TaoSetVariableBounds(gn->subsolver, tao->XL, tao->XU));
4669566063dSJacob Faibussowitsch     PetscCall(TaoSetResidualRoutine(gn->subsolver, tao->ls_res, tao->ops->computeresidual, tao->user_lsresP));
4679566063dSJacob Faibussowitsch     PetscCall(TaoSetJacobianResidualRoutine(gn->subsolver, tao->ls_jac, tao->ls_jac, tao->ops->computeresidualjacobian, tao->user_lsjacP));
4689566063dSJacob Faibussowitsch     PetscCall(TaoSetObjectiveAndGradient(gn->subsolver, NULL, GNObjectiveGradientEval, gn));
4699566063dSJacob Faibussowitsch     PetscCall(TaoSetHessian(gn->subsolver, gn->H, gn->H, GNComputeHessian, gn));
470e1e80dc8SAlp Dener     /* Propagate some options down */
4719566063dSJacob Faibussowitsch     PetscCall(TaoSetTolerances(gn->subsolver, tao->gatol, tao->grtol, tao->gttol));
4729566063dSJacob Faibussowitsch     PetscCall(TaoSetMaximumIterations(gn->subsolver, tao->max_it));
4739566063dSJacob Faibussowitsch     PetscCall(TaoSetMaximumFunctionEvaluations(gn->subsolver, tao->max_funcs));
474737f463aSAlp Dener     for (i = 0; i < tao->numbermonitors; ++i) {
47510978b7dSBarry Smith       PetscCall(TaoMonitorSet(gn->subsolver, tao->monitor[i], tao->monitorcontext[i], tao->monitordestroy[i]));
476f4f49eeaSPierre Jolivet       PetscCall(PetscObjectReference((PetscObject)tao->monitorcontext[i]));
477737f463aSAlp Dener     }
4789566063dSJacob Faibussowitsch     PetscCall(TaoSetUp(gn->subsolver));
479e1e80dc8SAlp Dener   }
4803ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
481737f463aSAlp Dener }
482737f463aSAlp Dener 
483d71ae5a4SJacob Faibussowitsch static PetscErrorCode TaoDestroy_BRGN(Tao tao)
484d71ae5a4SJacob Faibussowitsch {
485737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
486737f463aSAlp Dener 
487737f463aSAlp Dener   PetscFunctionBegin;
488737f463aSAlp Dener   if (tao->setupcalled) {
4899566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&tao->gradient));
4909566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->x_work));
4919566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->r_work));
4929566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->x_old));
4939566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->diag));
4949566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->y));
4959566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->y_work));
496737f463aSAlp Dener   }
4979566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&gn->damping));
4989566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&gn->diag));
4999566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&gn->H));
5009566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&gn->D));
5019566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&gn->Hreg));
5029566063dSJacob Faibussowitsch   PetscCall(TaoDestroy(&gn->subsolver));
503e1e80dc8SAlp Dener   gn->parent = NULL;
5049566063dSJacob Faibussowitsch   PetscCall(PetscFree(tao->data));
505c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNGetRegularizationType_C", NULL));
506c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizationType_C", NULL));
507c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNGetDampingVector_C", NULL));
508c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetDictionaryMatrix_C", NULL));
509c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNGetSubsolver_C", NULL));
510c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizerWeight_C", NULL));
511c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetL1SmoothEpsilon_C", NULL));
512c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizerObjectiveAndGradientRoutine_C", NULL));
513c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizerHessianRoutine_C", NULL));
514c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
515c0b7dd19SHansol Suh }
516c0b7dd19SHansol Suh 
517c0b7dd19SHansol Suh /*@
518c0b7dd19SHansol Suh   TaoBRGNGetSubsolver - Get the pointer to the subsolver inside a `TAOBRGN`
519c0b7dd19SHansol Suh 
520c0b7dd19SHansol Suh   Collective
521c0b7dd19SHansol Suh 
522c0b7dd19SHansol Suh   Input Parameters:
523c0b7dd19SHansol Suh + tao       - the Tao solver context
524c0b7dd19SHansol Suh - subsolver - the `Tao` sub-solver context
525c0b7dd19SHansol Suh 
526c0b7dd19SHansol Suh   Level: advanced
527c0b7dd19SHansol Suh 
528c0b7dd19SHansol Suh .seealso: `Tao`, `Mat`, `TAOBRGN`
529c0b7dd19SHansol Suh @*/
530c0b7dd19SHansol Suh PetscErrorCode TaoBRGNGetSubsolver(Tao tao, Tao *subsolver)
531c0b7dd19SHansol Suh {
532c0b7dd19SHansol Suh   PetscFunctionBegin;
533c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
534c0b7dd19SHansol Suh   PetscUseMethod((PetscObject)tao, "TaoBRGNGetSubsolver_C", (Tao, Tao *), (tao, subsolver));
535c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
536c0b7dd19SHansol Suh }
537c0b7dd19SHansol Suh 
538c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNGetSubsolver_BRGN(Tao tao, Tao *subsolver)
539c0b7dd19SHansol Suh {
540c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
541c0b7dd19SHansol Suh 
542c0b7dd19SHansol Suh   PetscFunctionBegin;
543c0b7dd19SHansol Suh   *subsolver = gn->subsolver;
544c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
545c0b7dd19SHansol Suh }
546c0b7dd19SHansol Suh 
547c0b7dd19SHansol Suh /*@
548c0b7dd19SHansol Suh   TaoBRGNSetRegularizerWeight - Set the regularizer weight for the Gauss-Newton least-squares algorithm
549c0b7dd19SHansol Suh 
550c0b7dd19SHansol Suh   Collective
551c0b7dd19SHansol Suh 
552c0b7dd19SHansol Suh   Input Parameters:
553c0b7dd19SHansol Suh + tao    - the `Tao` solver context
554c0b7dd19SHansol Suh - lambda - L1-norm regularizer weight
555c0b7dd19SHansol Suh 
556c0b7dd19SHansol Suh   Level: beginner
557c0b7dd19SHansol Suh 
558c0b7dd19SHansol Suh .seealso: `Tao`, `Mat`, `TAOBRGN`
559c0b7dd19SHansol Suh @*/
560c0b7dd19SHansol Suh PetscErrorCode TaoBRGNSetRegularizerWeight(Tao tao, PetscReal lambda)
561c0b7dd19SHansol Suh {
562c0b7dd19SHansol Suh   PetscFunctionBegin;
563c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
564c0b7dd19SHansol Suh   PetscValidLogicalCollectiveReal(tao, lambda, 2);
565c0b7dd19SHansol Suh   PetscTryMethod((PetscObject)tao, "TaoBRGNSetRegularizerWeight_C", (Tao, PetscReal), (tao, lambda));
566c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
567c0b7dd19SHansol Suh }
568c0b7dd19SHansol Suh 
569c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNSetRegularizerWeight_BRGN(Tao tao, PetscReal lambda)
570c0b7dd19SHansol Suh {
571c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
572c0b7dd19SHansol Suh 
573c0b7dd19SHansol Suh   PetscFunctionBegin;
574c0b7dd19SHansol Suh   gn->lambda = lambda;
575c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
576c0b7dd19SHansol Suh }
577c0b7dd19SHansol Suh 
578c0b7dd19SHansol Suh /*@
579c0b7dd19SHansol Suh   TaoBRGNSetL1SmoothEpsilon - Set the L1-norm smooth approximation parameter for L1-regularized least-squares algorithm
580c0b7dd19SHansol Suh 
581c0b7dd19SHansol Suh   Collective
582c0b7dd19SHansol Suh 
583c0b7dd19SHansol Suh   Input Parameters:
584c0b7dd19SHansol Suh + tao     - the `Tao` solver context
585c0b7dd19SHansol Suh - epsilon - L1-norm smooth approximation parameter
586c0b7dd19SHansol Suh 
587c0b7dd19SHansol Suh   Level: advanced
588c0b7dd19SHansol Suh 
589c0b7dd19SHansol Suh .seealso: `Tao`, `Mat`, `TAOBRGN`
590c0b7dd19SHansol Suh @*/
591c0b7dd19SHansol Suh PetscErrorCode TaoBRGNSetL1SmoothEpsilon(Tao tao, PetscReal epsilon)
592c0b7dd19SHansol Suh {
593c0b7dd19SHansol Suh   PetscFunctionBegin;
594c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
595c0b7dd19SHansol Suh   PetscValidLogicalCollectiveReal(tao, epsilon, 2);
596c0b7dd19SHansol Suh   PetscTryMethod((PetscObject)tao, "TaoBRGNSetL1SmoothEpsilon_C", (Tao, PetscReal), (tao, epsilon));
597c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
598c0b7dd19SHansol Suh }
599c0b7dd19SHansol Suh 
600c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNSetL1SmoothEpsilon_BRGN(Tao tao, PetscReal epsilon)
601c0b7dd19SHansol Suh {
602c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
603c0b7dd19SHansol Suh 
604c0b7dd19SHansol Suh   PetscFunctionBegin;
605c0b7dd19SHansol Suh   gn->epsilon = epsilon;
606c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
607c0b7dd19SHansol Suh }
608c0b7dd19SHansol Suh 
609c0b7dd19SHansol Suh /*@
610c0b7dd19SHansol Suh   TaoBRGNSetDictionaryMatrix - bind the dictionary matrix from user application context to gn->D, for compressed sensing (with least-squares problem)
611c0b7dd19SHansol Suh 
612c0b7dd19SHansol Suh   Input Parameters:
613c0b7dd19SHansol Suh + tao  - the `Tao` context
614c0b7dd19SHansol Suh - dict - the user specified dictionary matrix.  We allow to set a `NULL` dictionary, which means identity matrix by default
615c0b7dd19SHansol Suh 
616c0b7dd19SHansol Suh   Level: advanced
617c0b7dd19SHansol Suh 
618c0b7dd19SHansol Suh .seealso: `Tao`, `Mat`, `TAOBRGN`
619c0b7dd19SHansol Suh @*/
620c0b7dd19SHansol Suh PetscErrorCode TaoBRGNSetDictionaryMatrix(Tao tao, Mat dict)
621c0b7dd19SHansol Suh {
622c0b7dd19SHansol Suh   PetscFunctionBegin;
623c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
624c0b7dd19SHansol Suh   PetscTryMethod((PetscObject)tao, "TaoBRGNSetDictionaryMatrix_C", (Tao, Mat), (tao, dict));
625c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
626c0b7dd19SHansol Suh }
627c0b7dd19SHansol Suh 
628c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNSetDictionaryMatrix_BRGN(Tao tao, Mat dict)
629c0b7dd19SHansol Suh {
630c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
631c0b7dd19SHansol Suh 
632c0b7dd19SHansol Suh   PetscFunctionBegin;
633c0b7dd19SHansol Suh   if (dict) {
634c0b7dd19SHansol Suh     PetscValidHeaderSpecific(dict, MAT_CLASSID, 2);
635c0b7dd19SHansol Suh     PetscCheckSameComm(tao, 1, dict, 2);
636c0b7dd19SHansol Suh     PetscCall(PetscObjectReference((PetscObject)dict));
637c0b7dd19SHansol Suh   }
638c0b7dd19SHansol Suh   PetscCall(MatDestroy(&gn->D));
639c0b7dd19SHansol Suh   gn->D = dict;
640c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
641c0b7dd19SHansol Suh }
642c0b7dd19SHansol Suh 
643c0b7dd19SHansol Suh /*@C
644c0b7dd19SHansol Suh   TaoBRGNSetRegularizerObjectiveAndGradientRoutine - Sets the user-defined regularizer call-back
645c0b7dd19SHansol Suh   function into the algorithm.
646c0b7dd19SHansol Suh 
647c0b7dd19SHansol Suh   Input Parameters:
648c0b7dd19SHansol Suh + tao  - the Tao context
649c0b7dd19SHansol Suh . func - function pointer for the regularizer value and gradient evaluation
650c0b7dd19SHansol Suh - ctx  - user context for the regularizer
651c0b7dd19SHansol Suh 
652c0b7dd19SHansol Suh   Calling sequence:
653c0b7dd19SHansol Suh + tao - the `Tao` context
654c0b7dd19SHansol Suh . u   - the location at which to compute the objective and gradient
655c0b7dd19SHansol Suh . val - location to store objective function value
656c0b7dd19SHansol Suh . g   - location to store gradient
657c0b7dd19SHansol Suh - ctx - user context for the regularizer Hessian
658c0b7dd19SHansol Suh 
659c0b7dd19SHansol Suh   Level: advanced
660c0b7dd19SHansol Suh 
661c0b7dd19SHansol Suh .seealso: `Tao`, `Mat`, `TAOBRGN`
662c0b7dd19SHansol Suh @*/
663c0b7dd19SHansol Suh PetscErrorCode TaoBRGNSetRegularizerObjectiveAndGradientRoutine(Tao tao, PetscErrorCode (*func)(Tao tao, Vec u, PetscReal *val, Vec g, void *ctx), void *ctx)
664c0b7dd19SHansol Suh {
665c0b7dd19SHansol Suh   PetscFunctionBegin;
666c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
667c0b7dd19SHansol Suh   PetscTryMethod((PetscObject)tao, "TaoBRGNSetRegularizerObjectiveAndGradientRoutine_C", (Tao, PetscErrorCode (*)(Tao, Vec, PetscReal *, Vec, void *), void *), (tao, func, ctx));
668c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
669c0b7dd19SHansol Suh }
670c0b7dd19SHansol Suh 
671c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNSetRegularizerObjectiveAndGradientRoutine_BRGN(Tao tao, PetscErrorCode (*func)(Tao tao, Vec u, PetscReal *val, Vec g, void *ctx), void *ctx)
672c0b7dd19SHansol Suh {
673c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
674c0b7dd19SHansol Suh 
675c0b7dd19SHansol Suh   PetscFunctionBegin;
676c0b7dd19SHansol Suh   if (ctx) gn->reg_obj_ctx = ctx;
677c0b7dd19SHansol Suh   if (func) gn->regularizerobjandgrad = func;
678c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
679c0b7dd19SHansol Suh }
680c0b7dd19SHansol Suh 
681c0b7dd19SHansol Suh /*@C
682c0b7dd19SHansol Suh   TaoBRGNSetRegularizerHessianRoutine - Sets the user-defined regularizer call-back
683c0b7dd19SHansol Suh   function into the algorithm.
684c0b7dd19SHansol Suh 
685c0b7dd19SHansol Suh   Input Parameters:
686c0b7dd19SHansol Suh + tao  - the `Tao` context
687c0b7dd19SHansol Suh . Hreg - user-created matrix for the Hessian of the regularization term
688c0b7dd19SHansol Suh . func - function pointer for the regularizer Hessian evaluation
689c0b7dd19SHansol Suh - ctx  - user context for the regularizer Hessian
690c0b7dd19SHansol Suh 
691c0b7dd19SHansol Suh   Calling sequence:
692c0b7dd19SHansol Suh + tao  - the `Tao` context
693c0b7dd19SHansol Suh . u    - the location at which to compute the Hessian
694c0b7dd19SHansol Suh . Hreg - user-created matrix for the Hessian of the regularization term
695c0b7dd19SHansol Suh - ctx  - user context for the regularizer Hessian
696c0b7dd19SHansol Suh 
697c0b7dd19SHansol Suh   Level: advanced
698c0b7dd19SHansol Suh 
699c0b7dd19SHansol Suh .seealso: `Tao`, `Mat`, `TAOBRGN`
700c0b7dd19SHansol Suh @*/
701c0b7dd19SHansol Suh PetscErrorCode TaoBRGNSetRegularizerHessianRoutine(Tao tao, Mat Hreg, PetscErrorCode (*func)(Tao tao, Vec u, Mat Hreg, void *ctx), void *ctx)
702c0b7dd19SHansol Suh {
703c0b7dd19SHansol Suh   PetscFunctionBegin;
704c0b7dd19SHansol Suh   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
705c0b7dd19SHansol Suh   PetscTryMethod((PetscObject)tao, "TaoBRGNSetRegularizerHessianRoutine_C", (Tao, Mat, PetscErrorCode (*)(Tao, Vec, Mat, void *), void *), (tao, Hreg, func, ctx));
706c0b7dd19SHansol Suh   PetscFunctionReturn(PETSC_SUCCESS);
707c0b7dd19SHansol Suh }
708c0b7dd19SHansol Suh 
709c0b7dd19SHansol Suh static PetscErrorCode TaoBRGNSetRegularizerHessianRoutine_BRGN(Tao tao, Mat Hreg, PetscErrorCode (*func)(Tao tao, Vec u, Mat Hreg, void *ctx), void *ctx)
710c0b7dd19SHansol Suh {
711c0b7dd19SHansol Suh   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
712c0b7dd19SHansol Suh 
713c0b7dd19SHansol Suh   PetscFunctionBegin;
714c0b7dd19SHansol Suh   if (Hreg) {
715c0b7dd19SHansol Suh     PetscValidHeaderSpecific(Hreg, MAT_CLASSID, 2);
716c0b7dd19SHansol Suh     PetscCheckSameComm(tao, 1, Hreg, 2);
717c0b7dd19SHansol Suh   } else SETERRQ(PetscObjectComm((PetscObject)tao), PETSC_ERR_ARG_WRONG, "NULL Hessian detected! User must provide valid Hessian for the regularizer.");
718c0b7dd19SHansol Suh   if (ctx) gn->reg_hess_ctx = ctx;
719c0b7dd19SHansol Suh   if (func) gn->regularizerhessian = func;
720c0b7dd19SHansol Suh   if (Hreg) {
721c0b7dd19SHansol Suh     PetscCall(PetscObjectReference((PetscObject)Hreg));
722c0b7dd19SHansol Suh     PetscCall(MatDestroy(&gn->Hreg));
723c0b7dd19SHansol Suh     gn->Hreg = Hreg;
724c0b7dd19SHansol Suh   }
7253ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
726737f463aSAlp Dener }
727737f463aSAlp Dener 
7283850be85SAlp Dener /*MC
7293850be85SAlp Dener   TAOBRGN - Bounded Regularized Gauss-Newton method for solving nonlinear least-squares
7302fe279fdSBarry Smith             problems with bound constraints. This algorithm is a thin wrapper around `TAOBNTL`
731463fc0ecSAlp Dener             that constructs the Gauss-Newton problem with the user-provided least-squares
73260bb7533SHansol Suh             residual and Jacobian. The algorithm offers an L2-norm ("l2pure"), L2-norm proximal point ("l2prox")
73360bb7533SHansol Suh             regularizer, and L1-norm dictionary regularizer ("l1dict"), where we approximate the
73401b716f5SXiang Huang             L1-norm ||x||_1 by sum_i(sqrt(x_i^2+epsilon^2)-epsilon) with a small positive number epsilon.
735cd1c4666STristan Konolige             Also offered is the "lm" regularizer which uses a scaled diagonal of J^T J.
7362fe279fdSBarry Smith             With the "lm" regularizer, `TAOBRGN` is a Levenberg-Marquardt optimizer.
73701b716f5SXiang Huang             The user can also provide own regularization function.
7383850be85SAlp Dener 
7393850be85SAlp Dener   Options Database Keys:
740cd1c4666STristan Konolige + -tao_brgn_regularization_type - regularization type ("user", "l2prox", "l2pure", "l1dict", "lm") (default "l2prox")
741c061e8e2SXiang Huang . -tao_brgn_regularizer_weight  - regularizer weight (default 1e-4)
742c061e8e2SXiang Huang - -tao_brgn_l1_smooth_epsilon   - L1-norm smooth approximation parameter: ||x||_1 = sum(sqrt(x.^2+epsilon^2)-epsilon) (default 1e-6)
7433850be85SAlp Dener 
7443850be85SAlp Dener   Level: beginner
7452fe279fdSBarry Smith 
7462fe279fdSBarry Smith .seealso: `Tao`, `TaoBRGNGetSubsolver()`, `TaoBRGNSetRegularizerWeight()`, `TaoBRGNSetL1SmoothEpsilon()`, `TaoBRGNSetDictionaryMatrix()`,
7472fe279fdSBarry Smith           `TaoBRGNSetRegularizerObjectiveAndGradientRoutine()`, `TaoBRGNSetRegularizerHessianRoutine()`
7483850be85SAlp Dener M*/
749d71ae5a4SJacob Faibussowitsch PETSC_EXTERN PetscErrorCode TaoCreate_BRGN(Tao tao)
750d71ae5a4SJacob Faibussowitsch {
751737f463aSAlp Dener   TAO_BRGN *gn;
752737f463aSAlp Dener 
753737f463aSAlp Dener   PetscFunctionBegin;
7544dfa11a4SJacob Faibussowitsch   PetscCall(PetscNew(&gn));
755737f463aSAlp Dener 
756737f463aSAlp Dener   tao->ops->destroy        = TaoDestroy_BRGN;
757737f463aSAlp Dener   tao->ops->setup          = TaoSetUp_BRGN;
758737f463aSAlp Dener   tao->ops->setfromoptions = TaoSetFromOptions_BRGN;
759737f463aSAlp Dener   tao->ops->view           = TaoView_BRGN;
760737f463aSAlp Dener   tao->ops->solve          = TaoSolve_BRGN;
761737f463aSAlp Dener 
762606f75f6SBarry Smith   PetscCall(TaoParametersInitialize(tao));
763606f75f6SBarry Smith 
7643ec1f749SStefano Zampini   tao->data                  = gn;
765c0b7dd19SHansol Suh   gn->reg_type               = TAOBRGN_REGULARIZATION_L2PROX;
766e1e80dc8SAlp Dener   gn->lambda                 = 1e-4;
7678ac80d48SXiang Huang   gn->epsilon                = 1e-6;
768cd1c4666STristan Konolige   gn->downhill_lambda_change = 1. / 5.;
769cd1c4666STristan Konolige   gn->uphill_lambda_change   = 1.5;
770e1e80dc8SAlp Dener   gn->parent                 = tao;
771737f463aSAlp Dener 
7729566063dSJacob Faibussowitsch   PetscCall(TaoCreate(PetscObjectComm((PetscObject)tao), &gn->subsolver));
7739566063dSJacob Faibussowitsch   PetscCall(TaoSetType(gn->subsolver, TAOBNLS));
7749566063dSJacob Faibussowitsch   PetscCall(TaoSetOptionsPrefix(gn->subsolver, "tao_brgn_subsolver_"));
775c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNGetRegularizationType_C", TaoBRGNGetRegularizationType_BRGN));
776c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizationType_C", TaoBRGNSetRegularizationType_BRGN));
777c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNGetDampingVector_C", TaoBRGNGetDampingVector_BRGN));
778c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetDictionaryMatrix_C", TaoBRGNSetDictionaryMatrix_BRGN));
779c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNGetSubsolver_C", TaoBRGNGetSubsolver_BRGN));
780c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizerWeight_C", TaoBRGNSetRegularizerWeight_BRGN));
781c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetL1SmoothEpsilon_C", TaoBRGNSetL1SmoothEpsilon_BRGN));
782c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizerObjectiveAndGradientRoutine_C", TaoBRGNSetRegularizerObjectiveAndGradientRoutine_BRGN));
783c0b7dd19SHansol Suh   PetscCall(PetscObjectComposeFunction((PetscObject)tao, "TaoBRGNSetRegularizerHessianRoutine_C", TaoBRGNSetRegularizerHessianRoutine_BRGN));
7843ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
785a3c390cfSAlp Dener }
786