xref: /petsc/src/tao/leastsquares/impls/brgn/brgn.c (revision 48a46eb9bd028bec07ec0f396b1a3abb43f14558)
1463fc0ecSAlp Dener #include <../src/tao/leastsquares/impls/brgn/brgn.h> /*I "petsctao.h" I*/
2737f463aSAlp Dener 
3470ec3f8SXiang Huang #define BRGN_REGULARIZATION_USER   0
4470ec3f8SXiang Huang #define BRGN_REGULARIZATION_L2PROX 1
5a1c74439SHansol Suh #define BRGN_REGULARIZATION_L2PURE 2
6a1c74439SHansol Suh #define BRGN_REGULARIZATION_L1DICT 3
7cd1c4666STristan Konolige #define BRGN_REGULARIZATION_LM     4
8cd1c4666STristan Konolige #define BRGN_REGULARIZATION_TYPES  5
9a3c390cfSAlp Dener 
10cd1c4666STristan Konolige static const char *BRGN_REGULARIZATION_TABLE[64] = {"user", "l2prox", "l2pure", "l1dict", "lm"};
11a3c390cfSAlp Dener 
129371c9d4SSatish Balay static PetscErrorCode GNHessianProd(Mat H, Vec in, Vec out) {
130d71dc2bSXiang Huang   TAO_BRGN *gn;
140d71dc2bSXiang Huang 
150d71dc2bSXiang Huang   PetscFunctionBegin;
169566063dSJacob Faibussowitsch   PetscCall(MatShellGetContext(H, &gn));
179566063dSJacob Faibussowitsch   PetscCall(MatMult(gn->subsolver->ls_jac, in, gn->r_work));
189566063dSJacob Faibussowitsch   PetscCall(MatMultTranspose(gn->subsolver->ls_jac, gn->r_work, out));
19a3c390cfSAlp Dener   switch (gn->reg_type) {
20470ec3f8SXiang Huang   case BRGN_REGULARIZATION_USER:
219566063dSJacob Faibussowitsch     PetscCall(MatMult(gn->Hreg, in, gn->x_work));
229566063dSJacob Faibussowitsch     PetscCall(VecAXPY(out, gn->lambda, gn->x_work));
23a3c390cfSAlp Dener     break;
249371c9d4SSatish Balay   case BRGN_REGULARIZATION_L2PURE: PetscCall(VecAXPY(out, gn->lambda, in)); break;
259371c9d4SSatish Balay   case BRGN_REGULARIZATION_L2PROX: PetscCall(VecAXPY(out, gn->lambda, in)); break;
26470ec3f8SXiang Huang   case BRGN_REGULARIZATION_L1DICT:
27a3c390cfSAlp Dener     /* out = out + lambda*D'*(diag.*(D*in)) */
28a3c390cfSAlp Dener     if (gn->D) {
299566063dSJacob Faibussowitsch       PetscCall(MatMult(gn->D, in, gn->y)); /* y = D*in */
30a3c390cfSAlp Dener     } else {
319566063dSJacob Faibussowitsch       PetscCall(VecCopy(in, gn->y));
32a3c390cfSAlp Dener     }
339566063dSJacob 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 */
34a3c390cfSAlp Dener     if (gn->D) {
359566063dSJacob Faibussowitsch       PetscCall(MatMultTranspose(gn->D, gn->y_work, gn->x_work)); /* x_work = D'*(diag.*(D*in)) */
36a3c390cfSAlp Dener     } else {
379566063dSJacob Faibussowitsch       PetscCall(VecCopy(gn->y_work, gn->x_work));
38a3c390cfSAlp Dener     }
399566063dSJacob Faibussowitsch     PetscCall(VecAXPY(out, gn->lambda, gn->x_work));
40a3c390cfSAlp Dener     break;
41cd1c4666STristan Konolige   case BRGN_REGULARIZATION_LM:
429566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->x_work, gn->damping, in));
439566063dSJacob Faibussowitsch     PetscCall(VecAXPY(out, 1, gn->x_work));
44cd1c4666STristan Konolige     break;
45a3c390cfSAlp Dener   }
460d71dc2bSXiang Huang   PetscFunctionReturn(0);
470d71dc2bSXiang Huang }
489371c9d4SSatish Balay static PetscErrorCode ComputeDamping(TAO_BRGN *gn) {
49cd1c4666STristan Konolige   const PetscScalar *diag_ary;
50cd1c4666STristan Konolige   PetscScalar       *damping_ary;
51cd1c4666STristan Konolige   PetscInt           i, n;
52cd1c4666STristan Konolige 
53cd1c4666STristan Konolige   PetscFunctionBegin;
54cd1c4666STristan Konolige   /* update damping */
559566063dSJacob Faibussowitsch   PetscCall(VecGetArray(gn->damping, &damping_ary));
569566063dSJacob Faibussowitsch   PetscCall(VecGetArrayRead(gn->diag, &diag_ary));
579566063dSJacob Faibussowitsch   PetscCall(VecGetLocalSize(gn->damping, &n));
589371c9d4SSatish Balay   for (i = 0; i < n; i++) { damping_ary[i] = PetscClipInterval(diag_ary[i], PETSC_SQRT_MACHINE_EPSILON, PetscSqrtReal(PETSC_MAX_REAL)); }
599566063dSJacob Faibussowitsch   PetscCall(VecScale(gn->damping, gn->lambda));
609566063dSJacob Faibussowitsch   PetscCall(VecRestoreArray(gn->damping, &damping_ary));
619566063dSJacob Faibussowitsch   PetscCall(VecRestoreArrayRead(gn->diag, &diag_ary));
62cd1c4666STristan Konolige   PetscFunctionReturn(0);
63cd1c4666STristan Konolige }
64cd1c4666STristan Konolige 
659371c9d4SSatish Balay PetscErrorCode TaoBRGNGetDampingVector(Tao tao, Vec *d) {
66cd1c4666STristan Konolige   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
67cd1c4666STristan Konolige 
68cd1c4666STristan Konolige   PetscFunctionBegin;
693c859ba3SBarry Smith   PetscCheck(gn->reg_type == BRGN_REGULARIZATION_LM, PetscObjectComm((PetscObject)tao), PETSC_ERR_SUP, "Damping vector is only available if regularization type is lm.");
70cd1c4666STristan Konolige   *d = gn->damping;
71cd1c4666STristan Konolige   PetscFunctionReturn(0);
72cd1c4666STristan Konolige }
730d71dc2bSXiang Huang 
749371c9d4SSatish Balay static PetscErrorCode GNObjectiveGradientEval(Tao tao, Vec X, PetscReal *fcn, Vec G, void *ptr) {
750d71dc2bSXiang Huang   TAO_BRGN   *gn = (TAO_BRGN *)ptr;
768e85b1b3SXiang Huang   PetscInt    K; /* dimension of D*X */
777cea06e1SXiang Huang   PetscScalar yESum;
78a3c390cfSAlp Dener   PetscReal   f_reg;
790d71dc2bSXiang Huang 
800d71dc2bSXiang Huang   PetscFunctionBegin;
818e85b1b3SXiang Huang   /* compute objective *fcn*/
82a3c390cfSAlp Dener   /* compute first term 0.5*||ls_res||_2^2 */
839566063dSJacob Faibussowitsch   PetscCall(TaoComputeResidual(tao, X, tao->ls_res));
849566063dSJacob Faibussowitsch   PetscCall(VecDot(tao->ls_res, tao->ls_res, fcn));
85a3c390cfSAlp Dener   *fcn *= 0.5;
86a3c390cfSAlp Dener   /* compute gradient G */
879566063dSJacob Faibussowitsch   PetscCall(TaoComputeResidualJacobian(tao, X, tao->ls_jac, tao->ls_jac_pre));
889566063dSJacob Faibussowitsch   PetscCall(MatMultTranspose(tao->ls_jac, tao->ls_res, G));
89a3c390cfSAlp Dener   /* add the regularization contribution */
90a3c390cfSAlp Dener   switch (gn->reg_type) {
91470ec3f8SXiang Huang   case BRGN_REGULARIZATION_USER:
929566063dSJacob Faibussowitsch     PetscCall((*gn->regularizerobjandgrad)(tao, X, &f_reg, gn->x_work, gn->reg_obj_ctx));
93a3c390cfSAlp Dener     *fcn += gn->lambda * f_reg;
949566063dSJacob Faibussowitsch     PetscCall(VecAXPY(G, gn->lambda, gn->x_work));
95a3c390cfSAlp Dener     break;
96a1c74439SHansol Suh   case BRGN_REGULARIZATION_L2PURE:
97a1c74439SHansol Suh     /* compute f = f + lambda*0.5*xk'*xk */
989566063dSJacob Faibussowitsch     PetscCall(VecDot(X, X, &f_reg));
99a1c74439SHansol Suh     *fcn += gn->lambda * 0.5 * f_reg;
100a1c74439SHansol Suh     /* compute G = G + lambda*xk */
1019566063dSJacob Faibussowitsch     PetscCall(VecAXPY(G, gn->lambda, X));
102a1c74439SHansol Suh     break;
103470ec3f8SXiang Huang   case BRGN_REGULARIZATION_L2PROX:
1041fc140a9SXiang Huang     /* compute f = f + lambda*0.5*(xk - xkm1)'*(xk - xkm1) */
1059566063dSJacob Faibussowitsch     PetscCall(VecAXPBYPCZ(gn->x_work, 1.0, -1.0, 0.0, X, gn->x_old));
1069566063dSJacob Faibussowitsch     PetscCall(VecDot(gn->x_work, gn->x_work, &f_reg));
107a3c390cfSAlp Dener     *fcn += gn->lambda * 0.5 * f_reg;
108a3c390cfSAlp Dener     /* compute G = G + lambda*(xk - xkm1) */
1099566063dSJacob Faibussowitsch     PetscCall(VecAXPBYPCZ(G, gn->lambda, -gn->lambda, 1.0, X, gn->x_old));
110a3c390cfSAlp Dener     break;
111470ec3f8SXiang Huang   case BRGN_REGULARIZATION_L1DICT:
112a3c390cfSAlp Dener     /* compute f = f + lambda*sum(sqrt(y.^2+epsilon^2) - epsilon), where y = D*x*/
113a3c390cfSAlp Dener     if (gn->D) {
1149566063dSJacob Faibussowitsch       PetscCall(MatMult(gn->D, X, gn->y)); /* y = D*x */
115a3c390cfSAlp Dener     } else {
1169566063dSJacob Faibussowitsch       PetscCall(VecCopy(X, gn->y));
117a3c390cfSAlp Dener     }
1189566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->y_work, gn->y, gn->y));
1199566063dSJacob Faibussowitsch     PetscCall(VecShift(gn->y_work, gn->epsilon * gn->epsilon));
1209566063dSJacob Faibussowitsch     PetscCall(VecSqrtAbs(gn->y_work)); /* gn->y_work = sqrt(y.^2+epsilon^2) */
1219566063dSJacob Faibussowitsch     PetscCall(VecSum(gn->y_work, &yESum));
1229566063dSJacob Faibussowitsch     PetscCall(VecGetSize(gn->y, &K));
123a3c390cfSAlp Dener     *fcn += gn->lambda * (yESum - K * gn->epsilon);
1247cea06e1SXiang Huang     /* compute G = G + lambda*D'*(y./sqrt(y.^2+epsilon^2)),where y = D*x */
1259566063dSJacob Faibussowitsch     PetscCall(VecPointwiseDivide(gn->y_work, gn->y, gn->y_work)); /* reuse y_work = y./sqrt(y.^2+epsilon^2) */
126a3c390cfSAlp Dener     if (gn->D) {
1279566063dSJacob Faibussowitsch       PetscCall(MatMultTranspose(gn->D, gn->y_work, gn->x_work));
128a3c390cfSAlp Dener     } else {
1299566063dSJacob Faibussowitsch       PetscCall(VecCopy(gn->y_work, gn->x_work));
130a3c390cfSAlp Dener     }
1319566063dSJacob Faibussowitsch     PetscCall(VecAXPY(G, gn->lambda, gn->x_work));
132a3c390cfSAlp Dener     break;
133a3c390cfSAlp Dener   }
1340d71dc2bSXiang Huang   PetscFunctionReturn(0);
1350d71dc2bSXiang Huang }
1360d71dc2bSXiang Huang 
1379371c9d4SSatish Balay static PetscErrorCode GNComputeHessian(Tao tao, Vec X, Mat H, Mat Hpre, void *ptr) {
1388ac80d48SXiang Huang   TAO_BRGN    *gn = (TAO_BRGN *)ptr;
139cd1c4666STristan Konolige   PetscInt     i, n, cstart, cend;
140cd1c4666STristan Konolige   PetscScalar *cnorms, *diag_ary;
141737f463aSAlp Dener 
142737f463aSAlp Dener   PetscFunctionBegin;
1439566063dSJacob Faibussowitsch   PetscCall(TaoComputeResidualJacobian(tao, X, tao->ls_jac, tao->ls_jac_pre));
144*48a46eb9SPierre Jolivet   if (gn->mat_explicit) PetscCall(MatTransposeMatMult(tao->ls_jac, tao->ls_jac, MAT_REUSE_MATRIX, PETSC_DEFAULT, &gn->H));
1450d71dc2bSXiang Huang 
146a3c390cfSAlp Dener   switch (gn->reg_type) {
147470ec3f8SXiang Huang   case BRGN_REGULARIZATION_USER:
1489566063dSJacob Faibussowitsch     PetscCall((*gn->regularizerhessian)(tao, X, gn->Hreg, gn->reg_hess_ctx));
1491baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatAXPY(gn->H, 1.0, gn->Hreg, DIFFERENT_NONZERO_PATTERN));
150a3c390cfSAlp Dener     break;
151a1c74439SHansol Suh   case BRGN_REGULARIZATION_L2PURE:
1521baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatShift(gn->H, gn->lambda));
153a1c74439SHansol Suh     break;
154470ec3f8SXiang Huang   case BRGN_REGULARIZATION_L2PROX:
1551baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatShift(gn->H, gn->lambda));
156a3c390cfSAlp Dener     break;
157470ec3f8SXiang Huang   case BRGN_REGULARIZATION_L1DICT:
1587cea06e1SXiang 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 */
159a3c390cfSAlp Dener     if (gn->D) {
1609566063dSJacob Faibussowitsch       PetscCall(MatMult(gn->D, X, gn->y)); /* y = D*x */
161a3c390cfSAlp Dener     } else {
1629566063dSJacob Faibussowitsch       PetscCall(VecCopy(X, gn->y));
163a3c390cfSAlp Dener     }
1649566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->y_work, gn->y, gn->y));
1659566063dSJacob Faibussowitsch     PetscCall(VecShift(gn->y_work, gn->epsilon * gn->epsilon));
1669566063dSJacob Faibussowitsch     PetscCall(VecCopy(gn->y_work, gn->diag));                    /* gn->diag = y.^2+epsilon^2 */
1679566063dSJacob Faibussowitsch     PetscCall(VecSqrtAbs(gn->y_work));                           /* gn->y_work = sqrt(y.^2+epsilon^2) */
1689566063dSJacob Faibussowitsch     PetscCall(VecPointwiseMult(gn->diag, gn->y_work, gn->diag)); /* gn->diag = sqrt(y.^2+epsilon^2).^3 */
1699566063dSJacob Faibussowitsch     PetscCall(VecReciprocal(gn->diag));
1709566063dSJacob Faibussowitsch     PetscCall(VecScale(gn->diag, gn->epsilon * gn->epsilon));
1711baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatDiagonalSet(gn->H, gn->diag, ADD_VALUES));
172a3c390cfSAlp Dener     break;
173cd1c4666STristan Konolige   case BRGN_REGULARIZATION_LM:
174cd1c4666STristan Konolige     /* compute diagonal of J^T J */
1759566063dSJacob Faibussowitsch     PetscCall(MatGetSize(gn->parent->ls_jac, NULL, &n));
1769566063dSJacob Faibussowitsch     PetscCall(PetscMalloc1(n, &cnorms));
1779566063dSJacob Faibussowitsch     PetscCall(MatGetColumnNorms(gn->parent->ls_jac, NORM_2, cnorms));
1789566063dSJacob Faibussowitsch     PetscCall(MatGetOwnershipRangeColumn(gn->parent->ls_jac, &cstart, &cend));
1799566063dSJacob Faibussowitsch     PetscCall(VecGetArray(gn->diag, &diag_ary));
1809371c9d4SSatish Balay     for (i = 0; i < cend - cstart; i++) { diag_ary[i] = cnorms[cstart + i] * cnorms[cstart + i]; }
1819566063dSJacob Faibussowitsch     PetscCall(VecRestoreArray(gn->diag, &diag_ary));
1829566063dSJacob Faibussowitsch     PetscCall(PetscFree(cnorms));
1839566063dSJacob Faibussowitsch     PetscCall(ComputeDamping(gn));
1841baa6e33SBarry Smith     if (gn->mat_explicit) PetscCall(MatDiagonalSet(gn->H, gn->damping, ADD_VALUES));
185cd1c4666STristan Konolige     break;
186a3c390cfSAlp Dener   }
187e1e80dc8SAlp Dener   PetscFunctionReturn(0);
188e1e80dc8SAlp Dener }
189e1e80dc8SAlp Dener 
1909371c9d4SSatish Balay static PetscErrorCode GNHookFunction(Tao tao, PetscInt iter, void *ctx) {
1918fcddce6SStefano Zampini   TAO_BRGN *gn = (TAO_BRGN *)ctx;
192e1e80dc8SAlp Dener 
193e1e80dc8SAlp Dener   PetscFunctionBegin;
194e1e80dc8SAlp Dener   /* Update basic tao information from the subsolver */
195e1e80dc8SAlp Dener   gn->parent->nfuncs      = tao->nfuncs;
196e1e80dc8SAlp Dener   gn->parent->ngrads      = tao->ngrads;
197e1e80dc8SAlp Dener   gn->parent->nfuncgrads  = tao->nfuncgrads;
198e1e80dc8SAlp Dener   gn->parent->nhess       = tao->nhess;
199e1e80dc8SAlp Dener   gn->parent->niter       = tao->niter;
200e1e80dc8SAlp Dener   gn->parent->ksp_its     = tao->ksp_its;
201e1e80dc8SAlp Dener   gn->parent->ksp_tot_its = tao->ksp_tot_its;
202cd1c4666STristan Konolige   gn->parent->fc          = tao->fc;
2039566063dSJacob Faibussowitsch   PetscCall(TaoGetConvergedReason(tao, &gn->parent->reason));
204e1e80dc8SAlp Dener   /* Update the solution vectors */
205e1e80dc8SAlp Dener   if (iter == 0) {
2069566063dSJacob Faibussowitsch     PetscCall(VecSet(gn->x_old, 0.0));
207e1e80dc8SAlp Dener   } else {
2089566063dSJacob Faibussowitsch     PetscCall(VecCopy(tao->solution, gn->x_old));
2099566063dSJacob Faibussowitsch     PetscCall(VecCopy(tao->solution, gn->parent->solution));
210e1e80dc8SAlp Dener   }
211e1e80dc8SAlp Dener   /* Update the gradient */
2129566063dSJacob Faibussowitsch   PetscCall(VecCopy(tao->gradient, gn->parent->gradient));
213cd1c4666STristan Konolige 
214cd1c4666STristan Konolige   /* Update damping parameter for LM */
215cd1c4666STristan Konolige   if (gn->reg_type == BRGN_REGULARIZATION_LM) {
216cd1c4666STristan Konolige     if (iter > 0) {
217cd1c4666STristan Konolige       if (gn->fc_old > tao->fc) {
218cd1c4666STristan Konolige         gn->lambda = gn->lambda * gn->downhill_lambda_change;
219cd1c4666STristan Konolige       } else {
220cd1c4666STristan Konolige         /* uphill step */
221cd1c4666STristan Konolige         gn->lambda = gn->lambda * gn->uphill_lambda_change;
222cd1c4666STristan Konolige       }
223cd1c4666STristan Konolige     }
224cd1c4666STristan Konolige     gn->fc_old = tao->fc;
225cd1c4666STristan Konolige   }
226cd1c4666STristan Konolige 
227e1e80dc8SAlp Dener   /* Call general purpose update function */
2281baa6e33SBarry Smith   if (gn->parent->ops->update) PetscCall((*gn->parent->ops->update)(gn->parent, gn->parent->niter, gn->parent->user_update));
229737f463aSAlp Dener   PetscFunctionReturn(0);
230737f463aSAlp Dener }
231737f463aSAlp Dener 
2329371c9d4SSatish Balay static PetscErrorCode TaoSolve_BRGN(Tao tao) {
233737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
234737f463aSAlp Dener 
235737f463aSAlp Dener   PetscFunctionBegin;
2369566063dSJacob Faibussowitsch   PetscCall(TaoSolve(gn->subsolver));
237e1e80dc8SAlp Dener   /* Update basic tao information from the subsolver */
238e1e80dc8SAlp Dener   tao->nfuncs      = gn->subsolver->nfuncs;
239e1e80dc8SAlp Dener   tao->ngrads      = gn->subsolver->ngrads;
240e1e80dc8SAlp Dener   tao->nfuncgrads  = gn->subsolver->nfuncgrads;
241e1e80dc8SAlp Dener   tao->nhess       = gn->subsolver->nhess;
242e1e80dc8SAlp Dener   tao->niter       = gn->subsolver->niter;
243e1e80dc8SAlp Dener   tao->ksp_its     = gn->subsolver->ksp_its;
244e1e80dc8SAlp Dener   tao->ksp_tot_its = gn->subsolver->ksp_tot_its;
2459566063dSJacob Faibussowitsch   PetscCall(TaoGetConvergedReason(gn->subsolver, &tao->reason));
246e1e80dc8SAlp Dener   /* Update vectors */
2479566063dSJacob Faibussowitsch   PetscCall(VecCopy(gn->subsolver->solution, tao->solution));
2489566063dSJacob Faibussowitsch   PetscCall(VecCopy(gn->subsolver->gradient, tao->gradient));
249737f463aSAlp Dener   PetscFunctionReturn(0);
250737f463aSAlp Dener }
251737f463aSAlp Dener 
2529371c9d4SSatish Balay static PetscErrorCode TaoSetFromOptions_BRGN(Tao tao, PetscOptionItems *PetscOptionsObject) {
253737f463aSAlp Dener   TAO_BRGN     *gn = (TAO_BRGN *)tao->data;
254cd1c4666STristan Konolige   TaoLineSearch ls;
255737f463aSAlp Dener 
256737f463aSAlp Dener   PetscFunctionBegin;
257d0609cedSBarry 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.");
2589566063dSJacob 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));
2599566063dSJacob Faibussowitsch   PetscCall(PetscOptionsReal("-tao_brgn_regularizer_weight", "regularizer weight (default 1e-4)", "", gn->lambda, &gn->lambda, NULL));
2609566063dSJacob 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));
2619566063dSJacob 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));
2629566063dSJacob 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));
2639566063dSJacob Faibussowitsch   PetscCall(PetscOptionsEList("-tao_brgn_regularization_type", "regularization type", "", BRGN_REGULARIZATION_TABLE, BRGN_REGULARIZATION_TYPES, BRGN_REGULARIZATION_TABLE[gn->reg_type], &gn->reg_type, NULL));
264d0609cedSBarry Smith   PetscOptionsHeadEnd();
265cd1c4666STristan Konolige   /* set unit line search direction as the default when using the lm regularizer */
266cd1c4666STristan Konolige   if (gn->reg_type == BRGN_REGULARIZATION_LM) {
2679566063dSJacob Faibussowitsch     PetscCall(TaoGetLineSearch(gn->subsolver, &ls));
2689566063dSJacob Faibussowitsch     PetscCall(TaoLineSearchSetType(ls, TAOLINESEARCHUNIT));
269cd1c4666STristan Konolige   }
2709566063dSJacob Faibussowitsch   PetscCall(TaoSetFromOptions(gn->subsolver));
271737f463aSAlp Dener   PetscFunctionReturn(0);
272737f463aSAlp Dener }
273737f463aSAlp Dener 
2749371c9d4SSatish Balay static PetscErrorCode TaoView_BRGN(Tao tao, PetscViewer viewer) {
275737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
276737f463aSAlp Dener 
277737f463aSAlp Dener   PetscFunctionBegin;
2789566063dSJacob Faibussowitsch   PetscCall(PetscViewerASCIIPushTab(viewer));
2799566063dSJacob Faibussowitsch   PetscCall(TaoView(gn->subsolver, viewer));
2809566063dSJacob Faibussowitsch   PetscCall(PetscViewerASCIIPopTab(viewer));
281737f463aSAlp Dener   PetscFunctionReturn(0);
282737f463aSAlp Dener }
283737f463aSAlp Dener 
2849371c9d4SSatish Balay static PetscErrorCode TaoSetUp_BRGN(Tao tao) {
285737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
286737f463aSAlp Dener   PetscBool is_bnls, is_bntr, is_bntl;
2878e85b1b3SXiang Huang   PetscInt  i, n, N, K; /* dict has size K*N*/
288737f463aSAlp Dener 
289737f463aSAlp Dener   PetscFunctionBegin;
2903c859ba3SBarry Smith   PetscCheck(tao->ls_res, PetscObjectComm((PetscObject)tao), PETSC_ERR_ORDER, "TaoSetResidualRoutine() must be called before setup!");
2919566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)gn->subsolver, TAOBNLS, &is_bnls));
2929566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)gn->subsolver, TAOBNTR, &is_bntr));
2939566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)gn->subsolver, TAOBNTL, &is_bntl));
2943c859ba3SBarry Smith   PetscCheck((!is_bnls && !is_bntr && !is_bntl) || tao->ls_jac, PetscObjectComm((PetscObject)tao), PETSC_ERR_ORDER, "TaoSetResidualJacobianRoutine() must be called before setup!");
295*48a46eb9SPierre Jolivet   if (!tao->gradient) PetscCall(VecDuplicate(tao->solution, &tao->gradient));
296*48a46eb9SPierre Jolivet   if (!gn->x_work) PetscCall(VecDuplicate(tao->solution, &gn->x_work));
297*48a46eb9SPierre Jolivet   if (!gn->r_work) PetscCall(VecDuplicate(tao->ls_res, &gn->r_work));
298e1e80dc8SAlp Dener   if (!gn->x_old) {
2999566063dSJacob Faibussowitsch     PetscCall(VecDuplicate(tao->solution, &gn->x_old));
3009566063dSJacob Faibussowitsch     PetscCall(VecSet(gn->x_old, 0.0));
301e1e80dc8SAlp Dener   }
3027cea06e1SXiang Huang 
303470ec3f8SXiang Huang   if (BRGN_REGULARIZATION_L1DICT == gn->reg_type) {
3042036730cSSajid Ali     if (!gn->y) {
30530eeff36SXiang Huang       if (gn->D) {
3069566063dSJacob 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 */
3079566063dSJacob Faibussowitsch         PetscCall(MatCreateVecs(gn->D, NULL, &gn->y));
30830eeff36SXiang Huang       } else {
3099566063dSJacob Faibussowitsch         PetscCall(VecDuplicate(tao->solution, &gn->y)); /* If user does not setup dict matrix, use identiy matrix, K=N */
31030eeff36SXiang Huang       }
3119566063dSJacob Faibussowitsch       PetscCall(VecSet(gn->y, 0.0));
3127cea06e1SXiang Huang     }
313*48a46eb9SPierre Jolivet     if (!gn->y_work) PetscCall(VecDuplicate(gn->y, &gn->y_work));
3148ac80d48SXiang Huang     if (!gn->diag) {
3159566063dSJacob Faibussowitsch       PetscCall(VecDuplicate(gn->y, &gn->diag));
3169566063dSJacob Faibussowitsch       PetscCall(VecSet(gn->diag, 0.0));
3178ac80d48SXiang Huang     }
31830eeff36SXiang Huang   }
319cd1c4666STristan Konolige   if (BRGN_REGULARIZATION_LM == gn->reg_type) {
320*48a46eb9SPierre Jolivet     if (!gn->diag) PetscCall(MatCreateVecs(tao->ls_jac, &gn->diag, NULL));
321*48a46eb9SPierre Jolivet     if (!gn->damping) PetscCall(MatCreateVecs(tao->ls_jac, &gn->damping, NULL));
322cd1c4666STristan Konolige   }
3230d71dc2bSXiang Huang 
324e1e80dc8SAlp Dener   if (!tao->setupcalled) {
325737f463aSAlp Dener     /* Hessian setup */
3265eb5f4d6SAlp Dener     if (gn->mat_explicit) {
3279566063dSJacob Faibussowitsch       PetscCall(TaoComputeResidualJacobian(tao, tao->solution, tao->ls_jac, tao->ls_jac_pre));
3289566063dSJacob Faibussowitsch       PetscCall(MatTransposeMatMult(tao->ls_jac, tao->ls_jac, MAT_INITIAL_MATRIX, PETSC_DEFAULT, &gn->H));
3295eb5f4d6SAlp Dener     } else {
3309566063dSJacob Faibussowitsch       PetscCall(VecGetLocalSize(tao->solution, &n));
3319566063dSJacob Faibussowitsch       PetscCall(VecGetSize(tao->solution, &N));
3329566063dSJacob Faibussowitsch       PetscCall(MatCreate(PetscObjectComm((PetscObject)tao), &gn->H));
3339566063dSJacob Faibussowitsch       PetscCall(MatSetSizes(gn->H, n, n, N, N));
3349566063dSJacob Faibussowitsch       PetscCall(MatSetType(gn->H, MATSHELL));
3359566063dSJacob Faibussowitsch       PetscCall(MatSetOption(gn->H, MAT_SYMMETRIC, PETSC_TRUE));
3369566063dSJacob Faibussowitsch       PetscCall(MatShellSetOperation(gn->H, MATOP_MULT, (void (*)(void))GNHessianProd));
3379566063dSJacob Faibussowitsch       PetscCall(MatShellSetContext(gn->H, gn));
3385eb5f4d6SAlp Dener     }
3399566063dSJacob Faibussowitsch     PetscCall(MatSetUp(gn->H));
340a5b23f4aSJose E. Roman     /* Subsolver setup,include initial vector and dictionary D */
3419566063dSJacob Faibussowitsch     PetscCall(TaoSetUpdate(gn->subsolver, GNHookFunction, gn));
3429566063dSJacob Faibussowitsch     PetscCall(TaoSetSolution(gn->subsolver, tao->solution));
3431baa6e33SBarry Smith     if (tao->bounded) PetscCall(TaoSetVariableBounds(gn->subsolver, tao->XL, tao->XU));
3449566063dSJacob Faibussowitsch     PetscCall(TaoSetResidualRoutine(gn->subsolver, tao->ls_res, tao->ops->computeresidual, tao->user_lsresP));
3459566063dSJacob Faibussowitsch     PetscCall(TaoSetJacobianResidualRoutine(gn->subsolver, tao->ls_jac, tao->ls_jac, tao->ops->computeresidualjacobian, tao->user_lsjacP));
3469566063dSJacob Faibussowitsch     PetscCall(TaoSetObjectiveAndGradient(gn->subsolver, NULL, GNObjectiveGradientEval, gn));
3479566063dSJacob Faibussowitsch     PetscCall(TaoSetHessian(gn->subsolver, gn->H, gn->H, GNComputeHessian, gn));
348e1e80dc8SAlp Dener     /* Propagate some options down */
3499566063dSJacob Faibussowitsch     PetscCall(TaoSetTolerances(gn->subsolver, tao->gatol, tao->grtol, tao->gttol));
3509566063dSJacob Faibussowitsch     PetscCall(TaoSetMaximumIterations(gn->subsolver, tao->max_it));
3519566063dSJacob Faibussowitsch     PetscCall(TaoSetMaximumFunctionEvaluations(gn->subsolver, tao->max_funcs));
352737f463aSAlp Dener     for (i = 0; i < tao->numbermonitors; ++i) {
3539566063dSJacob Faibussowitsch       PetscCall(TaoSetMonitor(gn->subsolver, tao->monitor[i], tao->monitorcontext[i], tao->monitordestroy[i]));
3549566063dSJacob Faibussowitsch       PetscCall(PetscObjectReference((PetscObject)(tao->monitorcontext[i])));
355737f463aSAlp Dener     }
3569566063dSJacob Faibussowitsch     PetscCall(TaoSetUp(gn->subsolver));
357e1e80dc8SAlp Dener   }
358737f463aSAlp Dener   PetscFunctionReturn(0);
359737f463aSAlp Dener }
360737f463aSAlp Dener 
3619371c9d4SSatish Balay static PetscErrorCode TaoDestroy_BRGN(Tao tao) {
362737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
363737f463aSAlp Dener 
364737f463aSAlp Dener   PetscFunctionBegin;
365737f463aSAlp Dener   if (tao->setupcalled) {
3669566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&tao->gradient));
3679566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->x_work));
3689566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->r_work));
3699566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->x_old));
3709566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->diag));
3719566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->y));
3729566063dSJacob Faibussowitsch     PetscCall(VecDestroy(&gn->y_work));
373737f463aSAlp Dener   }
3749566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&gn->damping));
3759566063dSJacob Faibussowitsch   PetscCall(VecDestroy(&gn->diag));
3769566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&gn->H));
3779566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&gn->D));
3789566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&gn->Hreg));
3799566063dSJacob Faibussowitsch   PetscCall(TaoDestroy(&gn->subsolver));
380e1e80dc8SAlp Dener   gn->parent = NULL;
3819566063dSJacob Faibussowitsch   PetscCall(PetscFree(tao->data));
382737f463aSAlp Dener   PetscFunctionReturn(0);
383737f463aSAlp Dener }
384737f463aSAlp Dener 
3853850be85SAlp Dener /*MC
3863850be85SAlp Dener   TAOBRGN - Bounded Regularized Gauss-Newton method for solving nonlinear least-squares
3873850be85SAlp Dener             problems with bound constraints. This algorithm is a thin wrapper around TAOBNTL
388463fc0ecSAlp Dener             that constructs the Gauss-Newton problem with the user-provided least-squares
38960bb7533SHansol Suh             residual and Jacobian. The algorithm offers an L2-norm ("l2pure"), L2-norm proximal point ("l2prox")
39060bb7533SHansol Suh             regularizer, and L1-norm dictionary regularizer ("l1dict"), where we approximate the
39101b716f5SXiang Huang             L1-norm ||x||_1 by sum_i(sqrt(x_i^2+epsilon^2)-epsilon) with a small positive number epsilon.
392cd1c4666STristan Konolige             Also offered is the "lm" regularizer which uses a scaled diagonal of J^T J.
393cd1c4666STristan Konolige             With the "lm" regularizer, BRGN is a Levenberg-Marquardt optimizer.
39401b716f5SXiang Huang             The user can also provide own regularization function.
3953850be85SAlp Dener 
3963850be85SAlp Dener   Options Database Keys:
397cd1c4666STristan Konolige + -tao_brgn_regularization_type - regularization type ("user", "l2prox", "l2pure", "l1dict", "lm") (default "l2prox")
398c061e8e2SXiang Huang . -tao_brgn_regularizer_weight  - regularizer weight (default 1e-4)
399c061e8e2SXiang Huang - -tao_brgn_l1_smooth_epsilon   - L1-norm smooth approximation parameter: ||x||_1 = sum(sqrt(x.^2+epsilon^2)-epsilon) (default 1e-6)
4003850be85SAlp Dener 
4013850be85SAlp Dener   Level: beginner
4023850be85SAlp Dener M*/
4039371c9d4SSatish Balay PETSC_EXTERN PetscErrorCode TaoCreate_BRGN(Tao tao) {
404737f463aSAlp Dener   TAO_BRGN *gn;
405737f463aSAlp Dener 
406737f463aSAlp Dener   PetscFunctionBegin;
4079566063dSJacob Faibussowitsch   PetscCall(PetscNewLog(tao, &gn));
408737f463aSAlp Dener 
409737f463aSAlp Dener   tao->ops->destroy        = TaoDestroy_BRGN;
410737f463aSAlp Dener   tao->ops->setup          = TaoSetUp_BRGN;
411737f463aSAlp Dener   tao->ops->setfromoptions = TaoSetFromOptions_BRGN;
412737f463aSAlp Dener   tao->ops->view           = TaoView_BRGN;
413737f463aSAlp Dener   tao->ops->solve          = TaoSolve_BRGN;
414737f463aSAlp Dener 
4153ec1f749SStefano Zampini   tao->data                  = gn;
416d8bf7057SXiang Huang   gn->reg_type               = BRGN_REGULARIZATION_L2PROX;
417e1e80dc8SAlp Dener   gn->lambda                 = 1e-4;
4188ac80d48SXiang Huang   gn->epsilon                = 1e-6;
419cd1c4666STristan Konolige   gn->downhill_lambda_change = 1. / 5.;
420cd1c4666STristan Konolige   gn->uphill_lambda_change   = 1.5;
421e1e80dc8SAlp Dener   gn->parent                 = tao;
422737f463aSAlp Dener 
4239566063dSJacob Faibussowitsch   PetscCall(TaoCreate(PetscObjectComm((PetscObject)tao), &gn->subsolver));
4249566063dSJacob Faibussowitsch   PetscCall(TaoSetType(gn->subsolver, TAOBNLS));
4259566063dSJacob Faibussowitsch   PetscCall(TaoSetOptionsPrefix(gn->subsolver, "tao_brgn_subsolver_"));
426e1e80dc8SAlp Dener   PetscFunctionReturn(0);
427e1e80dc8SAlp Dener }
428e1e80dc8SAlp Dener 
429463fc0ecSAlp Dener /*@
430e1e80dc8SAlp Dener   TaoBRGNGetSubsolver - Get the pointer to the subsolver inside BRGN
431e1e80dc8SAlp Dener 
432e1e80dc8SAlp Dener   Collective on Tao
433e1e80dc8SAlp Dener 
434463fc0ecSAlp Dener   Level: advanced
435e1e80dc8SAlp Dener 
436e1e80dc8SAlp Dener   Input Parameters:
437e1e80dc8SAlp Dener +  tao - the Tao solver context
438e1e80dc8SAlp Dener -  subsolver - the Tao sub-solver context
439e1e80dc8SAlp Dener @*/
4409371c9d4SSatish Balay PetscErrorCode TaoBRGNGetSubsolver(Tao tao, Tao *subsolver) {
441e1e80dc8SAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
442e1e80dc8SAlp Dener 
443e1e80dc8SAlp Dener   PetscFunctionBegin;
444e1e80dc8SAlp Dener   *subsolver = gn->subsolver;
445737f463aSAlp Dener   PetscFunctionReturn(0);
446737f463aSAlp Dener }
447737f463aSAlp Dener 
448463fc0ecSAlp Dener /*@
449463fc0ecSAlp Dener   TaoBRGNSetRegularizerWeight - Set the regularizer weight for the Gauss-Newton least-squares algorithm
450737f463aSAlp Dener 
451737f463aSAlp Dener   Collective on Tao
452737f463aSAlp Dener 
453737f463aSAlp Dener   Input Parameters:
454737f463aSAlp Dener +  tao - the Tao solver context
4558e85b1b3SXiang Huang -  lambda - L1-norm regularizer weight
456463fc0ecSAlp Dener 
457463fc0ecSAlp Dener   Level: beginner
458737f463aSAlp Dener @*/
4599371c9d4SSatish Balay PetscErrorCode TaoBRGNSetRegularizerWeight(Tao tao, PetscReal lambda) {
460737f463aSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
461737f463aSAlp Dener 
4628ac80d48SXiang Huang   /* Initialize lambda here */
4630d71dc2bSXiang Huang 
464737f463aSAlp Dener   PetscFunctionBegin;
465737f463aSAlp Dener   gn->lambda = lambda;
466737f463aSAlp Dener   PetscFunctionReturn(0);
467737f463aSAlp Dener }
4680d71dc2bSXiang Huang 
469463fc0ecSAlp Dener /*@
4708ac80d48SXiang Huang   TaoBRGNSetL1SmoothEpsilon - Set the L1-norm smooth approximation parameter for L1-regularized least-squares algorithm
4718ac80d48SXiang Huang 
4728ac80d48SXiang Huang   Collective on Tao
4738ac80d48SXiang Huang 
4748ac80d48SXiang Huang   Input Parameters:
4758ac80d48SXiang Huang +  tao - the Tao solver context
4768ac80d48SXiang Huang -  epsilon - L1-norm smooth approximation parameter
477463fc0ecSAlp Dener 
478463fc0ecSAlp Dener   Level: advanced
4798ac80d48SXiang Huang @*/
4809371c9d4SSatish Balay PetscErrorCode TaoBRGNSetL1SmoothEpsilon(Tao tao, PetscReal epsilon) {
4818ac80d48SXiang Huang   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
4828ac80d48SXiang Huang 
4838ac80d48SXiang Huang   /* Initialize epsilon here */
4848ac80d48SXiang Huang 
4858ac80d48SXiang Huang   PetscFunctionBegin;
4868ac80d48SXiang Huang   gn->epsilon = epsilon;
4878ac80d48SXiang Huang   PetscFunctionReturn(0);
4888ac80d48SXiang Huang }
4898e85b1b3SXiang Huang 
490463fc0ecSAlp Dener /*@
4918e85b1b3SXiang Huang    TaoBRGNSetDictionaryMatrix - bind the dictionary matrix from user application context to gn->D, for compressed sensing (with least-squares problem)
4928e85b1b3SXiang Huang 
4938e85b1b3SXiang Huang    Input Parameters:
4948e85b1b3SXiang Huang +  tao  - the Tao context
495a2b725a8SWilliam Gropp -  dict - the user specified dictionary matrix.  We allow to set a null dictionary, which means identity matrix by default
4968e85b1b3SXiang Huang 
497463fc0ecSAlp Dener     Level: advanced
4988e85b1b3SXiang Huang @*/
4999371c9d4SSatish Balay PetscErrorCode TaoBRGNSetDictionaryMatrix(Tao tao, Mat dict) {
5008e85b1b3SXiang Huang   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
5018e85b1b3SXiang Huang   PetscFunctionBegin;
5028e85b1b3SXiang Huang   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
5038e85b1b3SXiang Huang   if (dict) {
5048e85b1b3SXiang Huang     PetscValidHeaderSpecific(dict, MAT_CLASSID, 2);
505a3c390cfSAlp Dener     PetscCheckSameComm(tao, 1, dict, 2);
5069566063dSJacob Faibussowitsch     PetscCall(PetscObjectReference((PetscObject)dict));
5078e85b1b3SXiang Huang   }
5089566063dSJacob Faibussowitsch   PetscCall(MatDestroy(&gn->D));
5091fc140a9SXiang Huang   gn->D = dict;
5108e85b1b3SXiang Huang   PetscFunctionReturn(0);
5118e85b1b3SXiang Huang }
5128e85b1b3SXiang Huang 
513a3c390cfSAlp Dener /*@C
514463fc0ecSAlp Dener    TaoBRGNSetRegularizerObjectiveAndGradientRoutine - Sets the user-defined regularizer call-back
515463fc0ecSAlp Dener    function into the algorithm.
516463fc0ecSAlp Dener 
517463fc0ecSAlp Dener    Input Parameters:
518463fc0ecSAlp Dener + tao - the Tao context
519463fc0ecSAlp Dener . func - function pointer for the regularizer value and gradient evaluation
520463fc0ecSAlp Dener - ctx - user context for the regularizer
521463fc0ecSAlp Dener 
522463fc0ecSAlp Dener    Level: advanced
523a3c390cfSAlp Dener @*/
5249371c9d4SSatish Balay PetscErrorCode TaoBRGNSetRegularizerObjectiveAndGradientRoutine(Tao tao, PetscErrorCode (*func)(Tao, Vec, PetscReal *, Vec, void *), void *ctx) {
525a3c390cfSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
526a3c390cfSAlp Dener 
527a3c390cfSAlp Dener   PetscFunctionBegin;
528a3c390cfSAlp Dener   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
5299371c9d4SSatish Balay   if (ctx) { gn->reg_obj_ctx = ctx; }
5309371c9d4SSatish Balay   if (func) { gn->regularizerobjandgrad = func; }
531a3c390cfSAlp Dener   PetscFunctionReturn(0);
532a3c390cfSAlp Dener }
533a3c390cfSAlp Dener 
534a3c390cfSAlp Dener /*@C
535463fc0ecSAlp Dener    TaoBRGNSetRegularizerHessianRoutine - Sets the user-defined regularizer call-back
536463fc0ecSAlp Dener    function into the algorithm.
537463fc0ecSAlp Dener 
538463fc0ecSAlp Dener    Input Parameters:
539463fc0ecSAlp Dener + tao - the Tao context
540463fc0ecSAlp Dener . Hreg - user-created matrix for the Hessian of the regularization term
541463fc0ecSAlp Dener . func - function pointer for the regularizer Hessian evaluation
542463fc0ecSAlp Dener - ctx - user context for the regularizer Hessian
543463fc0ecSAlp Dener 
544463fc0ecSAlp Dener    Level: advanced
545a3c390cfSAlp Dener @*/
5469371c9d4SSatish Balay PetscErrorCode TaoBRGNSetRegularizerHessianRoutine(Tao tao, Mat Hreg, PetscErrorCode (*func)(Tao, Vec, Mat, void *), void *ctx) {
547a3c390cfSAlp Dener   TAO_BRGN *gn = (TAO_BRGN *)tao->data;
548a3c390cfSAlp Dener 
549a3c390cfSAlp Dener   PetscFunctionBegin;
550a3c390cfSAlp Dener   PetscValidHeaderSpecific(tao, TAO_CLASSID, 1);
551a3c390cfSAlp Dener   if (Hreg) {
552a3c390cfSAlp Dener     PetscValidHeaderSpecific(Hreg, MAT_CLASSID, 2);
553a3c390cfSAlp Dener     PetscCheckSameComm(tao, 1, Hreg, 2);
55401b716f5SXiang Huang   } else SETERRQ(PetscObjectComm((PetscObject)tao), PETSC_ERR_ARG_WRONG, "NULL Hessian detected! User must provide valid Hessian for the regularizer.");
5559371c9d4SSatish Balay   if (ctx) { gn->reg_hess_ctx = ctx; }
5569371c9d4SSatish Balay   if (func) { gn->regularizerhessian = func; }
557a3c390cfSAlp Dener   if (Hreg) {
5589566063dSJacob Faibussowitsch     PetscCall(PetscObjectReference((PetscObject)Hreg));
5599566063dSJacob Faibussowitsch     PetscCall(MatDestroy(&gn->Hreg));
560a3c390cfSAlp Dener     gn->Hreg = Hreg;
561a3c390cfSAlp Dener   }
562a3c390cfSAlp Dener   PetscFunctionReturn(0);
563a3c390cfSAlp Dener }
564