<html><head><meta http-equiv="Content-Type" content="text/html; charset=us-ascii"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><br class=""><div><br class=""><blockquote type="cite" class=""><div class="">On Jul 21, 2020, at 5:11 PM, Alex Fleeter <<a href="mailto:luis.saturday@gmail.com" class="">luis.saturday@gmail.com</a>> wrote:</div><br class="Apple-interchange-newline"><div class=""><div dir="ltr" class="">Hi:<div class=""><br class=""></div><div class="">I guess I could rephrase my question like this. I am using MATNEST.</div><div class=""><br class=""></div><div class="">When I updated the matrix, I did assembly only for the sub-matrices. Then when I call the KSP solver, I can see that the pcsetup was called only in the first time the solver gets called. </div><div class=""><br class=""></div><div class="">I tried to add MatAssembly for the big nest matrix as a second test. Then, when I called my solver, I could see that the pcsetup gets called.</div><div class=""><br class=""></div><div class="">I assume that there is a state flag in the MatNest object that tells the solver if it needs to update the preconditioner. After an assembly routine gets called, this flag will be activated. Could you confirm that?</div></div></div></blockquote><div><br class=""></div>   PCSetUp() has </div><div><br class=""></div><div><div>  ierr = PetscObjectStateGet((PetscObject)pc->pmat,&matstate);CHKERRQ(ierr);</div><div>  ierr = MatGetNonzeroState(pc->pmat,&matnonzerostate);CHKERRQ(ierr);</div><div>  if (!pc->setupcalled) {</div><div>    ierr     = PetscInfo(pc,"Setting up PC for first time\n");CHKERRQ(ierr);</div><div>    pc->flag = DIFFERENT_NONZERO_PATTERN;</div><div>  } else if (matstate == pc->matstate) {</div><div>    ierr = PetscInfo(pc,"Leaving PC with identical preconditioner since operator is unchanged\n");CHKERRQ(ierr);</div><div>    PetscFunctionReturn(0);</div><div><br class=""></div><div>  so it rebuilds the preconditioner if the PetscObjectState of the matrix increased since the last application of the preconditioner. </div><div><br class=""></div><div>  Calling MatAssemblyBegin/End on a matrix always increases the matrix state (even if you have not changed the matrix entries). This is why</div><div><br class=""></div><div>"MatAssembly for the big nest matrix as a second test. Then, when I called my solver, I could see that the pcsetup gets called."</div><div><br class=""></div><div>  MatNest however is fundamentally broken because if you change entries in the sub-matrices you used to build the MatNest, the MatNest state is not increased hence any call to PCSetUp() with the MatNest as an argument does not trigger a new PCSetUp(). The author of MatNest knows about this flaw but has not bothered to fix it. </div><div><br class=""></div><div>  Thus every time you change values in the submatrices you must also call MatAssemblyBegin/End on the nest matrix (as you discovered) so the preconditioner  THAT DIRECTLY USES the MatNest knows it needs to rebuild.</div><div><br class=""></div><div>   Now I don't understand completely the code you have below and why or if there is a problem. The two KSP newctx->upper and newctx->lower do not know about the MatNest or use the MatNest state,  they only know about the sub matrices and the sub matrix states.  So if you change entries in the sub matrices then KSPSolve(newctx->upper) will trigger rebuilding the preconditioner inside it. That is you get the behavior you expect.  In other words blockpc_apply() will rebuild the two interior PCs each time you change the sub matrices. In other words for your custom PCShell you do not need to call MatAssemblyBegin/End on the nest matrix since the two interior PCs know directly about changes to the sub matrices, since their input is the sub matrices not the nest matrix.</div><div><br class=""></div><div>  Do you not get them rebuilding the preconditioner? If you run in the debugger and put a break point in PCSetUp() you should see those two interior PCs getting rebuilt exactly when they should be.</div><div><br class=""></div><div>  I hope this is not confusing: a KSP/PC that takes a MatNest argument will not rebuild the preconditioner that depends on the MatNest argument if you don't call MatAssemblyBegin/End on the MatNest. But any KSP/PC that depend on individual submatrices will get rebuilt correctly since they use the state of the individual sub matrices.</div><div><br class=""></div><div><br class=""></div><div>  Barry</div><div><br class=""></div><div><br class=""></div><div><br class=""></div><div><br class=""></div><div><br class=""></div><blockquote type="cite" class=""><div class=""><div dir="ltr" class=""><div class=""><br class=""></div><div class="">The problem is that I am trying to implement a shell preconditioner, which gets the sub-matrices in the setup routine like the following<br class=""></div><div class="">  MatNestGetSubMat(A, 0, 0, &newctx->submat[0]);<br class="">  MatNestGetSubMat(A, 0, 1, &newctx->submat[1]);<br class="">  MatNestGetSubMat(A, 1, 0, &newctx->submat[2]);<br class="">  MatNestGetSubMat(A, 1, 1, &newctx->submat[3]);<br class=""><br class="">  KSPSetOperators(newctx->upper, newctx->submat[0], newctx->submat[0]);<br class="">  KSPSetOperators(newctx->lower, newctx->submat[3], newctx->submat[3]);<br class=""></div><div class=""><br class=""></div><div class="">In the Apply stage, I do</div><div class=""><br class=""></div><div class="">PetscErrorCode blockpc_apply(PC pc, Vec x, Vec y)<br class="">{<br class="">  std::cout<<"APPLY PC\n";<br class="">  blockpc * newctx;<br class="">  PCShellGetContext(pc,(void**)&newctx);<br class=""><br class="">  Vec x_v, x_p, y_v, y_p;<br class="">  VecGetSubVector(x, newctx->is[0], &x_v);<br class="">  VecGetSubVector(x, newctx->is[1], &x_p);<br class="">  VecGetSubVector(y, newctx->is[0], &y_v);<br class="">  VecGetSubVector(y, newctx->is[1], &y_p);<br class=""><br class="">  KSPSolve( newctx -> upper, x_v, y_v );<br class="">  KSPSolve( newctx -> lower, x_p, y_p );<br class=""><br class="">  VecRestoreSubVector(x, newctx->is[0], &x_v);<br class="">  VecRestoreSubVector(x, newctx->is[1], &x_p);<br class="">  VecRestoreSubVector(y, newctx->is[0], &y_v);<br class="">  VecRestoreSubVector(y, newctx->is[1], &y_p);<br class=""><br class="">  return 0;<br class="">}<br class=""></div><div class=""><br class=""></div><div class="">I want to know if I shall call an assembly for the big nest matrix so that the pcsetup can be called everytime the solver is dealing with a new matrix.<br class=""></div><div class=""><br class=""></div><div class="">Thanks,</div><div class=""><br class=""></div><div class=""><br class=""></div><div class=""><br class=""></div><div class=""><br class=""></div><div class=""><br class=""></div></div><br class=""><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Tue, Jul 21, 2020 at 12:44 PM Alex Fleeter <<a href="mailto:luis.saturday@gmail.com" target="_blank" class="">luis.saturday@gmail.com</a>> wrote:<br class=""></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div dir="ltr" class="">Hi:<div class=""><br class=""></div><div class="">I want to ask under what circumstance will trigger a call for pc setup. </div><div class=""><br class=""></div><div class="">I call KSPSolve to solve with the same Mat object with different entry values each time. I can see that the pc setup is only called at the beginning of the first solve. </div><div class=""><br class=""></div><div class="">I tried to read the implementation, but quickly get lost...</div></div>
</blockquote></div>
</div></blockquote></div><br class=""></body></html>