Actual source code: stag1d.c
1: /*
2: Functions specific to the 1-dimensional implementation of DMStag
3: */
4: #include <petsc/private/dmstagimpl.h>
6: /*@C
7: DMStagCreate1d - Create an object to manage data living on the elements and vertices of a parallelized regular 1D grid.
9: Collective
11: Input Parameters:
12: + comm - MPI communicator
13: . bndx - boundary type: `DM_BOUNDARY_NONE`, `DM_BOUNDARY_PERIODIC`, or `DM_BOUNDARY_GHOSTED`
14: . M - global number of elements
15: . dof0 - number of degrees of freedom per vertex/0-cell
16: . dof1 - number of degrees of freedom per element/1-cell
17: . stencilType - ghost/halo region type: `DMSTAG_STENCIL_BOX` or `DMSTAG_STENCIL_NONE`
18: . stencilWidth - width, in elements, of halo/ghost region
19: - lx - array of local sizes, of length equal to the comm size, summing to `M` or `NULL`
21: Output Parameter:
22: . dm - the new `DMSTAG` object
24: Options Database Keys:
25: + -dm_view - calls `DMViewFromOptions()` at the conclusion of `DMSetUp()`
26: . -stag_grid_x <nx> - number of elements in the x direction
27: . -stag_ghost_stencil_width - width of ghost region, in elements
28: - -stag_boundary_type_x <none,ghosted,periodic> - `DMBoundaryType` value
30: Level: beginner
32: Notes:
33: You must call `DMSetUp()` after this call before using the `DM`.
34: If you wish to use the options database (see the keys above) to change values in the `DMSTAG`, you must call
35: `DMSetFromOptions()` after this function but before `DMSetUp()`.
37: .seealso: [](ch_stag), `DMSTAG`, `DMStagCreate2d()`, `DMStagCreate3d()`, `DMDestroy()`, `DMView()`, `DMCreateGlobalVector()`, `DMCreateLocalVector()`, `DMLocalToGlobalBegin()`, `DMDACreate1d()`
38: @*/
39: PETSC_EXTERN PetscErrorCode DMStagCreate1d(MPI_Comm comm, DMBoundaryType bndx, PetscInt M, PetscInt dof0, PetscInt dof1, DMStagStencilType stencilType, PetscInt stencilWidth, const PetscInt lx[], DM *dm)
40: {
41: PetscMPIInt size;
43: PetscFunctionBegin;
44: PetscCallMPI(MPI_Comm_size(comm, &size));
45: PetscCall(DMCreate(comm, dm));
46: PetscCall(DMSetDimension(*dm, 1));
47: PetscCall(DMStagInitialize(bndx, DM_BOUNDARY_NONE, DM_BOUNDARY_NONE, M, 0, 0, size, 0, 0, dof0, dof1, 0, 0, stencilType, stencilWidth, lx, NULL, NULL, *dm));
48: PetscFunctionReturn(PETSC_SUCCESS);
49: }
51: PETSC_INTERN PetscErrorCode DMStagRestrictSimple_1d(DM dmf, Vec xf_local, DM dmc, Vec xc_local)
52: {
53: PetscInt Mf, Mc, factorx, dof[2];
54: PetscInt xc, mc, nExtraxc, i, d;
55: PetscInt ileftf, ielemf, ileftc, ielemc;
56: const PetscScalar **arrf;
57: PetscScalar **arrc;
59: PetscFunctionBegin;
60: PetscCall(DMStagGetGlobalSizes(dmf, &Mf, NULL, NULL));
61: PetscCall(DMStagGetGlobalSizes(dmc, &Mc, NULL, NULL));
62: factorx = Mf / Mc;
63: PetscCall(DMStagGetDOF(dmc, &dof[0], &dof[1], NULL, NULL));
65: PetscCall(DMStagGetCorners(dmc, &xc, NULL, NULL, &mc, NULL, NULL, &nExtraxc, NULL, NULL));
66: PetscCall(VecZeroEntries(xc_local));
67: PetscCall(DMStagVecGetArray(dmf, xf_local, &arrf));
68: PetscCall(DMStagVecGetArray(dmc, xc_local, &arrc));
69: PetscCall(DMStagGetLocationSlot(dmf, DMSTAG_LEFT, 0, &ileftf));
70: PetscCall(DMStagGetLocationSlot(dmf, DMSTAG_ELEMENT, 0, &ielemf));
71: PetscCall(DMStagGetLocationSlot(dmc, DMSTAG_LEFT, 0, &ileftc));
72: PetscCall(DMStagGetLocationSlot(dmc, DMSTAG_ELEMENT, 0, &ielemc));
74: for (d = 0; d < dof[0]; ++d)
75: for (i = xc; i < xc + mc + nExtraxc; ++i) arrc[i][ileftc + d] = arrf[factorx * i][ileftf + d];
77: for (d = 0; d < dof[1]; ++d)
78: for (i = xc; i < xc + mc; ++i) {
79: if (factorx % 2 == 0) arrc[i][ielemc + d] = 0.5 * (arrf[factorx * i + factorx / 2 - 1][ielemf + d] + arrf[factorx * i + factorx / 2][ielemf + d]);
80: else arrc[i][ielemc + d] = arrf[factorx * i + factorx / 2][ielemf + d];
81: }
83: PetscCall(DMStagVecRestoreArray(dmf, xf_local, &arrf));
84: PetscCall(DMStagVecRestoreArray(dmc, xc_local, &arrc));
85: PetscFunctionReturn(PETSC_SUCCESS);
86: }
88: PETSC_INTERN PetscErrorCode DMStagSetUniformCoordinatesExplicit_1d(DM dm, PetscReal xmin, PetscReal xmax)
89: {
90: DM_Stag *stagCoord;
91: DM dmCoord;
92: Vec coordLocal;
93: PetscReal h, min;
94: PetscScalar **arr;
95: PetscInt start_ghost, n_ghost, s;
96: PetscInt ileft, ielement;
98: PetscFunctionBegin;
99: PetscCall(DMGetCoordinateDM(dm, &dmCoord));
100: stagCoord = (DM_Stag *)dmCoord->data;
101: for (s = 0; s < 2; ++s) {
102: PetscCheck(stagCoord->dof[s] == 0 || stagCoord->dof[s] == 1, PetscObjectComm((PetscObject)dm), PETSC_ERR_PLIB, "Coordinate DM in 1 dimensions must have 0 or 1 dof on each stratum, but stratum %" PetscInt_FMT " has %" PetscInt_FMT " dof", s,
103: stagCoord->dof[s]);
104: }
105: PetscCall(DMCreateLocalVector(dmCoord, &coordLocal));
107: PetscCall(DMStagVecGetArray(dmCoord, coordLocal, &arr));
108: if (stagCoord->dof[0]) PetscCall(DMStagGetLocationSlot(dmCoord, DMSTAG_LEFT, 0, &ileft));
109: if (stagCoord->dof[1]) PetscCall(DMStagGetLocationSlot(dmCoord, DMSTAG_ELEMENT, 0, &ielement));
110: PetscCall(DMStagGetGhostCorners(dmCoord, &start_ghost, NULL, NULL, &n_ghost, NULL, NULL));
112: min = xmin;
113: h = (xmax - xmin) / stagCoord->N[0];
115: for (PetscInt ind = start_ghost; ind < start_ghost + n_ghost; ++ind) {
116: if (stagCoord->dof[0]) {
117: const PetscReal off = 0.0;
118: arr[ind][ileft] = min + ((PetscReal)ind + off) * h;
119: }
120: if (stagCoord->dof[1]) {
121: const PetscReal off = 0.5;
122: arr[ind][ielement] = min + ((PetscReal)ind + off) * h;
123: }
124: }
125: PetscCall(DMStagVecRestoreArray(dmCoord, coordLocal, &arr));
126: PetscCall(DMSetCoordinatesLocal(dm, coordLocal));
127: PetscCall(VecDestroy(&coordLocal));
128: PetscFunctionReturn(PETSC_SUCCESS);
129: }
131: /* Helper functions used in DMSetUp_Stag() */
132: static PetscErrorCode DMStagComputeLocationOffsets_1d(DM);
134: PETSC_INTERN PetscErrorCode DMSetUp_Stag_1d(DM dm)
135: {
136: DM_Stag *const stag = (DM_Stag *)dm->data;
137: PetscMPIInt size, rank;
138: MPI_Comm comm;
139: PetscInt j;
141: PetscFunctionBegin;
142: PetscCall(PetscObjectGetComm((PetscObject)dm, &comm));
143: PetscCallMPI(MPI_Comm_size(comm, &size));
144: PetscCallMPI(MPI_Comm_rank(comm, &rank));
146: /* Check Global size */
147: PetscCheck(stag->N[0] >= 1, comm, PETSC_ERR_ARG_OUTOFRANGE, "Global grid size of %" PetscInt_FMT " < 1 specified", stag->N[0]);
149: /* Local sizes */
150: PetscCheck(stag->N[0] >= size, comm, PETSC_ERR_ARG_OUTOFRANGE, "More ranks (%d) than elements (%" PetscInt_FMT ") specified", size, stag->N[0]);
151: if (!stag->l[0]) {
152: /* Divide equally, giving an extra elements to higher ranks */
153: PetscCall(PetscMalloc1(stag->nRanks[0], &stag->l[0]));
154: for (j = 0; j < stag->nRanks[0]; ++j) stag->l[0][j] = stag->N[0] / stag->nRanks[0] + (stag->N[0] % stag->nRanks[0] > j ? 1 : 0);
155: }
156: {
157: PetscInt Nchk = 0;
158: for (j = 0; j < size; ++j) Nchk += stag->l[0][j];
159: PetscCheck(Nchk == stag->N[0], comm, PETSC_ERR_ARG_OUTOFRANGE, "Sum of specified local sizes (%" PetscInt_FMT ") is not equal to global size (%" PetscInt_FMT ")", Nchk, stag->N[0]);
160: }
161: stag->n[0] = stag->l[0][rank];
163: /* Rank (trivial in 1d) */
164: stag->rank[0] = rank;
165: stag->firstRank[0] = (PetscBool)(rank == 0);
166: stag->lastRank[0] = (PetscBool)(rank == size - 1);
168: /* Local (unghosted) numbers of entries */
169: stag->entriesPerElement = stag->dof[0] + stag->dof[1];
170: switch (stag->boundaryType[0]) {
171: case DM_BOUNDARY_NONE:
172: case DM_BOUNDARY_GHOSTED:
173: stag->entries = stag->n[0] * stag->entriesPerElement + (stag->lastRank[0] ? stag->dof[0] : 0);
174: break;
175: case DM_BOUNDARY_PERIODIC:
176: stag->entries = stag->n[0] * stag->entriesPerElement;
177: break;
178: default:
179: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unsupported x boundary type %s", DMBoundaryTypes[stag->boundaryType[0]]);
180: }
182: /* Starting element */
183: stag->start[0] = 0;
184: for (j = 0; j < stag->rank[0]; ++j) stag->start[0] += stag->l[0][j];
186: /* Local/ghosted size and starting element */
187: switch (stag->boundaryType[0]) {
188: case DM_BOUNDARY_NONE:
189: switch (stag->stencilType) {
190: case DMSTAG_STENCIL_NONE: /* Only dummy cells on the right */
191: stag->startGhost[0] = stag->start[0];
192: stag->nGhost[0] = stag->n[0] + (stag->lastRank[0] ? 1 : 0);
193: break;
194: case DMSTAG_STENCIL_STAR:
195: case DMSTAG_STENCIL_BOX:
196: stag->startGhost[0] = stag->firstRank[0] ? stag->start[0] : stag->start[0] - stag->stencilWidth;
197: stag->nGhost[0] = stag->n[0];
198: stag->nGhost[0] += stag->firstRank[0] ? 0 : stag->stencilWidth;
199: stag->nGhost[0] += stag->lastRank[0] ? 1 : stag->stencilWidth;
200: break;
201: default:
202: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unrecognized ghost stencil type %d", stag->stencilType);
203: }
204: break;
205: case DM_BOUNDARY_GHOSTED:
206: switch (stag->stencilType) {
207: case DMSTAG_STENCIL_NONE:
208: stag->startGhost[0] = stag->start[0];
209: stag->nGhost[0] = stag->n[0] + (stag->lastRank[0] ? 1 : 0);
210: break;
211: case DMSTAG_STENCIL_STAR:
212: case DMSTAG_STENCIL_BOX:
213: stag->startGhost[0] = stag->start[0] - stag->stencilWidth; /* This value may be negative */
214: stag->nGhost[0] = stag->n[0] + 2 * stag->stencilWidth + (stag->lastRank[0] && stag->stencilWidth == 0 ? 1 : 0);
215: break;
216: default:
217: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unrecognized ghost stencil type %d", stag->stencilType);
218: }
219: break;
220: case DM_BOUNDARY_PERIODIC:
221: switch (stag->stencilType) {
222: case DMSTAG_STENCIL_NONE:
223: stag->startGhost[0] = stag->start[0];
224: stag->nGhost[0] = stag->n[0];
225: break;
226: case DMSTAG_STENCIL_STAR:
227: case DMSTAG_STENCIL_BOX:
228: stag->startGhost[0] = stag->start[0] - stag->stencilWidth; /* This value may be negative */
229: stag->nGhost[0] = stag->n[0] + 2 * stag->stencilWidth;
230: break;
231: default:
232: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unrecognized ghost stencil type %d", stag->stencilType);
233: }
234: break;
235: default:
236: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unsupported x boundary type %s", DMBoundaryTypes[stag->boundaryType[0]]);
237: }
239: /* Total size of ghosted/local representation */
240: stag->entriesGhost = stag->nGhost[0] * stag->entriesPerElement;
242: /* Define neighbors */
243: PetscCall(PetscMalloc1(3, &stag->neighbors));
244: if (stag->firstRank[0]) {
245: switch (stag->boundaryType[0]) {
246: case DM_BOUNDARY_GHOSTED:
247: case DM_BOUNDARY_NONE:
248: stag->neighbors[0] = -1;
249: break;
250: case DM_BOUNDARY_PERIODIC:
251: stag->neighbors[0] = stag->nRanks[0] - 1;
252: break;
253: default:
254: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unsupported x boundary type %s", DMBoundaryTypes[stag->boundaryType[0]]);
255: }
256: } else {
257: stag->neighbors[0] = stag->rank[0] - 1;
258: }
259: stag->neighbors[1] = stag->rank[0];
260: if (stag->lastRank[0]) {
261: switch (stag->boundaryType[0]) {
262: case DM_BOUNDARY_GHOSTED:
263: case DM_BOUNDARY_NONE:
264: stag->neighbors[2] = -1;
265: break;
266: case DM_BOUNDARY_PERIODIC:
267: stag->neighbors[2] = 0;
268: break;
269: default:
270: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unsupported x boundary type %s", DMBoundaryTypes[stag->boundaryType[0]]);
271: }
272: } else {
273: stag->neighbors[2] = stag->rank[0] + 1;
274: }
276: PetscCheck(stag->n[0] >= stag->stencilWidth, PETSC_COMM_SELF, PETSC_ERR_SUP, "DMStag 1d setup does not support local sizes (%" PetscInt_FMT ") smaller than the elementwise stencil width (%" PetscInt_FMT ")", stag->n[0], stag->stencilWidth);
278: /* Create global->local VecScatter and ISLocalToGlobalMapping */
279: {
280: PetscInt *idxLocal, *idxGlobal, *idxGlobalAll;
281: PetscInt i, iLocal, d, entriesToTransferTotal, ghostOffsetStart, ghostOffsetEnd, nNonDummyGhost;
282: IS isLocal, isGlobal;
284: /* The offset on the right (may not be equal to the stencil width, as we
285: always have at least one ghost element, to account for the boundary
286: point, and may with ghosted boundaries), and the number of non-dummy ghost elements */
287: ghostOffsetStart = stag->start[0] - stag->startGhost[0];
288: ghostOffsetEnd = stag->startGhost[0] + stag->nGhost[0] - (stag->start[0] + stag->n[0]);
289: nNonDummyGhost = stag->nGhost[0] - (stag->lastRank[0] ? ghostOffsetEnd : 0) - (stag->firstRank[0] ? ghostOffsetStart : 0);
291: /* Compute the number of non-dummy entries in the local representation
292: This is equal to the number of non-dummy elements in the local (ghosted) representation,
293: plus some extra entries on the right boundary on the last rank*/
294: switch (stag->boundaryType[0]) {
295: case DM_BOUNDARY_GHOSTED:
296: case DM_BOUNDARY_NONE:
297: entriesToTransferTotal = nNonDummyGhost * stag->entriesPerElement + (stag->lastRank[0] ? stag->dof[0] : 0);
298: break;
299: case DM_BOUNDARY_PERIODIC:
300: entriesToTransferTotal = stag->entriesGhost; /* No dummy points */
301: break;
302: default:
303: SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unsupported x boundary type %s", DMBoundaryTypes[stag->boundaryType[0]]);
304: }
306: PetscCall(PetscMalloc1(entriesToTransferTotal, &idxLocal));
307: PetscCall(PetscMalloc1(entriesToTransferTotal, &idxGlobal));
308: PetscCall(PetscMalloc1(stag->entriesGhost, &idxGlobalAll));
309: if (stag->boundaryType[0] == DM_BOUNDARY_NONE) {
310: PetscInt count = 0, countAll = 0;
311: /* Left ghost points and native points */
312: for (i = stag->startGhost[0], iLocal = 0; iLocal < nNonDummyGhost; ++i, ++iLocal) {
313: for (d = 0; d < stag->entriesPerElement; ++d, ++count, ++countAll) {
314: idxLocal[count] = iLocal * stag->entriesPerElement + d;
315: idxGlobal[count] = i * stag->entriesPerElement + d;
316: idxGlobalAll[countAll] = i * stag->entriesPerElement + d;
317: }
318: }
319: /* Ghost points on the right
320: Special case for last (partial dummy) element on the last rank */
321: if (stag->lastRank[0]) {
322: i = stag->N[0];
323: iLocal = (stag->nGhost[0] - ghostOffsetEnd);
324: /* Only vertex (0-cell) dofs in global representation */
325: for (d = 0; d < stag->dof[0]; ++d, ++count, ++countAll) {
326: idxGlobal[count] = i * stag->entriesPerElement + d;
327: idxLocal[count] = iLocal * stag->entriesPerElement + d;
328: idxGlobalAll[countAll] = i * stag->entriesPerElement + d;
329: }
330: for (d = stag->dof[0]; d < stag->entriesPerElement; ++d, ++countAll) { /* Additional dummy entries */
331: idxGlobalAll[countAll] = -1;
332: }
333: }
334: } else if (stag->boundaryType[0] == DM_BOUNDARY_PERIODIC) {
335: PetscInt count = 0, iLocal = 0; /* No dummy points, so idxGlobal and idxGlobalAll are identical */
336: const PetscInt iMin = stag->firstRank[0] ? stag->start[0] : stag->startGhost[0];
337: const PetscInt iMax = stag->lastRank[0] ? stag->startGhost[0] + stag->nGhost[0] - stag->stencilWidth : stag->startGhost[0] + stag->nGhost[0];
338: /* Ghost points on the left */
339: if (stag->firstRank[0]) {
340: for (i = stag->N[0] - stag->stencilWidth; iLocal < stag->stencilWidth; ++i, ++iLocal) {
341: for (d = 0; d < stag->entriesPerElement; ++d, ++count) {
342: idxGlobal[count] = i * stag->entriesPerElement + d;
343: idxLocal[count] = iLocal * stag->entriesPerElement + d;
344: idxGlobalAll[count] = idxGlobal[count];
345: }
346: }
347: }
348: /* Native points */
349: for (i = iMin; i < iMax; ++i, ++iLocal) {
350: for (d = 0; d < stag->entriesPerElement; ++d, ++count) {
351: idxGlobal[count] = i * stag->entriesPerElement + d;
352: idxLocal[count] = iLocal * stag->entriesPerElement + d;
353: idxGlobalAll[count] = idxGlobal[count];
354: }
355: }
356: /* Ghost points on the right */
357: if (stag->lastRank[0]) {
358: for (i = 0; iLocal < stag->nGhost[0]; ++i, ++iLocal) {
359: for (d = 0; d < stag->entriesPerElement; ++d, ++count) {
360: idxGlobal[count] = i * stag->entriesPerElement + d;
361: idxLocal[count] = iLocal * stag->entriesPerElement + d;
362: idxGlobalAll[count] = idxGlobal[count];
363: }
364: }
365: }
366: } else if (stag->boundaryType[0] == DM_BOUNDARY_GHOSTED) {
367: PetscInt count = 0, countAll = 0;
368: /* Dummy elements on the left, on the first rank */
369: if (stag->firstRank[0]) {
370: for (iLocal = 0; iLocal < ghostOffsetStart; ++iLocal) {
371: /* Complete elements full of dummy entries */
372: for (d = 0; d < stag->entriesPerElement; ++d, ++countAll) idxGlobalAll[countAll] = -1;
373: }
374: i = 0; /* nonDummy entries start with global entry 0 */
375: } else {
376: /* nonDummy entries start as usual */
377: i = stag->startGhost[0];
378: iLocal = 0;
379: }
381: /* non-Dummy entries */
382: {
383: PetscInt iLocalNonDummyMax = stag->firstRank[0] ? nNonDummyGhost + ghostOffsetStart : nNonDummyGhost;
384: for (; iLocal < iLocalNonDummyMax; ++i, ++iLocal) {
385: for (d = 0; d < stag->entriesPerElement; ++d, ++count, ++countAll) {
386: idxLocal[count] = iLocal * stag->entriesPerElement + d;
387: idxGlobal[count] = i * stag->entriesPerElement + d;
388: idxGlobalAll[countAll] = i * stag->entriesPerElement + d;
389: }
390: }
391: }
393: /* (partial) dummy elements on the right, on the last rank */
394: if (stag->lastRank[0]) {
395: /* First one is partial dummy */
396: i = stag->N[0];
397: iLocal = (stag->nGhost[0] - ghostOffsetEnd);
398: for (d = 0; d < stag->dof[0]; ++d, ++count, ++countAll) { /* Only vertex (0-cell) dofs in global representation */
399: idxLocal[count] = iLocal * stag->entriesPerElement + d;
400: idxGlobal[count] = i * stag->entriesPerElement + d;
401: idxGlobalAll[countAll] = i * stag->entriesPerElement + d;
402: }
403: for (d = stag->dof[0]; d < stag->entriesPerElement; ++d, ++countAll) { /* Additional dummy entries */
404: idxGlobalAll[countAll] = -1;
405: }
406: for (iLocal = stag->nGhost[0] - ghostOffsetEnd + 1; iLocal < stag->nGhost[0]; ++iLocal) {
407: /* Additional dummy elements */
408: for (d = 0; d < stag->entriesPerElement; ++d, ++countAll) idxGlobalAll[countAll] = -1;
409: }
410: }
411: } else SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_SUP, "Unsupported x boundary type %s", DMBoundaryTypes[stag->boundaryType[0]]);
413: /* Create Local IS (transferring pointer ownership) */
414: PetscCall(ISCreateGeneral(PetscObjectComm((PetscObject)dm), entriesToTransferTotal, idxLocal, PETSC_OWN_POINTER, &isLocal));
416: /* Create Global IS (transferring pointer ownership) */
417: PetscCall(ISCreateGeneral(PetscObjectComm((PetscObject)dm), entriesToTransferTotal, idxGlobal, PETSC_OWN_POINTER, &isGlobal));
419: /* Create stag->gtol, which doesn't include dummy entries */
420: {
421: Vec local, global;
422: PetscCall(VecCreateMPIWithArray(PetscObjectComm((PetscObject)dm), 1, stag->entries, PETSC_DECIDE, NULL, &global));
423: PetscCall(VecCreateSeqWithArray(PETSC_COMM_SELF, stag->entriesPerElement, stag->entriesGhost, NULL, &local));
424: PetscCall(VecScatterCreate(global, isGlobal, local, isLocal, &stag->gtol));
425: PetscCall(VecDestroy(&global));
426: PetscCall(VecDestroy(&local));
427: }
429: /* In special cases, create a dedicated injective local-to-global map */
430: if (stag->boundaryType[0] == DM_BOUNDARY_PERIODIC && stag->nRanks[0] == 1) PetscCall(DMStagPopulateLocalToGlobalInjective(dm));
432: /* Destroy ISs */
433: PetscCall(ISDestroy(&isLocal));
434: PetscCall(ISDestroy(&isGlobal));
436: /* Create local-to-global map (transferring pointer ownership) */
437: PetscCall(ISLocalToGlobalMappingCreate(comm, 1, stag->entriesGhost, idxGlobalAll, PETSC_OWN_POINTER, &dm->ltogmap));
438: }
440: /* Precompute location offsets */
441: PetscCall(DMStagComputeLocationOffsets_1d(dm));
443: /* View from Options */
444: PetscCall(DMViewFromOptions(dm, NULL, "-dm_view"));
445: PetscFunctionReturn(PETSC_SUCCESS);
446: }
448: static PetscErrorCode DMStagComputeLocationOffsets_1d(DM dm)
449: {
450: DM_Stag *const stag = (DM_Stag *)dm->data;
451: const PetscInt epe = stag->entriesPerElement;
453: PetscFunctionBegin;
454: PetscCall(PetscMalloc1(DMSTAG_NUMBER_LOCATIONS, &stag->locationOffsets));
455: stag->locationOffsets[DMSTAG_LEFT] = 0;
456: stag->locationOffsets[DMSTAG_ELEMENT] = stag->locationOffsets[DMSTAG_LEFT] + stag->dof[0];
457: stag->locationOffsets[DMSTAG_RIGHT] = stag->locationOffsets[DMSTAG_LEFT] + epe;
458: PetscFunctionReturn(PETSC_SUCCESS);
459: }
461: PETSC_INTERN PetscErrorCode DMStagPopulateLocalToGlobalInjective_1d(DM dm)
462: {
463: DM_Stag *const stag = (DM_Stag *)dm->data;
464: PetscInt *idxLocal, *idxGlobal;
465: PetscInt i, iLocal, d, count;
466: IS isLocal, isGlobal;
468: PetscFunctionBegin;
469: PetscCall(PetscMalloc1(stag->entries, &idxLocal));
470: PetscCall(PetscMalloc1(stag->entries, &idxGlobal));
471: count = 0;
472: iLocal = stag->start[0] - stag->startGhost[0];
473: for (i = stag->start[0]; i < stag->start[0] + stag->n[0]; ++i, ++iLocal) {
474: for (d = 0; d < stag->entriesPerElement; ++d, ++count) {
475: idxGlobal[count] = i * stag->entriesPerElement + d;
476: idxLocal[count] = iLocal * stag->entriesPerElement + d;
477: }
478: }
479: if (stag->lastRank[0] && stag->boundaryType[0] != DM_BOUNDARY_PERIODIC) {
480: i = stag->start[0] + stag->n[0];
481: iLocal = stag->start[0] - stag->startGhost[0] + stag->n[0];
482: for (d = 0; d < stag->dof[0]; ++d, ++count) {
483: idxGlobal[count] = i * stag->entriesPerElement + d;
484: idxLocal[count] = iLocal * stag->entriesPerElement + d;
485: }
486: }
487: PetscCall(ISCreateGeneral(PetscObjectComm((PetscObject)dm), stag->entries, idxLocal, PETSC_OWN_POINTER, &isLocal));
488: PetscCall(ISCreateGeneral(PetscObjectComm((PetscObject)dm), stag->entries, idxGlobal, PETSC_OWN_POINTER, &isGlobal));
489: {
490: Vec local, global;
491: PetscCall(VecCreateMPIWithArray(PetscObjectComm((PetscObject)dm), 1, stag->entries, PETSC_DECIDE, NULL, &global));
492: PetscCall(VecCreateSeqWithArray(PETSC_COMM_SELF, stag->entriesPerElement, stag->entriesGhost, NULL, &local));
493: PetscCall(VecScatterCreate(local, isLocal, global, isGlobal, &stag->ltog_injective));
494: PetscCall(VecDestroy(&global));
495: PetscCall(VecDestroy(&local));
496: }
497: PetscCall(ISDestroy(&isLocal));
498: PetscCall(ISDestroy(&isGlobal));
499: PetscFunctionReturn(PETSC_SUCCESS);
500: }
502: PETSC_INTERN PetscErrorCode DMStagPopulateLocalToLocal1d_Internal(DM dm)
503: {
504: DM_Stag *const stag = (DM_Stag *)dm->data;
505: PetscInt *idxRemap;
506: PetscInt i, leftGhostEntries;
508: PetscFunctionBegin;
509: PetscCall(VecScatterCopy(stag->gtol, &stag->ltol));
510: PetscCall(PetscMalloc1(stag->entries, &idxRemap));
512: leftGhostEntries = (stag->start[0] - stag->startGhost[0]) * stag->entriesPerElement;
513: for (i = 0; i < stag->entries; ++i) idxRemap[i] = leftGhostEntries + i;
515: PetscCall(VecScatterRemap(stag->ltol, idxRemap, NULL));
516: PetscCall(PetscFree(idxRemap));
517: PetscFunctionReturn(PETSC_SUCCESS);
518: }
520: PETSC_INTERN PetscErrorCode DMCreateMatrix_Stag_1D_AIJ_Assemble(DM dm, Mat A)
521: {
522: DMStagStencilType stencil_type;
523: PetscInt dof[2], start, n, n_extra, stencil_width, N, epe;
524: DMBoundaryType boundary_type_x;
526: PetscFunctionBegin;
527: PetscCall(DMStagGetDOF(dm, &dof[0], &dof[1], NULL, NULL));
528: PetscCall(DMStagGetStencilType(dm, &stencil_type));
529: PetscCall(DMStagGetStencilWidth(dm, &stencil_width));
530: PetscCall(DMStagGetCorners(dm, &start, NULL, NULL, &n, NULL, NULL, &n_extra, NULL, NULL));
531: PetscCall(DMStagGetGlobalSizes(dm, &N, NULL, NULL));
532: PetscCall(DMStagGetEntriesPerElement(dm, &epe));
533: PetscCall(DMStagGetBoundaryTypes(dm, &boundary_type_x, NULL, NULL));
534: if (stencil_type == DMSTAG_STENCIL_NONE) {
535: /* Couple all DOF at each location to each other */
536: DMStagStencil *row_vertex, *row_element;
538: PetscCall(PetscMalloc1(dof[0], &row_vertex));
539: for (PetscInt c = 0; c < dof[0]; ++c) {
540: row_vertex[c].loc = DMSTAG_LEFT;
541: row_vertex[c].c = c;
542: }
544: PetscCall(PetscMalloc1(dof[1], &row_element));
545: for (PetscInt c = 0; c < dof[1]; ++c) {
546: row_element[c].loc = DMSTAG_ELEMENT;
547: row_element[c].c = c;
548: }
550: for (PetscInt e = start; e < start + n + n_extra; ++e) {
551: {
552: for (PetscInt c = 0; c < dof[0]; ++c) row_vertex[c].i = e;
553: PetscCall(DMStagMatSetValuesStencil(dm, A, dof[0], row_vertex, dof[0], row_vertex, NULL, INSERT_VALUES));
554: }
555: if (e < N) {
556: for (PetscInt c = 0; c < dof[1]; ++c) row_element[c].i = e;
557: PetscCall(DMStagMatSetValuesStencil(dm, A, dof[1], row_element, dof[1], row_element, NULL, INSERT_VALUES));
558: }
559: }
560: PetscCall(PetscFree(row_vertex));
561: PetscCall(PetscFree(row_element));
562: } else if (stencil_type == DMSTAG_STENCIL_STAR || stencil_type == DMSTAG_STENCIL_BOX) {
563: DMStagStencil *col, *row;
565: PetscCall(PetscMalloc1(epe, &row));
566: {
567: PetscInt nrows = 0;
568: for (PetscInt c = 0; c < dof[0]; ++c) {
569: row[nrows].c = c;
570: row[nrows].loc = DMSTAG_LEFT;
571: ++nrows;
572: }
573: for (PetscInt c = 0; c < dof[1]; ++c) {
574: row[nrows].c = c;
575: row[nrows].loc = DMSTAG_ELEMENT;
576: ++nrows;
577: }
578: }
579: PetscCall(PetscMalloc1(epe, &col));
580: {
581: PetscInt ncols = 0;
582: for (PetscInt c = 0; c < dof[0]; ++c) {
583: col[ncols].c = c;
584: col[ncols].loc = DMSTAG_LEFT;
585: ++ncols;
586: }
587: for (PetscInt c = 0; c < dof[1]; ++c) {
588: col[ncols].c = c;
589: col[ncols].loc = DMSTAG_ELEMENT;
590: ++ncols;
591: }
592: }
593: for (PetscInt e = start; e < start + n + n_extra; ++e) {
594: for (PetscInt i = 0; i < epe; ++i) row[i].i = e;
595: for (PetscInt offset = -stencil_width; offset <= stencil_width; ++offset) {
596: const PetscInt e_offset = e + offset;
598: /* Only set values corresponding to elements which can have non-dummy entries,
599: meaning those that map to unknowns in the global representation. In the periodic
600: case, this is the entire stencil, but in all other cases, only includes a single
601: "extra" element which is partially outside the physical domain (those points in the
602: global representation */
603: if (boundary_type_x == DM_BOUNDARY_PERIODIC || (e_offset < N + 1 && e_offset >= 0)) {
604: for (PetscInt i = 0; i < epe; ++i) col[i].i = e_offset;
605: PetscCall(DMStagMatSetValuesStencil(dm, A, epe, row, epe, col, NULL, INSERT_VALUES));
606: }
607: }
608: }
609: PetscCall(PetscFree(row));
610: PetscCall(PetscFree(col));
611: } else SETERRQ(PetscObjectComm((PetscObject)dm), PETSC_ERR_ARG_OUTOFRANGE, "Unsupported stencil type %s", DMStagStencilTypes[stencil_type]);
612: PetscCall(MatAssemblyBegin(A, MAT_FINAL_ASSEMBLY));
613: PetscCall(MatAssemblyEnd(A, MAT_FINAL_ASSEMBLY));
614: PetscFunctionReturn(PETSC_SUCCESS);
615: }