PICurv 0.1.0
A Parallel Particle-In-Cell Solver for Curvilinear LES
Loading...
Searching...
No Matches
Functions
runloop.h File Reference
#include <petscpf.h>
#include <petscdmswarm.h>
#include <stdlib.h>
#include <time.h>
#include <math.h>
#include <petsctime.h>
#include <petscsys.h>
#include <petscdmcomposite.h>
#include <petscsystypes.h>
#include "variables.h"
#include "ParticleSwarm.h"
#include "walkingsearch.h"
#include "grid.h"
#include "logging.h"
#include "io.h"
#include "interpolation.h"
#include "initialcondition.h"
#include "AnalyticalSolutions.h"
#include "ParticleMotion.h"
#include "ParticlePhysics.h"
#include "Boundaries.h"
#include "setup.h"
#include "solvers.h"
Include dependency graph for runloop.h:
This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Functions

PetscErrorCode InitializeRuntimeSignalHandlers (void)
 Installs lightweight signal handlers for graceful shutdown requests.
 
PetscReal RuntimeWalltimeGuardUpdateEWMA (PetscBool has_previous, PetscReal previous_ewma_seconds, PetscReal latest_step_seconds, PetscReal alpha)
 Update an EWMA estimate for timestep wall-clock duration.
 
PetscReal RuntimeWalltimeGuardConservativeEstimate (PetscReal warmup_average_seconds, PetscReal ewma_seconds, PetscReal latest_step_seconds)
 Return the conservative timestep estimate used by the walltime guard.
 
PetscReal RuntimeWalltimeGuardRequiredHeadroom (PetscReal min_seconds, PetscReal multiplier, PetscReal conservative_estimate_seconds)
 Compute the required shutdown headroom from timestep estimate and floor.
 
PetscBool RuntimeWalltimeGuardShouldTrigger (PetscInt completed_steps, PetscInt warmup_steps, PetscReal remaining_seconds, PetscReal min_seconds, PetscReal multiplier, PetscReal warmup_average_seconds, PetscReal ewma_seconds, PetscReal latest_step_seconds, PetscReal *required_headroom_seconds_out)
 Decide whether the runtime walltime guard should stop before another step.
 
PetscErrorCode UpdateSolverHistoryVectors (UserCtx *user)
 Copies the current time step's solution fields into history vectors (e.g., U(t_n) -> U_o, U_o -> U_rm1) for the next time step's calculations.
 
PetscErrorCode AdvanceSimulation (SimCtx *simCtx)
 Executes the main time-marching loop for the particle simulation.
 
PetscErrorCode PerformInitializedParticleSetup (SimCtx *simCtx)
 Finalizes the simulation setup at t=0, ensuring a consistent state before time marching.
 
PetscErrorCode PerformLoadedParticleSetup (SimCtx *simCtx)
 Finalizes the simulation state after particle and fluid data have been loaded from a restart.
 
PetscErrorCode FinalizeRestartState (SimCtx *simCtx)
 Performs post-load/post-init consistency checks for a restarted simulation.
 

Function Documentation

◆ InitializeRuntimeSignalHandlers()

PetscErrorCode InitializeRuntimeSignalHandlers ( void  )

Installs lightweight signal handlers for graceful shutdown requests.

The handlers only record that a shutdown signal was received. The actual output flush and exit path happens later at safe checkpoints in the run loop.

Returns
PetscErrorCode 0 on success.

Installs lightweight signal handlers for graceful shutdown requests.

Full API contract (arguments, ownership, side effects) is documented with the matching public header declaration.

See also
InitializeRuntimeSignalHandlers()

Definition at line 123 of file runloop.c.

124{
125 PetscErrorCode ierr;
126
127 PetscFunctionBeginUser;
130
131#ifdef SIGTERM
132 ierr = RegisterRuntimeSignalHandler(SIGTERM); CHKERRQ(ierr);
133#endif
134#ifdef SIGUSR1
135 ierr = RegisterRuntimeSignalHandler(SIGUSR1); CHKERRQ(ierr);
136#endif
137#ifdef SIGINT
138 ierr = RegisterRuntimeSignalHandler(SIGINT); CHKERRQ(ierr);
139#endif
140
141 PetscFunctionReturn(0);
142}
static PetscBool g_runtime_shutdown_auto_requested
Definition runloop.c:13
static PetscErrorCode RegisterRuntimeSignalHandler(int signum)
Internal helper implementation: RegisterRuntimeSignalHandler().
Definition runloop.c:98
static volatile sig_atomic_t g_runtime_shutdown_signal
Definition runloop.c:12
Here is the call graph for this function:
Here is the caller graph for this function:

◆ RuntimeWalltimeGuardUpdateEWMA()

PetscReal RuntimeWalltimeGuardUpdateEWMA ( PetscBool  has_previous,
PetscReal  previous_ewma_seconds,
PetscReal  latest_step_seconds,
PetscReal  alpha 
)

Update an EWMA estimate for timestep wall-clock duration.

Parameters
[in]has_previousWhether a previous EWMA estimate exists.
[in]previous_ewma_secondsPrior EWMA estimate in seconds.
[in]latest_step_secondsLatest completed timestep duration in seconds.
[in]alphaEWMA weighting factor in (0, 1].
Returns
PetscReal Updated EWMA estimate in seconds.

Update an EWMA estimate for timestep wall-clock duration.

Full API contract (arguments, ownership, side effects) is documented with the matching public header declaration.

See also
RuntimeWalltimeGuardUpdateEWMA()

Definition at line 150 of file runloop.c.

151{
152 if (!has_previous) return latest_step_seconds;
153 return alpha * latest_step_seconds + (1.0 - alpha) * previous_ewma_seconds;
154}
Here is the caller graph for this function:

◆ RuntimeWalltimeGuardConservativeEstimate()

PetscReal RuntimeWalltimeGuardConservativeEstimate ( PetscReal  warmup_average_seconds,
PetscReal  ewma_seconds,
PetscReal  latest_step_seconds 
)

Return the conservative timestep estimate used by the walltime guard.

Parameters
[in]warmup_average_secondsAverage duration across warmup steps.
[in]ewma_secondsCurrent EWMA duration estimate.
[in]latest_step_secondsMost recent completed timestep duration.
Returns
PetscReal Conservative timestep estimate in seconds.

Return the conservative timestep estimate used by the walltime guard.

Full API contract (arguments, ownership, side effects) is documented with the matching public header declaration.

See also
RuntimeWalltimeGuardConservativeEstimate()

Definition at line 162 of file runloop.c.

163{
164 return PetscMax(warmup_average_seconds, PetscMax(ewma_seconds, latest_step_seconds));
165}
Here is the caller graph for this function:

◆ RuntimeWalltimeGuardRequiredHeadroom()

PetscReal RuntimeWalltimeGuardRequiredHeadroom ( PetscReal  min_seconds,
PetscReal  multiplier,
PetscReal  conservative_estimate_seconds 
)

Compute the required shutdown headroom from timestep estimate and floor.

Parameters
[in]min_secondsAbsolute minimum shutdown headroom.
[in]multiplierSafety multiplier applied to the timestep estimate.
[in]conservative_estimate_secondsConservative timestep estimate in seconds.
Returns
PetscReal Required headroom in seconds.

Compute the required shutdown headroom from timestep estimate and floor.

Full API contract (arguments, ownership, side effects) is documented with the matching public header declaration.

See also
RuntimeWalltimeGuardRequiredHeadroom()

Definition at line 173 of file runloop.c.

174{
175 return PetscMax(min_seconds, multiplier * conservative_estimate_seconds);
176}
Here is the caller graph for this function:

◆ RuntimeWalltimeGuardShouldTrigger()

PetscBool RuntimeWalltimeGuardShouldTrigger ( PetscInt  completed_steps,
PetscInt  warmup_steps,
PetscReal  remaining_seconds,
PetscReal  min_seconds,
PetscReal  multiplier,
PetscReal  warmup_average_seconds,
PetscReal  ewma_seconds,
PetscReal  latest_step_seconds,
PetscReal *  required_headroom_seconds_out 
)

Decide whether the runtime walltime guard should stop before another step.

Parameters
[in]completed_stepsNumber of completed timesteps observed so far.
[in]warmup_stepsMinimum completed timesteps required before guarding.
[in]remaining_secondsRemaining walltime in seconds.
[in]min_secondsAbsolute minimum shutdown headroom.
[in]multiplierSafety multiplier applied to timestep estimate.
[in]warmup_average_secondsAverage duration across warmup steps.
[in]ewma_secondsCurrent EWMA duration estimate.
[in]latest_step_secondsLatest completed timestep duration.
[out]required_headroom_seconds_outComputed required headroom in seconds.
Returns
PetscBool PETSC_TRUE when shutdown should be requested.

Decide whether the runtime walltime guard should stop before another step.

Full API contract (arguments, ownership, side effects) is documented with the matching public header declaration.

See also
RuntimeWalltimeGuardShouldTrigger()

Definition at line 184 of file runloop.c.

185{
186 PetscReal conservative_estimate = 0.0;
187 PetscReal required_headroom = 0.0;
188
189 if (required_headroom_seconds_out) *required_headroom_seconds_out = 0.0;
190 if (completed_steps < warmup_steps) return PETSC_FALSE;
191
192 conservative_estimate = RuntimeWalltimeGuardConservativeEstimate(warmup_average_seconds, ewma_seconds, latest_step_seconds);
193 required_headroom = RuntimeWalltimeGuardRequiredHeadroom(min_seconds, multiplier, conservative_estimate);
194 if (required_headroom_seconds_out) *required_headroom_seconds_out = required_headroom;
195 return (PetscBool)(remaining_seconds <= required_headroom);
196}
PetscReal RuntimeWalltimeGuardConservativeEstimate(PetscReal warmup_average_seconds, PetscReal ewma_seconds, PetscReal latest_step_seconds)
Implementation of RuntimeWalltimeGuardConservativeEstimate().
Definition runloop.c:162
PetscReal RuntimeWalltimeGuardRequiredHeadroom(PetscReal min_seconds, PetscReal multiplier, PetscReal conservative_estimate_seconds)
Implementation of RuntimeWalltimeGuardRequiredHeadroom().
Definition runloop.c:173
Here is the call graph for this function:
Here is the caller graph for this function:

◆ UpdateSolverHistoryVectors()

PetscErrorCode UpdateSolverHistoryVectors ( UserCtx user)

Copies the current time step's solution fields into history vectors (e.g., U(t_n) -> U_o, U_o -> U_rm1) for the next time step's calculations.

This function is critical for multi-step time integration schemes (like BDF2) used by the legacy solver. It must be called at the end of every time step, after the new solution has been fully computed.

The order of operations is important to avoid overwriting data prematurely.

Parameters
userThe UserCtx for a single block. The function modifies the history vectors (Ucont_o, Ucont_rm1, etc.) within this context.
Returns
PetscErrorCode 0 on success.

Copies the current time step's solution fields into history vectors (e.g., U(t_n) -> U_o, U_o -> U_rm1) for the next time step's calculations.

Local to this translation unit.

Definition at line 321 of file runloop.c.

322{
323 PetscErrorCode ierr;
324 SimCtx *simCtx = user->simCtx; // Access global settings if needed
325
326 PetscFunctionBeginUser;
327 LOG_ALLOW(LOCAL, LOG_DEBUG, "Rank %d, Block %d: Updating solver history vectors.\n",
328 simCtx->rank, user->_this);
329
330 // --- Primary Contravariant Velocity History ---
331 // The order is critical here.
332 // 1. First, move the n-1 state (Ucont_o) to the n-2 slot (Ucont_rm1).
333 ierr = VecCopy(user->Ucont_o, user->Ucont_rm1); CHKERRQ(ierr);
334 // 2. Then, move the new n state (Ucont) to the n-1 slot (Ucont_o).
335 ierr = VecCopy(user->Ucont, user->Ucont_o); CHKERRQ(ierr);
336
337 LOG_ALLOW(LOCAL,LOG_DEBUG, "Rank %d, Block %d, Ucont history updated.\n",simCtx->rank,user->_this);
338
339 // --- Update History for Other Fields ---
340 // These are typically only needed at the n-1 state.
341 ierr = VecCopy(user->Ucat, user->Ucat_o); CHKERRQ(ierr);
342 ierr = VecCopy(user->P, user->P_o); CHKERRQ(ierr);
343 LOG_ALLOW(LOCAL,LOG_DEBUG, "Rank %d, Block %d, Ucat & P history updated.\n",simCtx->rank,user->_this);
344
345 if (simCtx->immersed) {
346 ierr = VecCopy(user->Nvert, user->Nvert_o); CHKERRQ(ierr);
347 }
348
349 // --- Update History for Turbulence Models (if active) ---
350 if (simCtx->rans) {
351 ierr = VecCopy(user->K_Omega, user->K_Omega_o); CHKERRQ(ierr);
352 }
353
354 // --- Synchronize Local Ghost Regions for the new history vectors ---
355 // This is essential so that stencils in the next time step's calculations
356 // have correct values from neighboring processes.
357 ierr = DMGlobalToLocalBegin(user->fda, user->Ucont_o, INSERT_VALUES, user->lUcont_o); CHKERRQ(ierr);
358 ierr = DMGlobalToLocalEnd(user->fda, user->Ucont_o, INSERT_VALUES, user->lUcont_o); CHKERRQ(ierr);
359
360 ierr = DMGlobalToLocalBegin(user->fda, user->Ucont_rm1, INSERT_VALUES, user->lUcont_rm1); CHKERRQ(ierr);
361 ierr = DMGlobalToLocalEnd(user->fda, user->Ucont_rm1, INSERT_VALUES, user->lUcont_rm1); CHKERRQ(ierr);
362
363 if (simCtx->immersed) {
364 ierr = DMGlobalToLocalBegin(user->da, user->Nvert_o, INSERT_VALUES, user->lNvert_o); CHKERRQ(ierr);
365 ierr = DMGlobalToLocalEnd(user->da, user->Nvert_o, INSERT_VALUES, user->lNvert_o); CHKERRQ(ierr);
366 }
367
368 if (simCtx->rans) {
369 ierr = DMGlobalToLocalBegin(user->fda2, user->K_Omega_o, INSERT_VALUES, user->lK_Omega_o); CHKERRQ(ierr);
370 ierr = DMGlobalToLocalEnd(user->fda2, user->K_Omega_o, INSERT_VALUES, user->lK_Omega_o); CHKERRQ(ierr);
371 }
372
373 PetscFunctionReturn(0);
374}
#define LOCAL
Logging scope definitions for controlling message output.
Definition logging.h:44
#define LOG_ALLOW(scope, level, fmt,...)
Logging macro that checks both the log level and whether the calling function is in the allowed-funct...
Definition logging.h:199
@ LOG_DEBUG
Detailed debugging information.
Definition logging.h:31
PetscMPIInt rank
Definition variables.h:654
SimCtx * simCtx
Back-pointer to the master simulation context.
Definition variables.h:833
PetscInt rans
Definition variables.h:746
Vec K_Omega_o
Definition variables.h:886
Vec K_Omega
Definition variables.h:886
Vec lUcont_rm1
Definition variables.h:864
PetscInt _this
Definition variables.h:843
Vec Ucont
Definition variables.h:856
Vec lUcont_o
Definition variables.h:863
Vec Ucat_o
Definition variables.h:863
Vec lK_Omega_o
Definition variables.h:886
Vec Ucat
Definition variables.h:856
Vec Ucont_o
Definition variables.h:863
Vec Nvert_o
Definition variables.h:863
Vec Ucont_rm1
Definition variables.h:864
Vec Nvert
Definition variables.h:856
Vec lNvert_o
Definition variables.h:863
PetscInt immersed
Definition variables.h:681
Vec P_o
Definition variables.h:863
The master context for the entire simulation.
Definition variables.h:651
Here is the caller graph for this function:

◆ AdvanceSimulation()

PetscErrorCode AdvanceSimulation ( SimCtx simCtx)

Executes the main time-marching loop for the particle simulation.

This version uses the new, integrated LocateAllParticlesInGrid orchestrator and the ResetAllParticleStatuses helper for a clean, robust, and understandable workflow.

For each timestep, it performs:

  1. Sets the background fluid velocity field (Ucat) for the current step.
  2. Updates particle positions using velocity from the previous step's interpolation.
  3. Removes any particles that have left the global domain.
  4. A single call to ‘LocateAllParticlesInGrid’, which handles all particle location and migration until the swarm is fully settled.
  5. Interpolates the current fluid velocity to the newly settled particle locations.
  6. Scatters particle data back to Eulerian fields.
  7. Outputs data at specified intervals.
Parameters
simCtxPointer to the master simulation context.
Returns
PetscErrorCode 0 on success, non-zero on failure.

Executes the main time-marching loop for the particle simulation.

Local to this translation unit.

Definition at line 574 of file runloop.c.

575{
576 PetscErrorCode ierr;
577 PetscReal step_start_seconds = 0.0;
578 PetscReal step_elapsed_local = 0.0;
579 PetscReal step_elapsed_max = 0.0;
580 // Get the master context from the first block. All blocks share it.
581 UserCtx *user = simCtx->usermg.mgctx[simCtx->usermg.mglevels-1].user;
582
583 // Retrieve control parameters from SimCtx for clarity
584 const PetscInt StartStep = simCtx->StartStep;
585 const PetscInt StepsToRun = simCtx->StepsToRun;
586 const PetscReal dt = simCtx->dt;
587
588 // Variables for particle removal statistics
589 //PetscInt removed_local_ob, removed_global_ob;
590 PetscInt removed_local_lost, removed_global_lost;
591 PetscBool terminated_early = PETSC_FALSE;
592 PetscInt last_completed_loop_index = StartStep - 1;
593
594 PetscFunctionBeginUser;
596 LOG_ALLOW(GLOBAL, LOG_INFO, "Starting main time-marching loop: %d steps from step %d (t=%.4f), dt=%.4f\n",
597 StepsToRun, StartStep, simCtx->StartTime, dt);
598
599 // --- Main Time-Marching Loop ---
600 for (PetscInt step = StartStep; step < StartStep + StepsToRun; step++) {
601 ierr = MaybeRequestRuntimeWalltimeGuardShutdown(simCtx, "pre-step checkpoint"); CHKERRQ(ierr);
603 ierr = WriteForcedTerminationOutput(simCtx, user, "pre-step checkpoint"); CHKERRQ(ierr);
604 terminated_early = PETSC_TRUE;
605 break;
606 }
607
608 step_start_seconds = MPI_Wtime();
609
610 // =================================================================
611 // 1. PRE-STEP SETUP
612 // =================================================================
613
614 // Update simulation time and step counters in the master context
615 simCtx->step = step + 1;
616 simCtx->ti += simCtx->dt; //simCtx->StartTime + step * simCtx->dt;
617
618 LOG_ALLOW(GLOBAL, LOG_INFO, "--- Advancing Step %d (To t=%.4f) ---\n", simCtx->step, simCtx->ti);
619
620
621 // For particles, reset their status to prepare for the new advection/location cycle
622 if (simCtx->np > 0) {
623 LOG_ALLOW(GLOBAL, LOG_DEBUG, "Resetting all particle statuses to NEEDS_LOCATION.\n");
624 ierr = ResetAllParticleStatuses(user); CHKERRQ(ierr);
625 }
626
627 // =================================================================
628 // 2. EULERIAN SOLVER STEP
629 // =================================================================
631 ierr = LOG_FIELD_ANATOMY(&user[0],"Coordinates","PreFlowSolver"); CHKERRQ(ierr);
632 ierr = LOG_FIELD_ANATOMY(&user[0],"Csi","PreFlowSolver"); CHKERRQ(ierr);
633 ierr = LOG_FIELD_ANATOMY(&user[0],"Eta","PreFlowSolver"); CHKERRQ(ierr);
634 ierr = LOG_FIELD_ANATOMY(&user[0],"Zet","PreFlowSolver"); CHKERRQ(ierr);
635 ierr = LOG_FIELD_ANATOMY(&user[0],"Center-Coordinates","PreFlowSolver"); CHKERRQ(ierr);
636 ierr = LOG_FIELD_ANATOMY(&user[0],"X-Face-Centers","PreFlowSolver"); CHKERRQ(ierr);
637 ierr = LOG_FIELD_ANATOMY(&user[0],"Y-Face-Centers","PreFlowSolver"); CHKERRQ(ierr);
638 ierr = LOG_FIELD_ANATOMY(&user[0],"Z-Face-Centers","PreFlowSolver"); CHKERRQ(ierr);
639 ierr = LOG_FIELD_ANATOMY(&user[0],"Ucat","PreFlowSolver"); CHKERRQ(ierr);
640 }
641 LOG_ALLOW(GLOBAL, LOG_INFO, "Updating Eulerian Field ...\n");
642 if(strcmp(simCtx->eulerianSource,"load")==0){
643 //LOAD mode: Read pre-computed fields for the current step.
644 LOG_ALLOW(GLOBAL,LOG_INFO,"Eulerian Source 'load': Reading fields (t=%.4f,step=%d)...\n",simCtx->ti,simCtx->step);
645 for(PetscInt bi = 0; bi < simCtx->block_number;bi++){
646 ierr = ReadSimulationFields(&user[bi],simCtx->step); CHKERRQ(ierr);
647 }
648 }else if(strcmp(simCtx->eulerianSource,"analytical")==0){
649 // ANALYTICAL mode:Call the Analytical Solution Prescription Engine to enable a variety of analytical functions
650 LOG_ALLOW(GLOBAL,LOG_INFO,"Eulerian Source 'analytical'. Updating Eulerian field via the Analytical Solution Engine ...\n");
651 ierr = AnalyticalSolutionEngine(simCtx); CHKERRQ(ierr);
652 }else if(strcmp(simCtx->eulerianSource,"solve")==0){
653 // SOLVE mode:Call the refactored, high-level legacy solver. This single function
654 // advances the entire multi-block fluid field from t_n to t_{n+1}.
655 LOG_ALLOW(GLOBAL,LOG_INFO,"Eulerian Source 'solve'. Updating Eulerian field via Solver...\n");
656 ierr = FlowSolver(simCtx); CHKERRQ(ierr);
657 }
658 LOG_ALLOW(GLOBAL, LOG_INFO, "Eulerian Field Updated ...\n");
660 LOG_ALLOW(GLOBAL, LOG_VERBOSE, "Post FlowSolver field states:\n");
661 ierr = LOG_FIELD_ANATOMY(&user[0],"Ucat","PostFlowSolver"); CHKERRQ(ierr);
662 ierr = LOG_FIELD_ANATOMY(&user[0],"P","PostFlowSolver"); CHKERRQ(ierr);
663 ierr = LOG_FIELD_ANATOMY(&user[0],"Ucont","PostFlowSolver"); CHKERRQ(ierr);
664 }
665
666
667 // =================================================================
668 // 3. LAGRANGIAN PARTICLE STEP
669 // =================================================================
670
671 if (simCtx->np > 0) {
672 LOG_ALLOW(GLOBAL, LOG_INFO, "Updating Lagrangian particle system...\n");
673 simCtx->particlesLostLastStep = 0;
674
675 // a. Update Eulerian Transport Properties:
676 // Optimization: Only recalculate if turbulence is active (Nu_t changes).
677 // For Laminar flow, the value calculated at Setup is constant.
678 if (simCtx->les || simCtx->rans) {
679 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
680 ierr = ComputeEulerianDiffusivity(&user[bi]); CHKERRQ(ierr);
681 ierr = ComputeEulerianDiffusivityGradient(&user[bi]); CHKERRQ(ierr);
682 }
683 }
684
685 // a.1 (Optional) Log Eulerian Diffusivity min/max and anatomy for debugging.
687 LOG_ALLOW(GLOBAL, LOG_VERBOSE, "Updated Diffusivity Min/Max:\n");
688 ierr = LOG_FIELD_MIN_MAX(&user[0],"Diffusivity"); CHKERRQ(ierr);
689 ierr = LOG_FIELD_MIN_MAX(&user[0],"DiffusivityGradient"); CHKERRQ(ierr);
690 //LOG_ALLOW(GLOBAL, LOG_VERBOSE, "Updated Diffusivity Anatomy:\n");
691 ierr = LOG_FIELD_ANATOMY(&user[0],"Diffusivity","PostDiffusivityUpdate"); CHKERRQ(ierr);
692 }
693 // b. Advect particles using the velocity interpolated from the *previous* step.
694 // P(t_{n+1}) = P(t_n) + V_p(t_n) * dt
695 ierr = UpdateAllParticlePositions(user); CHKERRQ(ierr);
696
697 // c. Settle all particles: find their new host cells and migrate them across ranks.
698 ierr = LocateAllParticlesInGrid(user, simCtx->bboxlist); CHKERRQ(ierr);
699
700 // d. Remove any particles that are now lost or out of the global domain.
701 ierr = CheckAndRemoveLostParticles(user, &removed_local_lost, &removed_global_lost); CHKERRQ(ierr);
702 //ierr = CheckAndRemoveOutOfBoundsParticles(user, &removed_local_ob, &removed_global_ob, simCtx->bboxlist); CHKERRQ(ierr);
703 simCtx->particlesLostLastStep = removed_global_lost;
704 simCtx->particlesLostCumulative += removed_global_lost;
705 if (removed_global_lost> 0) { // if(removed_global_lost + removed_global_ob > 0){
706 LOG_ALLOW(GLOBAL, LOG_INFO, "Removed %d particles globally this step.\n", removed_global_lost); // removed_global_lost + removed_global_ob;
707 }
708
709 // e. Interpolate the NEW fluid velocity (just computed by FlowSolver) onto the
710 // particles' new positions. This gives them V_p(t_{n+1}) for the *next* advection step.
711 ierr = InterpolateAllFieldsToSwarm(user); CHKERRQ(ierr);
712
713 // f. Update the Particle Fields (e.g., temperature, concentration) if applicable.
714 // This can be extended to include reactions, growth, etc.
715 ierr = UpdateAllParticleFields(user); CHKERRQ(ierr);
716
717 // g. (For Two-Way Coupling) Scatter particle data back to the grid to act as a source term.
718 ierr = CalculateParticleCountPerCell(user); CHKERRQ(ierr);
719 ierr = ScatterAllParticleFieldsToEulerFields(user); CHKERRQ(ierr);
720
721 // h. (Optional) Calculate advanced particle metrics for logging/debugging.
722 ierr = CalculateAdvancedParticleMetrics(user); CHKERRQ(ierr);
723 ierr = LOG_SEARCH_METRICS(user); CHKERRQ(ierr);
724
725 ierr = LOG_PARTICLE_METRICS(user, "Timestep Metrics"); CHKERRQ(ierr);
726
727
729 LOG_ALLOW(GLOBAL, LOG_VERBOSE, "Post Lagrangian update field states:\n");
730 ierr = LOG_FIELD_MIN_MAX(&user[0],"Psi"); CHKERRQ(ierr);
731 }
732 }
733
734 // =================================================================
735 // 4. UPDATE HISTORY & I/O
736 // =================================================================
737
738 PetscCall(LOG_SOLUTION_CONVERGENCE(simCtx));
739
740 // Copy the newly computed fields (Ucont, P, etc.) to the history vectors
741 // (_o, _rm1) to prepare for the next time step.
742 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
743 ierr = UpdateSolverHistoryVectors(&user[bi]); CHKERRQ(ierr);
744 }
745
746 //ierr = LOG_UCAT_ANATOMY(&user[0],"Final"); CHKERRQ(ierr);
747
748 // Handle periodic file output
749 if (ShouldWriteDataOutput(simCtx, simCtx->step)) {
750 LOG_ALLOW(GLOBAL, LOG_INFO, "Writing output for step %d.\n",simCtx->step);
751 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
752 ierr = WriteSimulationFields(&user[bi]); CHKERRQ(ierr);
753 }
754 if (simCtx->np > 0) {
755 ierr = WriteAllSwarmFields(user); CHKERRQ(ierr);
756 if (strcmp(simCtx->eulerianSource, "analytical") == 0 &&
759 }
760 ierr = LOG_SCATTER_METRICS(user); CHKERRQ(ierr);
761 }
762 }
763
764 if (ShouldEmitPeriodicParticleConsoleSnapshot(simCtx, simCtx->step)) {
765 ierr = EmitParticleConsoleSnapshot(user, simCtx, simCtx->step); CHKERRQ(ierr);
766 }
767
768 ProfilingLogTimestepSummary(simCtx, simCtx->step);
769 ierr = RuntimeMemoryLogSample(simCtx, simCtx->step, "Step", "-"); CHKERRQ(ierr);
770
771 // Update Progress Bar
772 if(simCtx->rank == 0) {
773 PrintProgressBar(step,StartStep,StepsToRun,simCtx->ti);
774 if(get_log_level()>=LOG_WARNING) PetscPrintf(PETSC_COMM_SELF,"\n");
775 }
776
777 last_completed_loop_index = step;
778
779 step_elapsed_local = MPI_Wtime() - step_start_seconds;
780 step_elapsed_max = step_elapsed_local;
781 ierr = MPI_Allreduce(&step_elapsed_local, &step_elapsed_max, 1, MPIU_REAL, MPI_MAX, PETSC_COMM_WORLD); CHKERRMPI(ierr);
782 if (simCtx->walltimeGuardActive) {
783 UpdateRuntimeWalltimeGuardEstimator(simCtx, step_elapsed_max);
784 ierr = MaybeRequestRuntimeWalltimeGuardShutdown(simCtx, "post-step checkpoint"); CHKERRQ(ierr);
785 }
786
788 ierr = WriteForcedTerminationOutput(simCtx, user, "post-step checkpoint"); CHKERRQ(ierr);
789 terminated_early = PETSC_TRUE;
790 break;
791 }
792 } // --- End of Time-Marching Loop ---
793
794 // After the loop, print the final progress state on rank 0 and add a newline
795 // to ensure subsequent terminal output starts on a fresh line.
796 if (simCtx->rank == 0 && StepsToRun > 0) {
797 if (!terminated_early && last_completed_loop_index >= StartStep) {
798 PrintProgressBar(StartStep + StepsToRun - 1, StartStep, StepsToRun, simCtx->ti);
799 } else if (terminated_early && last_completed_loop_index >= StartStep) {
800 PrintProgressBar(last_completed_loop_index, StartStep, StepsToRun, simCtx->ti);
801 }
802 PetscPrintf(PETSC_COMM_SELF, "\n");
803 fflush(stdout);
804 }
805
806 if (terminated_early) {
808 "Time marching stopped early after %s. Final retained state is step %d at t=%.4f.\n",
809 RuntimeShutdownReasonName(), simCtx->step, simCtx->ti);
810 } else {
811 LOG_ALLOW(GLOBAL, LOG_INFO, "Time marching completed. Final time t=%.4f.\n", simCtx->ti);
812 }
814 PetscFunctionReturn(0);
815}
PetscErrorCode AnalyticalSolutionEngine(SimCtx *simCtx)
Dispatches to the appropriate analytical solution function based on simulation settings.
PetscBool AnalyticalTypeSupportsInterpolationError(const char *analytical_type)
Reports whether an analytical type has a non-trivial velocity field for which interpolation error mea...
PetscErrorCode UpdateAllParticlePositions(UserCtx *user)
Loops over all local particles in the DMSwarm, updating their positions based on velocity and the glo...
PetscErrorCode CalculateParticleCountPerCell(UserCtx *user)
Counts particles in each cell of the DMDA 'da' and stores the result in user->ParticleCount.
PetscErrorCode LocateAllParticlesInGrid(UserCtx *user, BoundingBox *bboxlist)
Orchestrates the complete particle location and migration process for one timestep.
PetscErrorCode ResetAllParticleStatuses(UserCtx *user)
Marks all local particles as NEEDS_LOCATION for the next settlement pass.
PetscErrorCode CheckAndRemoveLostParticles(UserCtx *user, PetscInt *removedCountLocal, PetscInt *removedCountGlobal)
Removes particles that have been definitively flagged as LOST by the location algorithm.
PetscErrorCode UpdateAllParticleFields(UserCtx *user)
Orchestrates the update of all physical properties for particles.
PetscErrorCode ScatterAllParticleFieldsToEulerFields(UserCtx *user)
Scatters a predefined set of particle fields to their corresponding Eulerian fields.
PetscErrorCode InterpolateAllFieldsToSwarm(UserCtx *user)
Interpolates all relevant fields from the DMDA to the DMSwarm.
PetscErrorCode WriteAllSwarmFields(UserCtx *user)
Writes a predefined set of PETSc Swarm fields to files.
Definition io.c:1818
PetscErrorCode ReadSimulationFields(UserCtx *user, PetscInt ti)
Reads binary field data for velocity, pressure, and other required vectors.
Definition io.c:1129
PetscBool ShouldWriteDataOutput(const SimCtx *simCtx, PetscInt completed_step)
Returns whether full field/restart output should be written for the.
Definition io.c:70
PetscErrorCode WriteSimulationFields(UserCtx *user)
Writes simulation fields to files.
Definition io.c:1571
PetscErrorCode LOG_PARTICLE_METRICS(UserCtx *user, const char *stageName)
Logs particle swarm metrics, adapting its behavior based on a boolean flag in SimCtx.
Definition logging.c:3182
PetscBool is_function_allowed(const char *functionName)
Checks if a given function is in the allow-list.
Definition logging.c:183
PetscErrorCode LOG_INTERPOLATION_ERROR(UserCtx *user)
Logs the interpolation error between the analytical and computed solutions.
Definition logging.c:2719
#define GLOBAL
Scope for global logging across all processes.
Definition logging.h:45
PetscBool ShouldEmitPeriodicParticleConsoleSnapshot(const SimCtx *simCtx, PetscInt completed_step)
Returns whether a particle console snapshot should be emitted for the.
Definition logging.c:542
PetscErrorCode EmitParticleConsoleSnapshot(UserCtx *user, SimCtx *simCtx, PetscInt step)
Emits one particle console snapshot into the main solver log.
Definition logging.c:556
#define PROFILE_FUNCTION_END
Marks the end of a profiled code block.
Definition logging.h:820
PetscErrorCode LOG_FIELD_MIN_MAX(UserCtx *user, const char *fieldName)
Computes and logs the local and global min/max values of a 3-component vector field.
Definition logging.c:2301
void PrintProgressBar(PetscInt step, PetscInt startStep, PetscInt totalSteps, PetscReal currentTime)
Prints a progress bar to the console.
Definition logging.c:2254
PetscErrorCode LOG_FIELD_ANATOMY(UserCtx *user, const char *field_name, const char *stage_name)
Logs the anatomy of a specified field at key boundary locations, respecting the solver's specific gri...
Definition logging.c:2449
PetscErrorCode RuntimeMemoryLogSample(SimCtx *simCtx, PetscInt step, const char *event, const char *reason)
Append a reduced runtime memory sample to the configured memory log.
Definition logging.c:2037
LogLevel get_log_level()
Retrieves the current logging level from the environment variable LOG_LEVEL.
Definition logging.c:84
PetscErrorCode ProfilingLogTimestepSummary(SimCtx *simCtx, PetscInt step)
Logs the performance summary for the current timestep and resets timers.
Definition logging.c:1959
PetscErrorCode CalculateAdvancedParticleMetrics(UserCtx *user)
Computes advanced particle statistics and stores them in SimCtx.
Definition logging.c:3128
PetscErrorCode LOG_SCATTER_METRICS(UserCtx *user)
Logs particle-to-grid scatter verification metrics for the prescribed scalar truth path.
Definition logging.c:2795
PetscErrorCode LOG_SOLUTION_CONVERGENCE(SimCtx *simCtx)
Logs physical solution-convergence metrics once per completed timestep.
Definition logging.c:1560
PetscErrorCode LOG_SEARCH_METRICS(UserCtx *user)
Writes compact runtime search metrics to CSV and optionally to console.
Definition logging.c:2977
@ LOG_INFO
Informational messages about program execution.
Definition logging.h:30
@ LOG_WARNING
Non-critical issues that warrant attention.
Definition logging.h:29
@ LOG_VERBOSE
Extremely detailed logs, typically for development use only.
Definition logging.h:33
#define PROFILE_FUNCTION_BEGIN
Marks the beginning of a profiled code block (typically a function).
Definition logging.h:811
PetscErrorCode ComputeEulerianDiffusivity(UserCtx *user)
Computes the effective diffusivity scalar field (Gamma_eff) on the Eulerian grid.
Definition rhs.c:1959
PetscErrorCode ComputeEulerianDiffusivityGradient(UserCtx *user)
Computes the Eulerian gradient of the effective diffusivity field.
Definition rhs.c:2089
static PetscErrorCode MaybeRequestRuntimeWalltimeGuardShutdown(SimCtx *simCtx, const char *checkpoint_name)
Internal helper implementation: MaybeRequestRuntimeWalltimeGuardShutdown().
Definition runloop.c:242
PetscErrorCode UpdateSolverHistoryVectors(UserCtx *user)
Internal helper implementation: UpdateSolverHistoryVectors().
Definition runloop.c:321
static PetscErrorCode WriteForcedTerminationOutput(SimCtx *simCtx, UserCtx *user, const char *phase)
Internal helper implementation: WriteForcedTerminationOutput().
Definition runloop.c:287
static const char * RuntimeShutdownReasonName(void)
Internal helper implementation: RuntimeShutdownReasonName().
Definition runloop.c:83
static PetscBool RuntimeShutdownRequested(void)
Internal helper implementation: RuntimeShutdownRequested().
Definition runloop.c:41
#define __FUNCT__
Definition runloop.c:377
static void UpdateRuntimeWalltimeGuardEstimator(SimCtx *simCtx, PetscReal completed_step_seconds)
Internal helper implementation: UpdateRuntimeWalltimeGuardEstimator().
Definition runloop.c:214
PetscErrorCode FlowSolver(SimCtx *simCtx)
Orchestrates a single time step of the Eulerian fluid solver.
Definition solvers.c:11
UserCtx * user
Definition variables.h:536
PetscInt block_number
Definition variables.h:726
PetscBool walltimeGuardActive
Definition variables.h:796
PetscReal StartTime
Definition variables.h:665
PetscInt particlesLostLastStep
Definition variables.h:760
UserMG usermg
Definition variables.h:778
PetscReal dt
Definition variables.h:666
PetscInt StepsToRun
Definition variables.h:662
PetscInt np
Definition variables.h:753
PetscInt StartStep
Definition variables.h:661
BoundingBox * bboxlist
Definition variables.h:756
char eulerianSource[PETSC_MAX_PATH_LEN]
Definition variables.h:671
PetscInt mglevels
Definition variables.h:543
PetscInt particlesLostCumulative
Definition variables.h:761
char AnalyticalSolutionType[PETSC_MAX_PATH_LEN]
Definition variables.h:684
PetscInt step
Definition variables.h:659
PetscInt les
Definition variables.h:746
MGCtx * mgctx
Definition variables.h:546
PetscReal ti
Definition variables.h:660
User-defined context containing data specific to a single computational grid level.
Definition variables.h:830
Here is the call graph for this function:
Here is the caller graph for this function:

◆ PerformInitializedParticleSetup()

PetscErrorCode PerformInitializedParticleSetup ( SimCtx simCtx)

Finalizes the simulation setup at t=0, ensuring a consistent state before time marching.

This function is called from main() after the initial Eulerian and Lagrangian states have been created but before the main time loop begins. Its responsibilities are:

  1. Settling the particle swarm: Migrates particles to their correct owner ranks and finds their initial host cells. This includes handling special surface initializations.
  2. Coupling the fields: Interpolates the initial Eulerian fields to the settled particle locations.
  3. Preparing for the first step: Scatters particle data back to the grid.
  4. Writing the initial output for step 0.
Parameters
simCtxPointer to the main simulation context structure.
Returns
PetscErrorCode 0 on success, non-zero on failure.

Finalizes the simulation setup at t=0, ensuring a consistent state before time marching.

Local to this translation unit.

Definition at line 382 of file runloop.c.

383{
384 PetscErrorCode ierr;
385 // --- Get pointers from SimCtx instead of passing them as arguments ---
386 UserCtx *user = simCtx->usermg.mgctx[simCtx->usermg.mglevels-1].user;
387 BoundingBox *bboxlist = simCtx->bboxlist;
388
389 PetscFunctionBeginUser;
390
391 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Performing initial particle setup procedures.\n", simCtx->ti, simCtx->step);
392
393 // --- 0. Loop over all blocks to compute Eulerian diffusivity.
394 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
395 ierr = ComputeEulerianDiffusivity(&user[bi]); CHKERRQ(ierr);
396 ierr = ComputeEulerianDiffusivityGradient(&user[bi]); CHKERRQ(ierr);
397 }
398
399 // --- 1. Initial Particle Settlement (Location and Migration) ---
400 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Initial Settlement: Locating and migrating all particles...\n", simCtx->ti, simCtx->step);
401 ierr = LocateAllParticlesInGrid(user, bboxlist); CHKERRQ(ierr);
402
404 LOG_ALLOW(GLOBAL, LOG_DEBUG, "[T=%.4f, Step=%d] Particle field states after Initial settlement...\n", simCtx->ti, simCtx->step);
405 ierr = LOG_PARTICLE_FIELDS(user,simCtx->LoggingFrequency); CHKERRQ(ierr);
406 }
407
408 // --- 2. Re-initialize Particles on Inlet Surface (if applicable) ---
411 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Re-initializing particles on inlet surface...\n", simCtx->ti, simCtx->step);
412 ierr = ReinitializeParticlesOnInletSurface(user, simCtx->ti, simCtx->step); CHKERRQ(ierr);
413
414 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Resetting statuses for post-reinitialization settlement.\n", simCtx->ti, simCtx->step);
415 ierr = ResetAllParticleStatuses(user); CHKERRQ(ierr);
416
417 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Post-Reinitialization Settlement...\n", simCtx->ti, simCtx->step);
418 ierr = LocateAllParticlesInGrid(user, bboxlist); CHKERRQ(ierr);
419
420 }
421
422 // --- 3. Finalize State for t=0 ---
423 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Interpolating initial fields to settled particles.\n", simCtx->ti, simCtx->step);
424 ierr = InterpolateAllFieldsToSwarm(user); CHKERRQ(ierr);
425 ierr = RefreshVerificationScalarScatterState(user); CHKERRQ(ierr);
426
427 // --- 4. Initial History and Output ---
428 // Update solver history vectors with the t=0 state before the first real step
429 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
430 ierr = UpdateSolverHistoryVectors(&user[bi]); CHKERRQ(ierr);
431 }
432
433 if (simCtx->tiout > 0 || (simCtx->StepsToRun == 0 && simCtx->StartStep == 0)) {
434 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Writing initial simulation data.\n", simCtx->ti, simCtx->step);
435 ierr = WriteAllSwarmFields(user); CHKERRQ(ierr);
436
437 // --- Eulerian Field Output (MUST loop over all blocks) --- // <<< CHANGED/FIXED
438 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
439 ierr = WriteSimulationFields(&user[bi]); CHKERRQ(ierr);
440 }
441 ierr = LOG_SCATTER_METRICS(user); CHKERRQ(ierr);
442 }
443
445 ierr = EmitParticleConsoleSnapshot(user, simCtx, simCtx->step); CHKERRQ(ierr);
446 }
447
448 LOG_ALLOW(GLOBAL, LOG_INFO, "--- Initial setup complete. Ready for time marching. ---\n");
449 PetscFunctionReturn(0);
450}
PetscErrorCode ReinitializeParticlesOnInletSurface(UserCtx *user, PetscReal currentTime, PetscInt step)
Re-initializes the positions of particles currently on this rank if this rank owns part of the design...
PetscBool IsParticleConsoleSnapshotEnabled(const SimCtx *simCtx)
Returns whether periodic particle console snapshots are enabled.
Definition logging.c:525
PetscErrorCode LOG_PARTICLE_FIELDS(UserCtx *user, PetscInt printInterval)
Prints particle fields in a table that automatically adjusts its column widths.
Definition logging.c:397
static PetscErrorCode RefreshVerificationScalarScatterState(UserCtx *user)
Applies verification-only scalar truth and refreshes the scattered Eulerian scalar state.
Definition runloop.c:50
PetscBool inletFaceDefined
Definition variables.h:849
@ PARTICLE_INIT_SURFACE_RANDOM
Random placement on the inlet face.
Definition variables.h:517
@ PARTICLE_INIT_SURFACE_EDGES
Deterministic placement at inlet face edges.
Definition variables.h:520
PetscInt tiout
Definition variables.h:663
ParticleInitializationType ParticleInitialization
Definition variables.h:757
PetscInt LoggingFrequency
Definition variables.h:783
Defines a 3D axis-aligned bounding box.
Definition variables.h:154
Here is the call graph for this function:
Here is the caller graph for this function:

◆ PerformLoadedParticleSetup()

PetscErrorCode PerformLoadedParticleSetup ( SimCtx simCtx)

Finalizes the simulation state after particle and fluid data have been loaded from a restart.

This helper function performs the critical sequence of operations required to ensure the loaded Lagrangian and Eulerian states are fully consistent and the solver is ready to proceed. This includes:

  1. Verifying particle locations in the grid and building runtime links.
  2. Synchronizing particle velocity with the authoritative grid velocity via interpolation.
  3. Scattering particle source terms (e.g., volume fraction) back to the grid.
  4. Updating the solver's history vectors with the final, fully-coupled state.
  5. Writing the complete, consistent state to output files for the restart step.
Parameters
simCtxThe main simulation context.
Returns
PetscErrorCode 0 on success.

Finalizes the simulation state after particle and fluid data have been loaded from a restart.

Local to this translation unit.

Definition at line 458 of file runloop.c.

459{
460 PetscErrorCode ierr;
461 PetscFunctionBeginUser;
462 UserCtx *user = simCtx->usermg.mgctx[simCtx->usermg.mglevels-1].user;
463
464 // --- 0. Re-compute Eulerian Diffusivity from loaded fields.
465 LOG_ALLOW(GLOBAL, LOG_INFO, "Re-computing Eulerian Diffusivity from loaded fields...\n");
466 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
467 ierr = ComputeEulerianDiffusivity(&user[bi]); CHKERRQ(ierr);
468 ierr = ComputeEulerianDiffusivityGradient(&user[bi]); CHKERRQ(ierr);
469 }
470
471 // 0.1 This moves particles to their correct ranks immediately using the loaded Cell ID.
472 LOG_ALLOW(GLOBAL, LOG_INFO, "Performing fast restart migration using preloaded Cell IDs...\n");
473 ierr = MigrateRestartParticlesUsingCellID(user); CHKERRQ(ierr);
474
475 // 1. To catch any edge cases (particles with invalid CellIDs or newcomers).
476 // Because we kept the statuses, this function will now SKIP all the particles
477 // that are already on the correct rank,
478 ierr = LocateAllParticlesInGrid(user, simCtx->bboxlist); CHKERRQ(ierr);
479
480 if(get_log_level() == LOG_DEBUG){
481 LOG(GLOBAL, LOG_DEBUG, "[T=%.4f, Step=%d] Particle field states after locating loaded particles...\n", simCtx->ti, simCtx->step);
482 ierr = LOG_PARTICLE_FIELDS(user,simCtx->LoggingFrequency); CHKERRQ(ierr);
483 }
484
485 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Interpolating initial fields to settled particles.\n", simCtx->ti, simCtx->step);
486
487 // 2. Ensure particles have velocity from the authoritative loaded grid for consistency.
488 ierr = InterpolateAllFieldsToSwarm(user); CHKERRQ(ierr);
489
490 // 3. Update Eulerian source terms from the loaded particle data.
492 ierr = RefreshVerificationScalarScatterState(user); CHKERRQ(ierr);
493 } else {
494 ierr = ScatterAllParticleFieldsToEulerFields(user); CHKERRQ(ierr);
495 }
496
497 // --- 4. Initial History and Output ---
498 // Update solver history vectors with the t=0 state before the first real step
499 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
500 ierr = UpdateSolverHistoryVectors(&user[bi]); CHKERRQ(ierr);
501 }
502
503 if (simCtx->tiout > 0 || (simCtx->StepsToRun == 0 && simCtx->StartStep == 0)) {
504 LOG_ALLOW(GLOBAL, LOG_INFO, "[T=%.4f, Step=%d] Writing initial simulation data.\n", simCtx->ti, simCtx->step);
505 ierr = WriteAllSwarmFields(user); CHKERRQ(ierr);
506
507 // --- Eulerian Field Output (MUST loop over all blocks) --- // <<< CHANGED/FIXED
508 for (PetscInt bi = 0; bi < simCtx->block_number; bi++) {
509 ierr = WriteSimulationFields(&user[bi]); CHKERRQ(ierr);
510 }
511 ierr = LOG_SCATTER_METRICS(user); CHKERRQ(ierr);
512 }
513
515 ierr = EmitParticleConsoleSnapshot(user, simCtx, simCtx->step); CHKERRQ(ierr);
516 }
517
518 LOG_ALLOW(GLOBAL, LOG_INFO, "--- Initial setup complete. Ready for time marching. ---\n");
519 PetscFunctionReturn(0);
520}
PetscErrorCode MigrateRestartParticlesUsingCellID(UserCtx *user)
Fast-path migration for restart particles using preloaded Cell IDs.
#define LOG(scope, level, fmt,...)
Logging macro for PETSc-based applications with scope control.
Definition logging.h:83
PetscBool VerificationScalarOverrideActive(const SimCtx *simCtx)
Reports whether a verification-only scalar override is active.
Here is the call graph for this function:
Here is the caller graph for this function:

◆ FinalizeRestartState()

PetscErrorCode FinalizeRestartState ( SimCtx simCtx)

Performs post-load/post-init consistency checks for a restarted simulation.

This function is called from main() ONLY when a restart is being performed (i.e., StartStep > 0). It inspects the particle restart mode to determine the correct finalization procedure for the Lagrangian swarm.

  • If particles were loaded from a file (mode == "load"), it verifies their locations within the grid to establish necessary runtime links.
  • If new particles were initialized into the restarted flow (mode == "init"), it runs the full PerformInitialSetup sequence to migrate, locate, and couple the new particles with the existing fluid state.
Parameters
simCtxThe main simulation context.
Returns
PetscErrorCode 0 on success.

Performs post-load/post-init consistency checks for a restarted simulation.

Local to this translation unit.

Definition at line 528 of file runloop.c.

529{
530 PetscErrorCode ierr;
531 UserCtx *user = simCtx->usermg.mgctx[simCtx->usermg.mglevels - 1].user;
532
533 PetscFunctionBeginUser;
534
535 LOG_ALLOW(GLOBAL, LOG_INFO, "--- Finalizing RESTART from state (step=%d, t=%.4f) ---\n", simCtx->StartStep, simCtx->ti);
536
537 // This function only needs to handle the particle finalization logic.
538 // The Eulerian state is assumed to be fully loaded and consistent at this point.
539 if (simCtx->np > 0) {
540
541 // Use the particle restart mode to decide the workflow.
542 if (strcmp(simCtx->particleRestartMode, "load") == 0) {
543 // PARTICLES WERE LOADED: The state is complete, but we must verify
544 // the loaded CellIDs and build the in-memory grid-to-particle links.
545 LOG_ALLOW(GLOBAL, LOG_INFO, "Particle Mode 'load': Verifying particle locations and building grid links...\n");
546 ierr = PerformLoadedParticleSetup(simCtx); CHKERRQ(ierr);
547
548 } else { // Mode must be "init"
549 // PARTICLES WERE RE-INITIALIZED: They need to be fully settled and coupled
550 // to the surrounding (restarted) fluid state.
551 LOG_ALLOW(GLOBAL, LOG_INFO, "Particle Mode 'init': Running full initial setup for new particles in restarted flow.\n");
552 ierr = PerformInitializedParticleSetup(simCtx); CHKERRQ(ierr);
553 }
554 } else {
555 LOG_ALLOW(GLOBAL, LOG_INFO, "No particles in simulation, restart finalization is complete.\n");
556
557 // Write the initial eulerian fields (this is done in PerformInitialSetup if particles exist.)
558 for(PetscInt bi = 0; bi < simCtx->block_number; bi ++){
559 ierr = WriteSimulationFields(&user[bi]); CHKERRQ(ierr);
560 }
561 }
562
563 LOG_ALLOW(GLOBAL, LOG_INFO, "--- Restart state successfully finalized. --\n");
564
565 PetscFunctionReturn(0);
566}
PetscErrorCode PerformInitializedParticleSetup(SimCtx *simCtx)
Internal helper implementation: PerformInitializedParticleSetup().
Definition runloop.c:382
PetscErrorCode PerformLoadedParticleSetup(SimCtx *simCtx)
Internal helper implementation: PerformLoadedParticleSetup().
Definition runloop.c:458
char particleRestartMode[16]
Definition variables.h:759
Here is the call graph for this function:
Here is the caller graph for this function: