Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(prt): rework vertical tracking approach #2066

Merged
merged 49 commits into from
Dec 14, 2024

Conversation

wpbonelli
Copy link
Contributor

@wpbonelli wpbonelli commented Nov 20, 2024

Address #2014. This fixes a crash, avoids a potential infinite loop condition, and reworks vertical tracking behavior more generally. The aim is to produce results matching MF6.5 and MP7 by default (barring a few edge cases), while giving more control over how particles behave in dry conditions. Also some miscellaneous reorganization and cleanup.

Legend

Diagrams use the following conventions.

  • Stadium-shaped boxes represent steps or processes.
  • Square boxes represent outcomes.
  • Diamond boxes represent conditions (i.e. runtime state).
  • Round-corner boxes represent user options.
  • Thin lines represent decisions made by the program on the basis of runtime state, e.g. particle, cell, flows.
  • Thick lines represent decisions made by the user by way of options.
  • Green outcome boxes indicate the particle remains active.
  • Red outcome boxes indicate the particle terminates.
flowchart LR
     OPTION[Outcome]
    OPTION(OPTION) ==> |Yes| STEP([Step])
    STEP --> ACTIVE
    OPTION ==> |No| CONDITION{Condition}
    CONDITION --> |Yes| TERMINATE
    CONDITION --> |No| ACTIVE
    ACTIVE[Active]:::active
    TERMINATE[Termination]:::terminate

    classDef active stroke:#98fb98
    classDef terminate stroke:#f08080
Loading

The problem

When a particle is in the flow field, vertical motion can be solved in the same way as lateral motion. Special handling is necessary above the water table.

A "dry" cell is either 1) an inactive cell or 2) an active-but-dry cell, as can occur with the Newton formulation.

Normally, an inactive cell might be dry or explicitly disabled (idomain). With Newton, dry cells remain active.

The approach

Release-time and tracking-time considerations are described (and implemented) separately.

Each particle is either released into the simulation, or terminates unreleased. In the former case the particle's first record will be reason 0 (release), status 1 (active). In the latter reason 3 (termination), status 8 (permanently unreleased).

At each time step, the PRT model applies the tracking method to each particle. The particle's trajectory is solved over the model grid until the end of the time step, or until the particle terminates (due e.g. to stop time or encountering a termination condition), whichever occurs first.

Sometimes it is convenient to avoid "stranding" particles — rather than terminating dry particles, it is often convenient instead to move them down to the saturated zone and continue tracking. PRT allows particles (and indeed configures them by default) to move instantaneously down to the water table in dry conditions.

Release

At release time, PRT decides whether to release each particle or to terminate it unreleased.

If the release cell is active, the particle will be released at the specified coordinates.

If the release cell is inactive, behavior is determined by the DRAPE option. If the DRAPE option is enabled, the particle will be "draped" down to and released from the top-most active cell beneath it, if any. If there is no active cell underneath the particle in any layer, or if DRAPE is not enabled, the particle will terminate unreleased (with status code 8).

Since under the Newton formulation dry cells can remain active, the DRAPE option has no effect when Newton is on (assuming particles are not released into disabled grid regions). Vertical tracking behavior with Newton can be configured with tracking-time settings.

flowchart LR
    ACTIVE --> |No| DRAPE(DRAPE)
    ACTIVE{Cell active?} --> |Yes| RELEASE[Release in specified cell]:::release
    DRAPE ==> |No| TERMINATE:::terminate
    DRAPE ==> |Yes| ACTIVE_UNDER{Active under?}
    ACTIVE_UNDER --> |Yes| RELEASE_AT_TABLE[Drape to active cell]:::release
    ACTIVE_UNDER --> |No| TERMINATE[Terminate]

    classDef release stroke:#98fb98
    classDef terminate stroke:#f08080
Loading

Tracking

A particle might find itself above the water table for one of two reasons:

  1. It was released above the water table.

    With the Newton formulation, particles can be released into dry-but-active cells.

  2. The water table has receded.

    Particle trajectories are solved over the same time discretization used by the flow model. A particle may be immersed in the flow field in one time step, and find that the water table has dropped below it in the next.

Tracking and termination decisions are made on the basis of information like

  1. a cell's active status
  2. whether the cell is dry
  3. whether the cell has outgoing flow across any face
  4. whether the particle is dry (above the water table)
  5. the particle's prior path

A particle which finds itself in an inactive cell will terminate with status code 7. This is consistent with MODPATH 7's behavior.

A particle in a dry-but-active cell, or above the water table in a partially saturated cell, need not terminate. We call such a particle dry. MODFLOW version 6.6.0 introduces a new option DRY_TRACKING_METHOD for the PRP package, determining how dry particles should behave. Supported values are:

  • DROP (default)
  • STOP
  • STAY

If DROP is selected, or if a DRY_TRACKING_METHOD is unspecified, a particle in a dry position is passed vertically and instantaneously to the water table (if the cell is partially saturated) or to the bottom of the cell (if the cell is dry). This repeats (i.e. the particle may drop through multiple cells) until it reaches the water table. Tracking then proceeds as usual.

