24 Vec target_vec = NULL;
25 PetscReal scale_factor = 1.0;
26 char field_type[64] =
"Unknown";
27 PetscBool is_swarm_field = PETSC_FALSE;
28 const char *swarm_field_name = NULL;
30 PetscFunctionBeginUser;
32 if (!user) SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_NULL,
"UserCtx is NULL.");
33 if (!field_name) SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_NULL,
"field_name is NULL.");
36 if (strcasecmp(field_name,
"Ucat") == 0) {
37 target_vec = user->
Ucat;
39 strcpy(field_type,
"Cartesian Velocity (L/T)");
40 }
else if (strcasecmp(field_name,
"Ucont") == 0) {
41 target_vec = user->
Ucont;
43 strcpy(field_type,
"Contravariant Volume Flux (L^3/T)");
44 }
else if (strcasecmp(field_name,
"P") == 0) {
47 strcpy(field_type,
"Pressure (M L^-1 T^-2)");
48 }
else if (strcasecmp(field_name,
"Coordinates") == 0) {
49 ierr = DMGetCoordinates(user->
da, &target_vec); CHKERRQ(ierr);
51 strcpy(field_type,
"Grid Coordinates (L)");
52 }
else if (strcasecmp(field_name,
"ParticlePosition") == 0) {
53 is_swarm_field = PETSC_TRUE;
54 swarm_field_name =
"position";
56 strcpy(field_type,
"Particle Position (L)");
57 }
else if (strcasecmp(field_name,
"ParticleVelocity") == 0) {
58 is_swarm_field = PETSC_TRUE;
59 swarm_field_name =
"velocity";
61 strcpy(field_type,
"Particle Velocity (L/T)");
63 LOG(
GLOBAL,
LOG_WARNING,
"DimensionalizeField: Unknown or unhandled field_name '%s'. Field will not be scaled.\n", field_name);
65 PetscFunctionReturn(0);
69 if (PetscAbsReal(scale_factor - 1.0) < PETSC_MACHINE_EPSILON) {
70 LOG(
GLOBAL,
LOG_DEBUG,
"DimensionalizeField: Scaling factor for '%s' is 1.0. Skipping operation.\n", field_name);
72 PetscFunctionReturn(0);
76 LOG(
GLOBAL,
LOG_INFO,
"Scaling '%s' field (%s) by factor %.4e.\n", field_name, field_type, scale_factor);
80 ierr = DMSwarmCreateGlobalVectorFromField(user->
swarm, swarm_field_name, &target_vec); CHKERRQ(ierr);
81 ierr = VecScale(target_vec, scale_factor); CHKERRQ(ierr);
82 ierr = DMSwarmDestroyGlobalVectorFromField(user->
swarm, swarm_field_name, &target_vec); CHKERRQ(ierr);
86 ierr = VecScale(target_vec, scale_factor); CHKERRQ(ierr);
88 SETERRQ(PETSC_COMM_WORLD, PETSC_ERR_ARG_WRONGSTATE,
"Target vector for field '%s' was not found or is NULL.", field_name);
93 if (strcasecmp(field_name,
"Coordinates") == 0) {
95 ierr = DMGetCoordinates(user->
da, &target_vec); CHKERRQ(ierr);
96 ierr = DMGetCoordinatesLocal(user->
da, &l_coords); CHKERRQ(ierr);
97 ierr = DMGlobalToLocalBegin(user->
fda, target_vec, INSERT_VALUES, l_coords); CHKERRQ(ierr);
98 ierr = DMGlobalToLocalEnd(user->
fda, target_vec, INSERT_VALUES, l_coords); CHKERRQ(ierr);
102 PetscFunctionReturn(0);
169 Vec in_vec_local = NULL, out_vec_global = NULL;
170 DM dm_in = NULL, dm_out = NULL;
173 PetscFunctionBeginUser;
175 LOG_ALLOW(
GLOBAL,
LOG_INFO,
"-> KERNEL: Running ComputeNodalAverage on '%s' -> '%s'.\n", in_field_name, out_field_name);
178 if (strcasecmp(in_field_name,
"P") == 0) { in_vec_local = user->
lP; dm_in = user->
da; dof = 1; }
179 else if (strcasecmp(in_field_name,
"Ucat") == 0) { in_vec_local = user->
lUcat; dm_in = user->
fda; dof = 3; }
180 else if (strcasecmp(in_field_name,
"Psi") == 0) { in_vec_local = user->
lPsi; dm_in = user->
da; dof = 1; }
182 else SETERRQ(PETSC_COMM_WORLD, PETSC_ERR_ARG_WRONG,
"Unknown input field name for nodal averaging: %s", in_field_name);
184 if (strcasecmp(out_field_name,
"P_nodal") == 0) { out_vec_global = user->
P_nodal; dm_out = user->
da; }
185 else if (strcasecmp(out_field_name,
"Ucat_nodal") == 0) { out_vec_global = user->
Ucat_nodal; dm_out = user->
fda; }
186 else if (strcasecmp(out_field_name,
"Psi_nodal") == 0) { out_vec_global = user->
Psi_nodal; dm_out = user->
da; }
188 else SETERRQ(PETSC_COMM_WORLD, PETSC_ERR_ARG_WRONG,
"Unknown output field name for nodal averaging: %s", out_field_name);
195 ierr = DMDAGetLocalInfo(dm_out, &info); CHKERRQ(ierr);
198 const PetscReal ***l_in_arr;
199 PetscReal ***g_out_arr;
200 ierr = DMDAVecGetArrayRead(dm_in,in_vec_local, (
void*)&l_in_arr); CHKERRQ(ierr);
201 ierr = DMDAVecGetArray(dm_out,out_vec_global, (
void*)&g_out_arr); CHKERRQ(ierr);
205 for (PetscInt k = info.zs; k < info.zs + info.zm - 1; k++) {
206 for (PetscInt j = info.ys; j < info.ys + info.ym - 1; j++) {
207 for (PetscInt i = info.xs; i < info.xs + info.xm - 1; i++) {
208 g_out_arr[k][j][i] = 0.125 * (l_in_arr[k][j][i] + l_in_arr[k][j][i+1] +
209 l_in_arr[k][j+1][i] + l_in_arr[k][j+1][i+1] +
210 l_in_arr[k+1][j][i] + l_in_arr[k+1][j][i+1] +
211 l_in_arr[k+1][j+1][i] + l_in_arr[k+1][j+1][i+1]);
215 ierr = DMDAVecRestoreArrayRead(dm_in,in_vec_local, (
void*)&l_in_arr); CHKERRQ(ierr);
216 ierr = DMDAVecRestoreArray(dm_out,out_vec_global, (
void*)&g_out_arr); CHKERRQ(ierr);
218 }
else if (dof == 3) {
221 ierr = DMDAVecGetArrayRead(dm_in,in_vec_local, (
void*)&l_in_arr); CHKERRQ(ierr);
222 ierr = DMDAVecGetArray(dm_out,out_vec_global, (
void*)&g_out_arr); CHKERRQ(ierr);
224 for (PetscInt k = info.zs; k < info.zs + info.zm - 1; k++) {
225 for (PetscInt j = info.ys; j < info.ys + info.ym - 1; j++) {
226 for (PetscInt i = info.xs; i < info.xs + info.xm - 1; i++) {
227 g_out_arr[k][j][i].
x = 0.125 * (l_in_arr[k][j][i].
x + l_in_arr[k][j][i+1].
x +
228 l_in_arr[k][j+1][i].
x + l_in_arr[k][j+1][i+1].
x +
229 l_in_arr[k+1][j][i].
x + l_in_arr[k+1][j][i+1].
x +
230 l_in_arr[k+1][j+1][i].
x + l_in_arr[k+1][j+1][i+1].
x);
232 g_out_arr[k][j][i].
y = 0.125 * (l_in_arr[k][j][i].
y + l_in_arr[k][j][i+1].
y +
233 l_in_arr[k][j+1][i].
y + l_in_arr[k][j+1][i+1].
y +
234 l_in_arr[k+1][j][i].
y + l_in_arr[k+1][j][i+1].
y +
235 l_in_arr[k+1][j+1][i].
y + l_in_arr[k+1][j+1][i+1].
y);
237 g_out_arr[k][j][i].
z = 0.125 * (l_in_arr[k][j][i].
z + l_in_arr[k][j][i+1].
z +
238 l_in_arr[k][j+1][i].
z + l_in_arr[k][j+1][i+1].
z +
239 l_in_arr[k+1][j][i].
z + l_in_arr[k+1][j][i+1].
z +
240 l_in_arr[k+1][j+1][i].
z + l_in_arr[k+1][j+1][i+1].
z);
244 ierr = DMDAVecRestoreArrayRead(dm_in,in_vec_local, (
void*)&l_in_arr); CHKERRQ(ierr);
245 ierr = DMDAVecRestoreArray(dm_out,out_vec_global, (
void*)&g_out_arr); CHKERRQ(ierr);
248 PetscFunctionReturn(0);
268 const Cmpnts ***lucat, ***lcsi, ***leta, ***lzet;
269 const PetscReal***laj, ***lnvert;
272 PetscFunctionBeginUser;
285 ierr = DMDAGetLocalInfo(user->
da, &info); CHKERRQ(ierr);
287 ierr = DMDAVecGetArrayRead(user->
fda, user->
lUcat, (
void*)&lucat); CHKERRQ(ierr);
288 ierr = DMDAVecGetArrayRead(user->
fda, user->
lCsi, (
void*)&lcsi); CHKERRQ(ierr);
289 ierr = DMDAVecGetArrayRead(user->
fda, user->
lEta, (
void*)&leta); CHKERRQ(ierr);
290 ierr = DMDAVecGetArrayRead(user->
fda, user->
lZet, (
void*)&lzet); CHKERRQ(ierr);
291 ierr = DMDAVecGetArrayRead(user->
da, user->
lAj, (
void*)&laj); CHKERRQ(ierr);
292 ierr = DMDAVecGetArrayRead(user->
da, user->
lNvert, (
void*)&lnvert); CHKERRQ(ierr);
293 ierr = DMDAVecGetArray(user->
da, user->
Qcrit, (
void*)&gq); CHKERRQ(ierr);
296 PetscInt i_start = (info.xs == 0) ? 1 : info.xs;
297 PetscInt i_end = (info.xs + info.xm == info.mx) ? info.mx - 1 : info.xs + info.xm;
298 PetscInt j_start = (info.ys == 0) ? 1 : info.ys;
299 PetscInt j_end = (info.ys + info.ym == info.my) ? info.my - 1 : info.ys + info.ym;
300 PetscInt k_start = (info.zs == 0) ? 1 : info.zs;
301 PetscInt k_end = (info.zs + info.zm == info.mz) ? info.mz - 1 : info.zs + info.zm;
304 for (PetscInt k = k_start; k < k_end; k++) {
305 for (PetscInt j = j_start; j < j_end; j++) {
306 for (PetscInt i = i_start; i < i_end; i++) {
309 PetscReal uc = 0.5 * (lucat[k][j][i+1].
x - lucat[k][j][i-1].
x);
310 PetscReal vc = 0.5 * (lucat[k][j][i+1].
y - lucat[k][j][i-1].
y);
311 PetscReal wc = 0.5 * (lucat[k][j][i+1].
z - lucat[k][j][i-1].
z);
313 PetscReal ue = 0.5 * (lucat[k][j+1][i].
x - lucat[k][j-1][i].
x);
314 PetscReal ve = 0.5 * (lucat[k][j+1][i].
y - lucat[k][j-1][i].
y);
315 PetscReal we = 0.5 * (lucat[k][j+1][i].
z - lucat[k][j-1][i].
z);
317 PetscReal uz = 0.5 * (lucat[k+1][j][i].
x - lucat[k-1][j][i].
x);
318 PetscReal vz = 0.5 * (lucat[k+1][j][i].
y - lucat[k-1][j][i].
y);
319 PetscReal wz = 0.5 * (lucat[k+1][j][i].
z - lucat[k-1][j][i].
z);
322 PetscReal csi1 = 0.5 * (lcsi[k][j][i].
x + lcsi[k][j][i-1].
x) * laj[k][j][i];
323 PetscReal csi2 = 0.5 * (lcsi[k][j][i].
y + lcsi[k][j][i-1].
y) * laj[k][j][i];
324 PetscReal csi3 = 0.5 * (lcsi[k][j][i].
z + lcsi[k][j][i-1].
z) * laj[k][j][i];
326 PetscReal eta1 = 0.5 * (leta[k][j][i].
x + leta[k][j-1][i].
x) * laj[k][j][i];
327 PetscReal eta2 = 0.5 * (leta[k][j][i].
y + leta[k][j-1][i].
y) * laj[k][j][i];
328 PetscReal eta3 = 0.5 * (leta[k][j][i].
z + leta[k][j-1][i].
z) * laj[k][j][i];
330 PetscReal zet1 = 0.5 * (lzet[k][j][i].
x + lzet[k-1][j][i].
x) * laj[k][j][i];
331 PetscReal zet2 = 0.5 * (lzet[k][j][i].
y + lzet[k-1][j][i].
y) * laj[k][j][i];
332 PetscReal zet3 = 0.5 * (lzet[k][j][i].
z + lzet[k-1][j][i].
z) * laj[k][j][i];
335 PetscReal d11 = uc * csi1 + ue * eta1 + uz * zet1;
336 PetscReal d12 = uc * csi2 + ue * eta2 + uz * zet2;
337 PetscReal d13 = uc * csi3 + ue * eta3 + uz * zet3;
339 PetscReal d21 = vc * csi1 + ve * eta1 + vz * zet1;
340 PetscReal d22 = vc * csi2 + ve * eta2 + vz * zet2;
341 PetscReal d23 = vc * csi3 + ve * eta3 + vz * zet3;
343 PetscReal d31 = wc * csi1 + we * eta1 + wz * zet1;
344 PetscReal d32 = wc * csi2 + we * eta2 + wz * zet2;
345 PetscReal d33 = wc * csi3 + we * eta3 + wz * zet3;
349 PetscReal s12 = 0.5 * (d12 + d21);
350 PetscReal s13 = 0.5 * (d13 + d31);
352 PetscReal s23 = 0.5 * (d23 + d32);
356 PetscReal w12 = 0.5 * (d12 - d21);
357 PetscReal w13 = 0.5 * (d13 - d31);
358 PetscReal w23 = 0.5 * (d23 - d32);
361 PetscReal s_norm_sq = s11*s11 + s22*s22 + s33*s33 + 2.0*(s12*s12 + s13*s13 + s23*s23);
362 PetscReal w_norm_sq = 2.0 * (w12*w12 + w13*w13 + w23*w23);
364 gq[k][j][i] = 0.5 * (w_norm_sq - s_norm_sq);
366 if (lnvert[k][j][i] > 0.1) {
374 ierr = DMDAVecRestoreArrayRead(user->
fda, user->
lUcat, (
void*)&lucat); CHKERRQ(ierr);
375 ierr = DMDAVecRestoreArrayRead(user->
fda, user->
lCsi, (
void*)&lcsi); CHKERRQ(ierr);
376 ierr = DMDAVecRestoreArrayRead(user->
fda, user->
lEta, (
void*)&leta); CHKERRQ(ierr);
377 ierr = DMDAVecRestoreArrayRead(user->
fda, user->
lZet, (
void*)&lzet); CHKERRQ(ierr);
378 ierr = DMDAVecRestoreArrayRead(user->
da, user->
lAj, (
void*)&laj); CHKERRQ(ierr);
379 ierr = DMDAVecRestoreArrayRead(user->
da, user->
lNvert, (
void*)&lnvert); CHKERRQ(ierr);
380 ierr = DMDAVecRestoreArray(user->
da, user->
Qcrit, (
void*)&gq); CHKERRQ(ierr);
383 PetscFunctionReturn(0);
405 PetscInt ip=1, jp=1, kp=1;
406 PetscReal p_ref = 0.0;
407 PetscInt ref_point_global_idx[1];
408 PetscScalar ref_value_local[1];
410 VecScatter scatter_ctx;
419 PetscFunctionBeginUser;
424 if (strcasecmp(relative_field_name,
"P") == 0) {
427 SETERRQ(PETSC_COMM_SELF, 1,
"NormalizeRelativeField only supports the primary 'P' field , not '%s' currently.", relative_field_name);
431 ierr = MPI_Comm_rank(PETSC_COMM_WORLD, &rank); CHKERRQ(ierr);
434 ref_point_global_idx[0] = kp * (user->
IM * user->
JM) + jp * user->
IM + ip;
438 ierr = ISCreateGeneral(PETSC_COMM_WORLD, 1, ref_point_global_idx, PETSC_COPY_VALUES, &is_from); CHKERRQ(ierr);
441 ierr = VecCreateSeq(PETSC_COMM_SELF, 1, &ref_vec_seq); CHKERRQ(ierr);
444 PetscInt dest_idx[1] = {0};
445 ierr = ISCreateGeneral(PETSC_COMM_SELF, 1, dest_idx, PETSC_COPY_VALUES, &is_to); CHKERRQ(ierr);
448 ierr = VecScatterCreate(P_vec, is_from, ref_vec_seq, is_to, &scatter_ctx); CHKERRQ(ierr);
449 ierr = VecScatterBegin(scatter_ctx, P_vec, ref_vec_seq, INSERT_VALUES, SCATTER_FORWARD); CHKERRQ(ierr);
450 ierr = VecScatterEnd(scatter_ctx, P_vec, ref_vec_seq, INSERT_VALUES, SCATTER_FORWARD); CHKERRQ(ierr);
454 ierr = VecGetValues(ref_vec_seq, 1, dest_idx, ref_value_local); CHKERRQ(ierr);
455 p_ref = ref_value_local[0];
456 LOG_ALLOW(
LOCAL,
LOG_DEBUG,
"%s reference point (%" PetscInt_FMT
", %" PetscInt_FMT
", %" PetscInt_FMT
") has value %g.\n", relative_field_name, jp, kp, ip, p_ref);
460 ierr = MPI_Bcast(&p_ref, 1, MPIU_REAL, 0, PETSC_COMM_WORLD); CHKERRQ(ierr);
463 ierr = VecShift(P_vec, -p_ref); CHKERRQ(ierr);
467 ierr = ISDestroy(&is_from); CHKERRQ(ierr);
468 ierr = ISDestroy(&is_to); CHKERRQ(ierr);
469 ierr = VecScatterDestroy(&scatter_ctx); CHKERRQ(ierr);
470 ierr = VecDestroy(&ref_vec_seq); CHKERRQ(ierr);
473 PetscFunctionReturn(0);
497 const PetscScalar (*vel_arr)[3];
498 PetscScalar *ske_arr;
500 PetscFunctionBeginUser;
502 LOG_ALLOW(
GLOBAL,
LOG_INFO,
"-> KERNEL: Running ComputeSpecificKE ('%s' -> '%s').\n", velocity_field, ske_field);
505 ierr = DMSwarmGetLocalSize(user->
swarm, &n_local); CHKERRQ(ierr);
506 if (n_local == 0) PetscFunctionReturn(0);
509 ierr = DMSwarmGetField(user->
swarm, velocity_field, NULL, NULL, (
const void**)&vel_arr); CHKERRQ(ierr);
510 ierr = DMSwarmGetField(user->
post_swarm, ske_field, NULL, NULL, (
void**)&ske_arr); CHKERRQ(ierr);
513 for (PetscInt p = 0; p < n_local; p++) {
514 const PetscScalar u = vel_arr[p][0];
515 const PetscScalar v = vel_arr[p][1];
516 const PetscScalar w = vel_arr[p][2];
517 const PetscScalar vel_sq = u*u + v*v + w*w;
518 ske_arr[p] = 0.5 * vel_sq;
522 ierr = DMSwarmRestoreField(user->
swarm, velocity_field, NULL, NULL, (
const void**)&vel_arr); CHKERRQ(ierr);
523 ierr = DMSwarmRestoreField(user->
post_swarm, ske_field, NULL, NULL, (
void**)&ske_arr); CHKERRQ(ierr);
526 PetscFunctionReturn(0);
PetscErrorCode ComputeQCriterion(UserCtx *user)
Computes the Q-Criterion, a scalar value identifying vortex cores.
PetscErrorCode ComputeSpecificKE(UserCtx *user, const char *velocity_field, const char *ske_field)
Computes the specific kinetic energy (KE per unit mass) for each particle.
PetscErrorCode NormalizeRelativeField(UserCtx *user, const char *relative_field_name)
Normalizes a relative field by subtracting a reference value.
PetscErrorCode DimensionalizeField(UserCtx *user, const char *field_name)
Scales a specified field from non-dimensional to dimensional units in-place.
PetscErrorCode DimensionalizeAllLoadedFields(UserCtx *user)
Orchestrates the dimensionalization of all relevant fields loaded from a file.
PetscErrorCode ComputeNodalAverage(UserCtx *user, const char *in_field_name, const char *out_field_name)
Computes node-centered data by averaging 8 surrounding cell-centered values, exactly replicating the ...
Holds all configuration parameters for a post-processing run.