Actual source code: tsrhssplit.c
 
   petsc-3.10.3 2018-12-18
   
  1:  #include <petsc/private/tsimpl.h>
  3: static PetscErrorCode TSRHSSplitGetRHSSplit(TS ts,const char splitname[],TS_RHSSplitLink *isplit)
  4: {
  5:   PetscBool       found = PETSC_FALSE;
  6:   PetscErrorCode  ierr;
  9:   *isplit = ts->tsrhssplit;
 10:   /* look up the split */
 11:   while (*isplit) {
 12:     PetscStrcmp((*isplit)->splitname,splitname,&found);
 13:     if (found) break;
 14:     *isplit = (*isplit)->next;
 15:   }
 16:   return(0);
 17: }
 19: /*@C
 20:    TSRHSSplitSetIS - Set the index set for the specified split
 22:    Logically Collective on TS
 24:    Input Parameters:
 25: +  ts        - the TS context obtained from TSCreate()
 26: .  splitname - name of this split, if NULL the number of the split is used
 27: -  is        - the index set for part of the solution vector
 29:    Level: intermediate
 31: .seealso: TSRHSSplitGetIS()
 33: .keywords: TS, TSRHSSplit
 34: @*/
 35: PetscErrorCode TSRHSSplitSetIS(TS ts,const char splitname[],IS is)
 36: {
 37:   TS_RHSSplitLink newsplit,next = ts->tsrhssplit;
 38:   char            prefix[128];
 39:   PetscErrorCode  ierr;
 45:   PetscNew(&newsplit);
 46:   if (splitname) {
 47:     PetscStrallocpy(splitname,&newsplit->splitname);
 48:   } else {
 49:     PetscMalloc1(8,&newsplit->splitname);
 50:     PetscSNPrintf(newsplit->splitname,7,"%D",ts->num_rhs_splits);
 51:   }
 52:   PetscObjectReference((PetscObject)is);
 53:   newsplit->is = is;
 54:   TSCreate(PetscObjectComm((PetscObject)ts),&newsplit->ts);
 55:   PetscObjectIncrementTabLevel((PetscObject)newsplit->ts,(PetscObject)ts,1);
 56:   PetscLogObjectParent((PetscObject)ts,(PetscObject)newsplit->ts);
 57:   PetscSNPrintf(prefix,sizeof(prefix),"%srhsplit_%s_",((PetscObject)ts)->prefix ? ((PetscObject)ts)->prefix : "",newsplit->splitname);
 58:   TSSetOptionsPrefix(newsplit->ts,prefix);
 59:   if (!next) ts->tsrhssplit = newsplit;
 60:   else {
 61:     while (next->next) next = next->next;
 62:     next->next = newsplit;
 63:   }
 64:   ts->num_rhs_splits++;
 65:   return(0);
 66: }
 68: /*@C
 69:    TSRHSSplitGetIS - Retrieves the elements for a split as an IS
 71:    Logically Collective on TS
 73:    Input Parameters:
 74: +  ts        - the TS context obtained from TSCreate()
 75: -  splitname - name of this split
 77:    Output Parameters:
 78: -  is        - the index set for part of the solution vector
 80:    Level: intermediate
 82: .seealso: TSRHSSplitSetIS()
 84: .keywords: TS, TSRHSSplit
 85: @*/
 86: PetscErrorCode TSRHSSplitGetIS(TS ts,const char splitname[],IS *is)
 87: {
 88:   TS_RHSSplitLink isplit;
 89:   PetscErrorCode  ierr;
 93:   *is = NULL;
 94:   /* look up the split */
 95:   TSRHSSplitGetRHSSplit(ts,splitname,&isplit);
 96:   if (isplit) *is = isplit->is;
 97:   return(0);
 98: }
100: /*@C
101:    TSRHSSplitSetRHSFunction - Set the split right-hand-side functions.
103:    Logically Collective on TS
105:    Input Parameters:
106: +  ts        - the TS context obtained from TSCreate()
107: .  splitname - name of this split
108: .  r         - vector to hold the residual (or NULL to have it created internally)
109: .  rhsfunc   - the RHS function evaluation routine
110: -  ctx       - user-defined context for private data for the split function evaluation routine (may be NULL)
112:  Calling sequence of fun:
113: $  rhsfunc(TS ts,PetscReal t,Vec u,Vec f,ctx);
115: +  t    - time at step/stage being solved
116: .  u    - state vector
117: .  f    - function vector
118: -  ctx  - [optional] user-defined context for matrix evaluation routine (may be NULL)
120:  Level: beginner
122: .keywords: TS, timestep, set, ODE, Hamiltonian, Function
123: @*/
124: PetscErrorCode TSRHSSplitSetRHSFunction(TS ts,const char splitname[],Vec r,TSRHSFunction rhsfunc,void *ctx)
125: {
126:   TS_RHSSplitLink isplit;
127:   Vec             subvec,ralloc = NULL;
128:   PetscErrorCode  ierr;
134:   /* look up the split */
135:   TSRHSSplitGetRHSSplit(ts,splitname,&isplit);
136:   if (!isplit) SETERRQ1(PETSC_COMM_SELF,PETSC_ERR_USER,"The split %s is not created, check the split name or call TSRHSSplitSetIS() to create one",splitname);
138:   if (!r && ts->vec_sol) {
139:     VecGetSubVector(ts->vec_sol,isplit->is,&subvec);
140:     VecDuplicate(subvec,&ralloc);
141:     r    = ralloc;
142:     VecRestoreSubVector(ts->vec_sol,isplit->is,&subvec);
143:   }
144:   TSSetRHSFunction(isplit->ts,r,rhsfunc,ctx);
145:   VecDestroy(&ralloc);
146:   return(0);
147: }
149: /*@C
150:    TSRHSSplitGetSubTS - Get the sub-TS by split name.
152:    Logically Collective on TS
154:    Output Parameters:
155: +  splitname - the number of the split
156: -  subts - the array of TS contexts
158:    Level: advanced
160: .seealso: TSGetRHSSplitFunction()
161: @*/
162: PetscErrorCode TSRHSSplitGetSubTS(TS ts,const char splitname[],TS *subts)
163: {
164:   TS_RHSSplitLink isplit;
165:   PetscErrorCode  ierr;
170:   *subts = NULL;
171:   /* look up the split */
172:   TSRHSSplitGetRHSSplit(ts,splitname,&isplit);
173:   if (isplit) *subts = isplit->ts;
174:   return(0);
175: }
177: /*@C
178:    TSRHSSplitGetSubTSs - Get an array of all sub-TS contexts.
180:    Logically Collective on TS
182:    Output Parameters:
183: +  n - the number of splits
184: -  subksp - the array of TS contexts
186:    Note:
187:    After TSRHSSplitGetSubTS() the array of TSs is to be freed by the user with PetscFree()
188:    (not the TS just the array that contains them).
190:    Level: advanced
192: .seealso: TSGetRHSSplitFunction()
193: @*/
194: PetscErrorCode TSRHSSplitGetSubTSs(TS ts,PetscInt *n,TS *subts[])
195: {
196:   TS_RHSSplitLink ilink = ts->tsrhssplit;
197:   PetscInt        i = 0;
198:   PetscErrorCode  ierr;
202:   if (subts) {
203:     PetscMalloc1(ts->num_rhs_splits,subts);
204:     while (ilink) {
205:       (*subts)[i++] = ilink->ts;
206:       ilink = ilink->next;
207:     }
208:   }
209:   if (n) *n = ts->num_rhs_splits;
210:   return(0);
211: }