Note: A divide-by-zero crash has been fixed for gfortran, which could occur upon a particle's entry into a dry cell in a structured grid.

If STOP is selected, dry particles will be terminated.

If STAY is selected, a dry particle will remain stationary until a) the water table rises and tracking can continue or b) the simulation ends.

flowchart LR
    ACTIVE{Cell active?} --> |No| TERMINATE{Terminate}
    ACTIVE{Cell active?} --> |Yes| PARTICLE_DRY
    PARTICLE_DRY{Particle dry?} --> |Yes| DRY_TRACKING_METHOD(DRY_TRACKING_METHOD)
    DRY_TRACKING_METHOD ==> |STOP| TERMINATE[Terminate]:::terminate
    DRY_TRACKING_METHOD ==> |DROP| CELL_DRY{Cell dry?}
    CELL_DRY --> |Yes| DROP_BOTTOM[Pass to cell bottom]:::track
    CELL_DRY --> |No| DROP_TABLE([Pass to water table])
    DRY_TRACKING_METHOD ==> |STAY| STAY[Stationary]:::track
    DROP_TABLE --> TRACK:::track
    PARTICLE_DRY --> |No| TRACK[Track normally]

    classDef track stroke:#98fb98
    classDef terminate stroke:#f08080
Loading

Caveat

In MF6.5, behavior was as described by DROP, with one major exception: lack of an exit face (i.e. any face with outgoing flow) took precedence over cell saturation; a particle finding itself in a dry cell with no outgoing flow would previously terminate, where if DROP is selected (or a dry tracking method unspecified) the pass-to-bottom method will now be applied instead.

With this change, it also becomes necessary to prohibit backtracking between vertically adjacent pairs of cells within the same time step, in order to avoid the possibility of infinite loops — a particle might otherwise be passed endlessly between e.g. the bottom face of a cell containing a pumping well and the top face of the cell below. Note that this limitation applies only to vertically adjacent cells, and only to the immediately previous cell — a particle may re-enter a cell after entering a third cell.


Checklist of items for pull request

  • Replaced section above with description of pull request
  • Added new test or modified an existing test
  • Formatted new and modified Fortran source files with fprettify
  • Updated definition files
  • Updated develop.tex with a plain-language description of the bug fix, change, feature; required for changes that may affect users
  • Updated input and output guide
  • Removed checklist items not relevant to this pull request

@wpbonelli wpbonelli added the bug label Nov 20, 2024
@wpbonelli wpbonelli added this to the 6.6.0 milestone Nov 20, 2024
@wpbonelli wpbonelli force-pushed the prt-vertical branch 2 times, most recently from 3e19940 to 5038e59 Compare November 23, 2024 17:46
@wpbonelli wpbonelli changed the title fix(prt-prp): fix drape with newton fix(prt): rework vertical tracking approach Nov 25, 2024
@wpbonelli wpbonelli marked this pull request as ready for review November 26, 2024 02:01
@wpbonelli wpbonelli marked this pull request as draft December 1, 2024 19:38
@@ -21,6 +21,48 @@ \subsection{Specifying Cell Face Flows using IFLOWFACE}
\subsection{Particle Mass Budget}
A summary of all inflow (sources) and outflow (sinks) of particle mass is called a mass budget. The particle mass budget is printed to the PRT Model Listing File for selected time steps. In the current implementation, each particle is assigned unit mass, and the numerical value of the flow can be interpreted as particles per time.

\subsection{Vertical Tracking}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should this be here or in the PRP chapter? Also, should it be restated in the supplemental technical info?

..planning to repackage some of this with illustration in an informal migration guide as well, to distribute with the release to testing partners / users

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My opinion is that it is fine to go here, but it may be worthwhile pointing to it from the PRP chapter.

It probably wouldn't hurt to also summarize this in the supplemental technical information.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agree. Though in supptechinfo I'd favor a very generic description that points to mf6io for specifics, since we'll likely be refining the dry tracking options down the road

@wpbonelli wpbonelli marked this pull request as ready for review December 5, 2024 20:53
Copy link
Contributor

@langevin-usgs langevin-usgs left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice work on this PR. Looks like a lot of work went into it. I like the STAY option in particular. Not sure this has ever been possible with any particle trackers. Will be nice to have this in the next release.

I made a few comments for consideration. None of them are substantial.

doc/mf6io/prt/prt.tex Outdated Show resolved Hide resolved
@@ -21,6 +21,48 @@ \subsection{Specifying Cell Face Flows using IFLOWFACE}
\subsection{Particle Mass Budget}
A summary of all inflow (sources) and outflow (sinks) of particle mass is called a mass budget. The particle mass budget is printed to the PRT Model Listing File for selected time steps. In the current implementation, each particle is assigned unit mass, and the numerical value of the flow can be interpreted as particles per time.

\subsection{Vertical Tracking}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My opinion is that it is fine to go here, but it may be worthwhile pointing to it from the PRP chapter.

It probably wouldn't hurt to also summarize this in the supplemental technical information.

doc/ReleaseNotes/develop.tex Show resolved Hide resolved
doc/ReleaseNotes/develop.tex Show resolved Hide resolved
doc/mf6io/prt/prt.tex Show resolved Hide resolved
doc/mf6io/prt/prt.tex Outdated Show resolved Hide resolved
src/Model/ParticleTracking/prt-prp.f90 Outdated Show resolved Hide resolved
src/Solution/ParticleTracker/vertical.md Outdated Show resolved Hide resolved
@wpbonelli wpbonelli force-pushed the prt-vertical branch 2 times, most recently from f8f35ef to 398b9bb Compare December 6, 2024 15:47
doc/mf6io/prt/prt.tex Outdated Show resolved Hide resolved
doc/mf6io/prt/prt.tex Outdated Show resolved Hide resolved
doc/mf6io/prt/prt.tex Outdated Show resolved Hide resolved
Copy link
Contributor

@aprovost-usgs aprovost-usgs left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for all your good planning and hard work on this

doc/mf6io/prt/prt.tex Show resolved Hide resolved
doc/mf6io/prt/prt.tex Outdated Show resolved Hide resolved
src/Model/ParticleTracking/prt-prp.f90 Show resolved Hide resolved
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's discuss whether the Method module is the best place for cell-related checks. The Method module is meant to be a generic parent for all tracking methods. These are currently limited to methods that track across grids/cells, but will someday include methods that track, e.g., across streams/reaches

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I had a similar thought but did not follow it up.

This "check to see if we need to report anything or terminate" logic has always been in the base method — at some point I think we factored out a routine as it got more complicated.

Could have an intermediate CellMethodType between MethodType and the cell methods, and put these checks there? Maybe eventually worth considering a less ad-hoc way to implement termination/reporting checks, since they are also necessary at other levels (in subcell methods they are just interspersed throughout)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

seems ok to have generic "check to see if we need to report anything or terminate" logic in the base method as long as it's agnostic of specific tracking domains like cells. if the original checks are tracking-domain-specific, we should (ideally) refactor them so they aren't. agree it would be best to put cell-specific checks to the CellMethod level. also agree that (eventually) it would be good to make such checks less ad hoc at all levels

src/Solution/ParticleTracker/MethodDis.f90 Show resolved Hide resolved
src/Solution/ParticleTracker/MethodDisv.f90 Show resolved Hide resolved
src/Solution/ParticleTracker/Particle.f90 Show resolved Hide resolved
src/Solution/ParticleTracker/Particle.f90 Show resolved Hide resolved
autotest/test_prt_dry.py Show resolved Hide resolved
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is markdown that's in with the source code. This will presumably be visible when someone is editing the source code using appropriate software, like we were the other day. If so, that's cool. however, it does seem that when we modify the dry tracking behavior, we'll need to update this description as well as the similar description in mf6io. Any way to use one file for both? Would be cool to have the flow diagrams in mf6io, as well (if they already are, i missed it, but i didn't actually build the document)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't know of any way to render the diagrams in LaTeX sadly. I figured I'd keep the notes in there until the release, while drafting the migration guide, then maybe it can be removed. It's always accessible in the git history again. And long term maybe we can look into adding diagrams to mf6io?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sounds like a good plan, now and long term

@wpbonelli
Copy link
Contributor Author

merging as @aprovost-usgs has given the OK offline

@wpbonelli wpbonelli merged commit 3045d44 into MODFLOW-USGS:develop Dec 14, 2024
20 checks passed
@wpbonelli wpbonelli deleted the prt-vertical branch December 14, 2024 16:41
wpbonelli added a commit that referenced this pull request Dec 17, 2024
…es (#2107)

Mention the EXIT_SOLVE_TOLERANCE option now in the PRP package in the release notes, I missed this before. This was previously a required option, but need not be — set a default of 1e-5, and simplify some tests in light of this.

Also draft a PRT migration guide, building on the development notes added in #2066, which could be distributed with the release. I'm not sure whether this is something to version, and if so, where it should go, but figured it can't hurt to have it.
@javgs-bd
Copy link

javgs-bd commented Jan 7, 2025

Big time apologies for not replying to the messages in #2014. A silly outlook rule kept me blind to all the subsequent development there. I sent an email, but in case it did not get through, I can't thank you enough for all the effort put into this. I second the kudos on the STAY option. Game changer for us.
We will try the new options right now as our need for PRT will reactivate soon.

Please use the test model at will. No attribution needed at all.

@aprovost-usgs
Copy link
Contributor

@javgs-bd No worries! Glad you're finding the STAY option useful, and thanks for letting us use the test model

wpbonelli added a commit that referenced this pull request Jan 8, 2025
#2066 introduced a bug where particle coordinates were prematurely transformed back from cell-local coords to model coords. This produced incorrect particle positions in/near quad-refined cells.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants