Computes the mean-squared displacement (MSD) of a particle cloud.
Reference point r0 = (simCtx->psrc_x, psrc_y, psrc_z). Diffusivity D = 1 / (Re * Sc), time t = ti * dt. Computes MSD_x, MSD_y, MSD_z (isotropy check), MSD_total, r_rms, centre-of-mass drift, and fractions inside 1σ/2σ/3σ theoretical shells. Appends one row to {stats_prefix}_msd.csv; writes header on first call.
Computes the mean-squared displacement (MSD) of a particle cloud.
Local to this translation unit.
130{
131 PetscErrorCode ierr;
133 PetscMPIInt rank;
134
135 PetscFunctionBeginUser;
137
138 ierr = MPI_Comm_rank(PETSC_COMM_WORLD, &rank); CHKERRQ(ierr);
139
140
141
142
143 const PetscReal Re = (simCtx->
ren > 0.0) ? simCtx->
ren : 1.0;
145 const PetscReal D = 1.0 / (Re * Sc);
146 const PetscReal t = (PetscReal)ti * simCtx->
dt;
147 const PetscReal x0 = simCtx->psrc_x;
148 const PetscReal y0 = simCtx->psrc_y;
149 const PetscReal z0 = simCtx->psrc_z;
150
151
152
153
154 PetscInt n_local;
155 const PetscReal (*pos_arr)[3];
156
157 ierr = DMSwarmGetLocalSize(user->
swarm, &n_local); CHKERRQ(ierr);
158 ierr = DMSwarmGetField(user->
swarm,
"position", NULL, NULL, (
void**)&pos_arr); CHKERRQ(ierr);
159
160 PetscReal local_sq_x = 0.0, local_sq_y = 0.0, local_sq_z = 0.0;
161 PetscReal local_sx = 0.0, local_sy = 0.0, local_sz = 0.0;
162
163 for (PetscInt p = 0; p < n_local; p++) {
164 const PetscReal dx = pos_arr[p][0] - x0;
165 const PetscReal dy = pos_arr[p][1] - y0;
166 const PetscReal dz = pos_arr[p][2] - z0;
167 local_sq_x += dx * dx;
168 local_sq_y += dy * dy;
169 local_sq_z += dz * dz;
170 local_sx += dx;
171 local_sy += dy;
172 local_sz += dz;
173 }
174
175 ierr = DMSwarmRestoreField(user->
swarm,
"position", NULL, NULL, (
void**)&pos_arr); CHKERRQ(ierr);
176
177
178
179
180 PetscReal local_buf[7] = { local_sq_x, local_sq_y, local_sq_z,
181 local_sx, local_sy, local_sz,
182 (PetscReal)n_local };
183 PetscReal global_buf[7] = {0};
184 ierr = MPI_Allreduce(local_buf, global_buf, 7, MPI_DOUBLE, MPI_SUM, PETSC_COMM_WORLD); CHKERRMPI(ierr);
185
186 const PetscReal N_total = global_buf[6];
187 if (N_total < 1.0) PetscFunctionReturn(0);
188
189 const PetscReal MSD_x = global_buf[0] / N_total;
190 const PetscReal MSD_y = global_buf[1] / N_total;
191 const PetscReal MSD_z = global_buf[2] / N_total;
192 const PetscReal com_x = global_buf[3] / N_total;
193 const PetscReal com_y = global_buf[4] / N_total;
194 const PetscReal com_z = global_buf[5] / N_total;
195
196 const PetscReal MSD_total = MSD_x + MSD_y + MSD_z;
197 const PetscReal r_rms_meas = PetscSqrtReal(MSD_total);
198 const PetscReal r_theory = (t > 1e-300) ? PetscSqrtReal(6.0 * D * t) : 0.0;
199 const PetscReal rel_err_pct = (r_theory > 1e-12)
200 ? PetscAbsReal(r_rms_meas - r_theory) / r_theory * 100.0
201 : 0.0;
202
203
204
205
206 ierr = DMSwarmGetField(user->
swarm,
"position", NULL, NULL, (
void**)&pos_arr); CHKERRQ(ierr);
207
208 PetscInt local_n1 = 0, local_n2 = 0, local_n3 = 0;
209 const PetscReal r1 = r_theory;
210 const PetscReal r2 = 2.0 * r_theory;
211 const PetscReal r3 = 3.0 * r_theory;
212
213 for (PetscInt p = 0; p < n_local; p++) {
214 const PetscReal dx = pos_arr[p][0] - x0;
215 const PetscReal dy = pos_arr[p][1] - y0;
216 const PetscReal dz = pos_arr[p][2] - z0;
217 const PetscReal r = PetscSqrtReal(dx*dx + dy*dy + dz*dz);
218 if (r < r1) local_n1++;
219 if (r < r2) local_n2++;
220 if (r < r3) local_n3++;
221 }
222
223 ierr = DMSwarmRestoreField(user->
swarm,
"position", NULL, NULL, (
void**)&pos_arr); CHKERRQ(ierr);
224
225 PetscReal local_counts[3] = { (PetscReal)local_n1,
226 (PetscReal)local_n2,
227 (PetscReal)local_n3 };
228 PetscReal global_counts[3] = {0};
229 ierr = MPI_Allreduce(local_counts, global_counts, 3, MPI_DOUBLE, MPI_SUM, PETSC_COMM_WORLD); CHKERRMPI(ierr);
230
231 const PetscReal frac_1s = global_counts[0] / N_total * 100.0;
232 const PetscReal frac_2s = global_counts[1] / N_total * 100.0;
233 const PetscReal frac_3s = global_counts[2] / N_total * 100.0;
234
235
236
237
238 if (rank == 0) {
239 char csv_path[PETSC_MAX_PATH_LEN];
240 char row_line[1024];
241 PetscSNPrintf(csv_path, sizeof(csv_path), "%s_msd.csv", stats_prefix);
242 PetscSNPrintf(
243 row_line,
244 sizeof(row_line),
245 "%d,%.6e,%.0f,%.6e,%.6e,%.6e,%.6e,%.6e,%.6e,%.4f,%.6e,%.6e,%.6e,%.2f,%.2f,%.2f\n",
246 (int)ti, t, N_total,
247 MSD_x, MSD_y, MSD_z, MSD_total,
248 r_rms_meas, r_theory, rel_err_pct,
249 com_x, com_y, com_z,
250 frac_1s, frac_2s, frac_3s
251 );
253
255 "[MSD ti=%d t=%.4f] total=%.4e theory=%.4e err=%.2f%% | "
256 "fracs: %.1f/%.1f/%.1f%% | COM: (%.2e,%.2e,%.2e)\n",
257 (int)ti, t, MSD_total, r_theory * r_theory, rel_err_pct,
258 frac_1s, frac_2s, frac_3s,
259 com_x, com_y, com_z);
260 }
261
263 PetscFunctionReturn(0);
264}
#define GLOBAL
Scope for global logging across all processes.
#define LOG_ALLOW(scope, level, fmt,...)
Logging macro that checks both the log level and whether the calling function is in the allowed-funct...
#define PROFILE_FUNCTION_END
Marks the end of a profiled code block.
@ LOG_INFO
Informational messages about program execution.
#define PROFILE_FUNCTION_BEGIN
Marks the beginning of a profiled code block (typically a function).
static PetscErrorCode RewriteParticleMSDCSV(const char *csv_path, PetscInt ti, const char *row_line)
Rewrite the MSD CSV so the requested timestep appears exactly once.
SimCtx * simCtx
Back-pointer to the master simulation context.
The master context for the entire simulation.