diff --git a/autotest/__snapshots__/test_prt_dry/test_mf6model[0-prtdry].npy b/autotest/__snapshots__/test_prt_dry/test_mf6model[0-prtdry].npy new file mode 100644 index 00000000000..5e8e184fe31 Binary files /dev/null and b/autotest/__snapshots__/test_prt_dry/test_mf6model[0-prtdry].npy differ diff --git a/autotest/__snapshots__/test_prt_dry/test_mf6model[1-prtdry_drape].npy b/autotest/__snapshots__/test_prt_dry/test_mf6model[1-prtdry_drape].npy new file mode 100644 index 00000000000..131af10426f Binary files /dev/null and b/autotest/__snapshots__/test_prt_dry/test_mf6model[1-prtdry_drape].npy differ diff --git a/autotest/__snapshots__/test_prt_dry/test_mf6model[2-prtdry_drop].npy b/autotest/__snapshots__/test_prt_dry/test_mf6model[2-prtdry_drop].npy new file mode 100644 index 00000000000..de1b2802bb4 Binary files /dev/null and b/autotest/__snapshots__/test_prt_dry/test_mf6model[2-prtdry_drop].npy differ diff --git a/autotest/__snapshots__/test_prt_dry/test_mf6model[3-prtdry_stop].npy b/autotest/__snapshots__/test_prt_dry/test_mf6model[3-prtdry_stop].npy new file mode 100644 index 00000000000..ff9b2023675 Binary files /dev/null and b/autotest/__snapshots__/test_prt_dry/test_mf6model[3-prtdry_stop].npy differ diff --git a/autotest/__snapshots__/test_prt_dry/test_mf6model[4-prtdry_stay].npy b/autotest/__snapshots__/test_prt_dry/test_mf6model[4-prtdry_stay].npy new file mode 100644 index 00000000000..5a921e5f6f3 Binary files /dev/null and b/autotest/__snapshots__/test_prt_dry/test_mf6model[4-prtdry_stay].npy differ diff --git a/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[0-prtvor1l2r].npy b/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[0-prtvor1l2r].npy index e52948316f1..ff1e38b14d3 100644 Binary files a/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[0-prtvor1l2r].npy and b/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[0-prtvor1l2r].npy differ diff --git a/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[1-prtvor1welp].npy b/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[1-prtvor1welp].npy index 3a4f3890ddd..c3a6a65497d 100644 Binary files a/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[1-prtvor1welp].npy and b/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[1-prtvor1welp].npy differ diff --git a/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[2-prtvor1weli].npy b/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[2-prtvor1weli].npy index a0e91bdb042..9ec69de0ee0 100644 Binary files a/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[2-prtvor1weli].npy and b/autotest/__snapshots__/test_prt_voronoi1/test_mf6model[2-prtvor1weli].npy differ diff --git a/autotest/test_prt_dry.py b/autotest/test_prt_dry.py new file mode 100644 index 00000000000..abc11fa7fe5 --- /dev/null +++ b/autotest/test_prt_dry.py @@ -0,0 +1,474 @@ +""" +Tests particle tracking in "dry" conditions. + +The PRP package provides the `DRY` option +to specify how particles should behave in +dry conditions when the flow model enables +the Newton formulation. + +This test case is adapted from the example +simulation provided by @javgs-bd in +https://github.com/MODFLOW-USGS/modflow6/issues/2014. +""" + +import os +from warnings import warn + +import flopy +import matplotlib.colors as clt +import matplotlib.pyplot as plt +import numpy as np +import pandas as pd +import pytest +from framework import TestFramework +from prt_test_utils import get_model_name + +simname = "prtdry" +cases = [ + # expect termination with status 7 immediately in 1st time step + simname, + # expect all particles to be draped to water table and tracked + simname + "_drape", + # the rest of the test cases activate newton and test the behavior + # of the DRY option. with drop, expect all particles to be dropped + # to the highest active cell below and then to be tracked as usual. + simname + "_drop", + # expect termination with status 7 immediately in 1st time step + simname + "_stop", + # expect particles to remain in release positions until the water + # table rises to meet them + simname + "_stay", +] + +nper = 3 +perlen = [1, 300, 1000] +nstp = [1, 30, 1] +tsmult = [1, 1.1, 1] + +Lx = 100.0 +Ly = 100.0 + +nlay = 3 +ncol = 20 +nrow = 20 +arot = 0 + +delr = Lx / ncol +delc = Ly / nrow + +strt = 1600 + +z_top = 1600 +thickness = 10 +xorigin, yorigin = (0, 0) + +z_bot = np.zeros((nlay, nrow, ncol)) +for i in range(nlay): + z_bot[i, :, :] = z_top - (i + 1) * thickness + +period_data = [] +for i in range(nper): + period_data.append((perlen[i], nstp[i], tsmult[i])) + + +def build_gwf_sim(name, gwf_ws, mf6, newton=False): + sim = flopy.mf6.MFSimulation( + sim_name=simname, exe_name=mf6, version="mf6", continue_=True, sim_ws=gwf_ws + ) + + gwf_name = get_model_name(name, "gwf") + gwf = flopy.mf6.ModflowGwf( + sim, + modelname=gwf_name, + model_nam_file=gwf_name + ".nam", + newtonoptions="NEWTON" if newton else None, + save_flows=True, + ) + + tdis = flopy.mf6.ModflowTdis( + sim, time_units="days", nper=nper, perioddata=period_data + ) + + dis = flopy.mf6.ModflowGwfdis( + gwf, + length_units="METERS", + xorigin=xorigin, + yorigin=yorigin, + angrot=arot, + nlay=nlay, + nrow=nrow, + ncol=ncol, + delr=delr, + delc=delc, + top=z_top, + botm=z_bot, + filename=gwf.name + ".dis", + pname="dis", + ) + + ims = flopy.mf6.ModflowIms( + sim, + print_option="SUMMARY", + csv_outer_output_filerecord="ims_outer.csv", + csv_inner_output_filerecord="ims_inner.csv", + outer_dvclose=1e-5, + outer_maximum=100, + under_relaxation="DBD", + under_relaxation_gamma=0.01, + under_relaxation_theta=0.7, + under_relaxation_kappa=0.01, + under_relaxation_momentum=0.0, + inner_maximum=100, + inner_dvclose=1e-6, + rcloserecord=0.1, + linear_acceleration="BICGSTAB", + relaxation_factor=0.99, + number_orthogonalizations=2, + reordering_method="NONE", + pname=gwf.name, + ) + + oc = flopy.mf6.ModflowGwfoc( + gwf, + budget_filerecord=gwf.name + ".cbb", + head_filerecord=gwf.name + ".hds", + saverecord=[("HEAD", "ALL"), ("BUDGET", "ALL")], + printrecord=[("HEAD", "ALL"), ("BUDGET", "ALL")], + ) + + ic = flopy.mf6.modflow.mfgwfic.ModflowGwfic(gwf, strt=strt) + + bdry_name = "upstr.chd" + + filename = gwf.name + "." + bdry_name + pname = bdry_name + + stress_period_data = {} + lay = 1 + row, col = (0, 0) + h_upstr = 1587 + + for i in range(nper): + stress_period_data[i] = [lay, row, col, h_upstr, bdry_name] + + chd = flopy.mf6.ModflowGwfchd( + gwf, + stress_period_data=stress_period_data, + filename=filename, + pname=pname, + boundnames=True, + save_flows=True, + ) + + bdry_name = "dwstr.chd" + + filename = gwf.name + "." + bdry_name + pname = bdry_name + + stress_period_data = {} + lay = 1 + row, col = (nrow - 1, ncol - 1) + h_dwstr = 1582 + + for i in range(nper): + stress_period_data[i] = [lay, row, col, h_dwstr, bdry_name] + + chd = flopy.mf6.ModflowGwfchd( + gwf, + stress_period_data=stress_period_data, + filename=filename, + pname=pname, + boundnames=True, + save_flows=True, + ) + + bdry_name = "inj.wel" + + filename = gwf.name + "." + bdry_name + pname = bdry_name + + stress_period_data = {} + + lay = 1 + row, col = (int(nrow / 4), int(ncol / 4)) + q = 100 + + for i in range(1, nper): + stress_period_data[i] = [lay, row, col, q, bdry_name] + + wel = flopy.mf6.ModflowGwfwel( + gwf, + stress_period_data=stress_period_data, + pname=pname, + boundnames=True, + save_flows=True, + ) + + icelltype = 1 + + npf = flopy.mf6.ModflowGwfnpf( + gwf, + save_flows=True, + icelltype=icelltype, + k=0.5, + k33=0.1, + save_specific_discharge=True, + save_saturation=True, + ) + + sto = flopy.mf6.ModflowGwfsto( + gwf, + save_flows=True, + iconvert=1, + ss=0.0001, + sy=0.1, + steady_state={0: True, 2: True}, + transient={1: True}, + ) + return sim + + +def build_prt_sim(name, gwf, prt_ws, mf6, drape=False, dry_tracking_method=False): + sim = flopy.mf6.MFSimulation( + sim_name=name, exe_name=mf6, version="mf6", continue_=True, sim_ws=prt_ws + ) + + gwf_ws = gwf.model_ws + + prt_name = get_model_name(name, "prt") + prt = flopy.mf6.ModflowPrt( + sim, + modelname=prt_name, + model_nam_file=prt_name + ".nam", + ) + + tdis = flopy.mf6.ModflowTdis( + sim, time_units="days", nper=nper, perioddata=period_data + ) + + dis = flopy.mf6.ModflowPrtdis( + prt, + length_units="METERS", + xorigin=xorigin, + yorigin=yorigin, + angrot=arot, + nlay=nlay, + nrow=nrow, + ncol=ncol, + delr=delr, + delc=delc, + top=z_top, + botm=z_bot, + filename=prt_name + ".dis", + pname="dis", + ) + + porosity = 0.1 + + flopy.mf6.ModflowPrtmip(prt, porosity=porosity, pname="mip") + + # particles are released in layer 0 + offsets = [ + (-1, 0, 0), + (-1, -1, 0), + (-1, 1, 0), + (-1, 0, -1), + (-1, 0, 1), + ] + + lay = 1 + row, col = (int(nrow / 4), int(ncol / 4)) + prp_cells = [(lay + k, row + i, col + j) for k, i, j in offsets] + + prp_coords = [] + prp_data = [] + + for i, (lay, row, col) in enumerate(prp_cells): + x, y, z = ( + gwf.modelgrid.xyzcellcenters[0][row, col], + gwf.modelgrid.xyzcellcenters[1][row, col], + gwf.modelgrid.xyzcellcenters[2][lay, row, col], + ) + + prp_lst = [i, (lay, row, col), x, y, z] + + prp_data.append(prp_lst) + prp_coords.append((x, y, z)) + + flopy.mf6.ModflowPrtprp( + prt, + nreleasepts=len(prp_data), + packagedata=prp_data, + nreleasetimes=1, + releasetimes=[(0.0,)], + exit_solve_tolerance=1e-7, + drape=drape, + dry_tracking_method=dry_tracking_method, + pname="prp", + filename="tracking_1.prp", + print_input=True, + ) + + budgetfile = prt.name + ".bud" + trackfile = prt.name + ".trk" + trackcsvfile = prt.name + ".csv" + + flopy.mf6.ModflowPrtoc( + prt, + budget_filerecord=[budgetfile], + track_filerecord=[trackfile], + trackcsv_filerecord=[trackcsvfile], + saverecord=[("BUDGET", "ALL")], + pname="oc", + ) + + rel_prt_folder = os.path.relpath(gwf_ws, start=prt_ws) + + packagedata = [ + ("GWFHEAD", f"{rel_prt_folder}/{gwf.name}.hds"), + ("GWFBUDGET", f"{rel_prt_folder}/{gwf.name}.cbb"), + ] + + flopy.mf6.ModflowPrtfmi( + prt, + packagedata=packagedata, + ) + + ems = flopy.mf6.ModflowEms(sim, pname="ems", filename=prt.name + ".ems") + + return sim + + +def build_models(idx, test, newton, drape=False, dry_tracking_method=False): + gwf_sim = build_gwf_sim( + test.name, test.workspace / "gwf", test.targets["mf6"], newton=newton + ) + prt_sim = build_prt_sim( + test.name, + gwf_sim.get_model(), + test.workspace / "prt", + test.targets["mf6"], + drape=drape, + dry_tracking_method=dry_tracking_method, + ) + return gwf_sim, prt_sim + + +def check_output(idx, test, snapshot): + name = test.name + gwf_ws = test.workspace / "gwf" + prt_ws = test.workspace / "prt" + gwf_name = get_model_name(name, "gwf") + prt_name = get_model_name(name, "prt") + gwf_sim = test.sims[0] + gwf = gwf_sim.get_model(gwf_name) + hds_file = gwf.name + ".hds" + trackcsv_file = prt_name + ".csv" + trackcsv_path = prt_ws / trackcsv_file + pls = pd.read_csv(trackcsv_path) + strtpts = pls[pls.ireason == 0] + + # compare to expected results + decimals = 1 if "drop" in name else 2 + actual = pls.drop(["name", "icell"], axis=1).round(decimals).reset_index(drop=True) + # ignore particle 4, it terminates early with optimization=2 when built with ifort + if "drop" in name: + actual = actual.drop(actual[actual.irpt == 4].index) + assert snapshot == actual.to_records(index=False) + + plot_pathlines = False + if plot_pathlines: + + def plot_pathlines_and_timeseries( + ax, mg, ibd, pathlines, timeseries, plottitle, layer=1 + ): + ax.set_aspect("equal") + mm = flopy.plot.PlotMapView(model=gwf, ax=ax, layer=layer) + mm.plot_grid(color=(0.4, 0.4, 0.4, 0.5), lw=0.2) + mm.plot_grid(color=(0.4, 0.4, 0.4, 0.5), lw=0.2) + mm.plot_bc("WEL", plotAll=True) + mm.plot_bc("CHD", plotAll=True) + mm.plot_grid(lw=0.5) + # mm.plot_array(gwf.output.head().get_data()) + v = mm.plot_array(ibd, cmap=cmapbd, edgecolor="gray") + plt.scatter(pathlines["x"], pathlines["y"]) + mm.plot_pathline(pathlines, layer="all", colors=["blue"], lw=0.75) + ax.set_title(plottitle, fontsize=12) + ax.scatter(strtpts.x, strtpts.y) + from shapely.geometry import Polygon + + cellids = [105, 85, 125, 104, 106] + polys = [Polygon(gwf.modelgrid.get_cell_vertices(ic)) for ic in cellids] + mm.plot_shapes(polys, alpha=0.2) + plt.show() + + fig, ax = plt.subplots(1, 1, figsize=(8, 8)) + cmapbd = clt.ListedColormap(["b", "r"]) + chdcells = [] + for pckge in gwf.get_package("chd"): + chd_nodes = pckge.stress_period_data.get_data(0).cellid + for item in chd_nodes: + chdcells.append(item) + + welcells = [] + pckg = gwf.get_package("wel") + wel_nodes = pckg.stress_period_data.get_data(1).cellid + for item in wel_nodes: + welcells.append(item) + + # identify the boundary locations + ibd = np.zeros((nlay, nrow, ncol), dtype=int) + ilay, irow, icol = zip(*chdcells) + ibd[ilay, irow, icol] = 1 + ilay, irow, icol = zip(*welcells) + ibd[ilay, irow, icol] = 2 + ibd = np.ma.masked_equal(ibd, 0) + + plot_pathlines_and_timeseries(ax, gwf.modelgrid, ibd, pls, None, name) + + plot_3d = False + if plot_3d: + try: + import pyvista as pv + from flopy.export.vtk import Vtk + except: + warn("Couldn't make 3D plots, need pyvista/vtk") + return + + vert_exag = 1 + vtk = Vtk(model=gwf, binary=False, vertical_exageration=vert_exag, smooth=False) + vtk.add_model(gwf) + vtk.add_pathline_points(pls.to_records(index=False)) + gwf_mesh, prt_mesh = vtk.to_pyvista() + axes = pv.Axes(show_actor=False, actor_scale=2.0, line_width=5) + p = pv.Plotter(window_size=[700, 700]) + p.enable_anti_aliasing() + p.add_mesh(gwf_mesh, opacity=0.025, style="wireframe") + p.add_mesh( + prt_mesh, + point_size=8, + line_width=2.5, + smooth_shading=True, + color="blue", + ) + p.show() + + +@pytest.mark.parametrize("idx, name", enumerate(cases)) +def test_mf6model(idx, name, function_tmpdir, targets, array_snapshot): + dry_tracking_methods = ["drop", "stop", "stay"] + if any(t in name for t in dry_tracking_methods): + dry_tracking_method = name[-4:] + else: + dry_tracking_method = None + newton = any(t in name for t in dry_tracking_methods) + drape = "drape" in name + test = TestFramework( + name=name, + workspace=function_tmpdir, + build=lambda t: build_models(idx, t, newton, drape, dry_tracking_method), + check=lambda t: check_output(idx, t, array_snapshot), + targets=targets, + compare=None, + ) + test.run() diff --git a/autotest/test_prt_disv1.py b/autotest/test_prt_newton_disv1.py similarity index 98% rename from autotest/test_prt_disv1.py rename to autotest/test_prt_newton_disv1.py index d9d62e7793a..bbba8e6ad04 100644 --- a/autotest/test_prt_disv1.py +++ b/autotest/test_prt_newton_disv1.py @@ -6,13 +6,7 @@ refined cells, instead of the new ternary method which applies more generally to polygonal cells. -The simulation includes a single stress period -with multiple time steps. This serves to test -whether PRT properly solves trajectories over -"internal" time steps, i.e. within the step's -slice of simulation time, as well as extending -tracking to termination or particle stop times -during the simulation's final time step. +The simulation uses the Newton formulation. Several cases are provided: - default: No user-specified tracking times, MP7 in pathline mode. diff --git a/autotest/test_prt_voronoi1.py b/autotest/test_prt_voronoi1.py index 16e0ebf1d65..a36eee6c4b4 100644 --- a/autotest/test_prt_voronoi1.py +++ b/autotest/test_prt_voronoi1.py @@ -10,7 +10,6 @@ """ from pathlib import Path -from platform import processor, system import flopy import matplotlib as mpl @@ -267,6 +266,7 @@ def plot_output(idx, test): ax = plt.subplot(1, 1, 1, aspect="equal") pmv = flopy.plot.PlotMapView(model=gwf, ax=ax) pmv.plot_grid(alpha=0.25) + pmv.plot_ibound(alpha=0.5) headmesh = pmv.plot_array(head, alpha=0.25) cv = pmv.contour_array(head, levels=np.linspace(0, 1, 9), colors="black") @@ -331,7 +331,7 @@ def plot_output(idx, test): plt.savefig(prt_ws / f"{name}.png") -def check_output(idx, test, snapshot): +def check_output(idx, test): name = test.name prt_ws = test.workspace / "prt" prt_name = get_model_name(name, "prt") @@ -349,22 +349,20 @@ def check_output(idx, test, snapshot): pls = pd.read_csv(prt_ws / prt_track_csv_file, na_filter=False) endpts = pls[pls.ireason == 3] # termination - # compare pathlines with snapshot. particles shouldn't - # have moved vertically. round for cross-platform error. - # skip macos-14 in CI because grid is slightly different - if not (system() == "Darwin" and processor() == "arm"): - assert snapshot == endpts.drop("name", axis=1).round(1).to_records(index=False) + assert np.allclose(endpts.z, 0.5) + assert np.isclose(endpts.y.min(), 1, atol=4) + assert np.isclose(endpts.y.max(), 996, atol=4) @requires_pkg("syrupy") @pytest.mark.slow @pytest.mark.parametrize("idx, name", enumerate(cases)) -def test_mf6model(idx, name, function_tmpdir, targets, benchmark, array_snapshot, plot): +def test_mf6model(idx, name, function_tmpdir, targets, benchmark, plot): test = TestFramework( name=name, workspace=function_tmpdir, build=lambda t: build_models(idx, t), - check=lambda t: check_output(idx, t, array_snapshot), + check=lambda t: check_output(idx, t), plot=lambda t: plot_output(idx, t) if plot else None, targets=targets, compare=None, diff --git a/doc/ReleaseNotes/develop.tex b/doc/ReleaseNotes/develop.tex index e222b010532..4c5e63e92cc 100644 --- a/doc/ReleaseNotes/develop.tex +++ b/doc/ReleaseNotes/develop.tex @@ -51,8 +51,9 @@ \item The number of characters used to represent integers and floating point numbers in MODFLOW input files was restricted to 30. The program was modified to accept any number of characters provided the number is valid. This may be useful for parameter estimation programs that use character substitution to create new input files. \item A string to character array conversion function in the BMI interface could fail on Apple silicon macOS with recent versions of GNU Fortran, producing array extent errors at runtime. This has been fixed by properly specifying the intent of a dummy argument in the relevant function. \item The UZF Package will facilitate UZF objects with areas less than the area of the host cell. However, within the GWE model type, the UZE package will not work properly when the area of UZF objects is not equal to the area of the host cell. New code was added to ensure that the area of each UZF object is equal to that of the host grid cell. When this condition is not satisfied, the new code will stop the simulation with an error message indicating which cell is in violation. - % \item xxx - + \item With a flow model using the Newton formulation, the PRT model could crash upon a particle's entry into a dry cell. This has been fixed. + \item With a flow model using the Newton formulation, the PRT model could enter an endless loop upon a particle's entry to a dry cell if that cell contains a boundary package (e.g. a pumping well). Where the particle should be captured and terminate, it would instead be passed back and forth between the cell bottom and the top of the cell below. To avoid this, particles are forbidden from backtracking (reentering the previous cell) within the same time step. + \item The PRT model now allows more control over vertical particle motion in dry conditions. In addition to the existing DRAPE option, which controls release-time behavior, the PRP package now provides a DRY\_TRACKING\_METHOD option which configures how dry particles (particles in dry cells, or above the water table in partially saturated cells) behave at tracking time. This option is relevant only when the Newton formulation is used, in which case dry cells remain active; otherwise, dry cells are inactive and particles will terminate. See the MF6IO document for a detailed explanation of DRY\_TRACKING\_METHOD. \end{itemize} %\underline{INTERNAL FLOW PACKAGES} diff --git a/doc/mf6io/mf6ivar/dfn/prt-prp.dfn b/doc/mf6io/mf6ivar/dfn/prt-prp.dfn index bb5edd48604..76aa68560c8 100644 --- a/doc/mf6io/mf6ivar/dfn/prt-prp.dfn +++ b/doc/mf6io/mf6ivar/dfn/prt-prp.dfn @@ -167,6 +167,15 @@ optional true longname drape description is a text keyword to indicate that if a particle's release point is in a cell that happens to be inactive at release time, the particle is to be moved to the topmost active cell below it, if any. By default, a particle is not released into the simulation if its release point's cell is inactive at release time. +block options +name dry_tracking_method +type string +valid drop stop stay +reader urword +optional true +longname what to do in dry-but-active cells +description is a string indicating how particles should behave in dry-but-active cells (as can occur with the Newton formulation). The value can be ``DROP'', ``STOP'', or ``STAY''. The default is ``DROP'', which passes particles vertically and instantaneously to the water table. ``STOP'' causes particles to terminate. ``STAY'' causes particles to remain stationary but active. + block options name dev_forceternary type keyword diff --git a/doc/mf6io/prt/prp.tex b/doc/mf6io/prt/prp.tex index 473e1c930b7..5b3ac83e6b9 100644 --- a/doc/mf6io/prt/prp.tex +++ b/doc/mf6io/prt/prp.tex @@ -2,6 +2,8 @@ The PRP Package offers multiple ways to specify particle release times. Particle release times may either be provided explicitly, relative to the simulation start time, or configured relative to the time discretization of stress periods via period block settings. When multiple ways of specifying release times are used together, the resulting set of release times is the union of the times specified by each method. +This package contains options to configure how particles behave in dry conditions. Each of these is described in detail below. Their interaction is summarized in the PRT chapter introduction. + \vspace{5mm} \subsubsection{Structure of Blocks} \lstinputlisting[style=blockdefinition]{./mf6ivar/tex/prt-prp-options.dat} diff --git a/doc/mf6io/prt/prt.tex b/doc/mf6io/prt/prt.tex index 196eba092c6..385f45f4260 100644 --- a/doc/mf6io/prt/prt.tex +++ b/doc/mf6io/prt/prt.tex @@ -21,6 +21,42 @@ \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} + +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 an inactive cell or 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. + +Release-time and tracking-time considerations are described (and implemented) separately. + +\subsubsection{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 user-specified location. + +If the release cell is inactive, behavior is determined by the DRAPE option. If the DRAPE option is enabled, the particle will be 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 using the DRY\_TRACKING\_METHOD option discussed below. + +\subsubsection{Tracking} + +With the Newton formulation, particles can be released into dry-but-active cells. + +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. + +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. The PRP package provides an option DRY\_TRACKING\_METHOD determining how dry particles should behave. Supported values are DROP (default), STOP, and STAY. + +If DROP is selected, or if a DRY\_TRACKING\_METHOD is unspecified, a dry particle 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. If the vertical column containing the particle is entirely dry, the particle will terminate upon reaching the bottom of the model 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, b) the particle terminates due to reaching its STOPTIME or STOPTRAVELTIME, or c) the simulation ends. + \subsection{Particle Track Output} The PRT Model supports both binary and CSV particle track output files. A particle track CSV file contains the output data in tabular format. The first line of the CSV file contains column names. Each subsequent line in the file contains a row of data for a single particle track record, with the following fields: diff --git a/src/Model/ModelUtilities/TrackFile.f90 b/src/Model/ModelUtilities/TrackFile.f90 index 4ba94b71e9c..a09be74b7a8 100644 --- a/src/Model/ModelUtilities/TrackFile.f90 +++ b/src/Model/ModelUtilities/TrackFile.f90 @@ -35,7 +35,7 @@ module TrackFileModule !! 1: particle transitioned between cells !! 2: current time step ended**** !! 3: particle terminated - !! 4: particle exited weak sink + !! 4: particle in weak sink !! 5: user-specified tracking time !! !! Each record has an "istatus" property, which is the tracking status; diff --git a/src/Model/ParticleTracking/prt-prp.f90 b/src/Model/ParticleTracking/prt-prp.f90 index fe69f888cec..96ebf145415 100644 --- a/src/Model/ParticleTracking/prt-prp.f90 +++ b/src/Model/ParticleTracking/prt-prp.f90 @@ -49,6 +49,7 @@ module PrtPrpModule integer(I4B), pointer :: istopweaksink => null() !< weak sink option: 0 = no stop, 1 = stop integer(I4B), pointer :: istopzone => null() !< optional stop zone number: 0 = no stop zone integer(I4B), pointer :: idrape => null() !< drape option: 0 = do not drape, 1 = drape to topmost active cell + integer(I4B), pointer :: idrymeth => null() !< dry tracking method: 0 = drop, 1 = stop, 2 = stay integer(I4B), pointer :: itrkout => null() !< binary track file integer(I4B), pointer :: itrkhdr => null() !< track header file integer(I4B), pointer :: itrkcsv => null() !< CSV track file @@ -89,6 +90,7 @@ module PrtPrpModule procedure :: release procedure :: log_release procedure :: validate_release_point + procedure :: initialize_particle procedure, public :: bnd_obs_supported => prp_obs_supported procedure, public :: bnd_df_obs => prp_df_obs end type PrtPrpType @@ -158,6 +160,7 @@ subroutine prp_da(this) call mem_deallocate(this%istopweaksink) call mem_deallocate(this%istopzone) call mem_deallocate(this%idrape) + call mem_deallocate(this%idrymeth) call mem_deallocate(this%nreleasepoints) call mem_deallocate(this%nreleasetimes) call mem_deallocate(this%nparticles) @@ -247,6 +250,7 @@ subroutine prp_allocate_scalars(this) call mem_allocate(this%istopweaksink, 'ISTOPWEAKSINK', this%memoryPath) call mem_allocate(this%istopzone, 'ISTOPZONE', this%memoryPath) call mem_allocate(this%idrape, 'IDRAPE', this%memoryPath) + call mem_allocate(this%idrymeth, 'IDRYMETH', this%memoryPath) call mem_allocate(this%nreleasepoints, 'NRELEASEPOINTS', this%memoryPath) call mem_allocate(this%nreleasetimes, 'NRELEASETIMES', this%memoryPath) call mem_allocate(this%nparticles, 'NPARTICLES', this%memoryPath) @@ -270,6 +274,7 @@ subroutine prp_allocate_scalars(this) this%istopweaksink = 0 this%istopzone = 0 this%idrape = 0 + this%idrymeth = 0 this%nreleasepoints = 0 this%nreleasetimes = 0 this%nparticles = 0 @@ -413,47 +418,52 @@ subroutine release(this, ip, trelease) integer(I4B), intent(in) :: ip !< particle index real(DP), intent(in) :: trelease !< release time ! local - integer(I4B) :: irow, icol, ilay, icpl - integer(I4B) :: ic, icu integer(I4B) :: np - real(DP) :: x, y, z - real(DP) :: top, bot, hds type(ParticleType), pointer :: particle + call this%initialize_particle(particle, ip, trelease) + ! Increment cumulative particle count np = this%nparticles + 1 this%nparticles = np - ! Get reduced and user node numbers - ic = this%rptnode(ip) - icu = this%dis%get_nodeuser(ic) + ! Save the particle to the store + call this%particles%save_particle(particle, np) + deallocate (particle) - ! Load coordinates and transform if needed - x = this%rptx(ip) - y = this%rpty(ip) - if (this%ilocalz > 0) then - top = this%fmi%dis%top(ic) - bot = this%fmi%dis%bot(ic) - hds = this%fmi%gwfhead(ic) - z = bot + this%rptz(ip) * (hds - bot) - else - z = this%rptz(ip) - end if + ! Accumulate mass for release point + this%rptm(ip) = this%rptm(ip) + DONE - ! Make sure release point is in the grid/cell - call this%validate_release_point(ic, x, y, z) + end subroutine release + + subroutine initialize_particle(this, particle, ip, trelease) + class(PrtPrpType), intent(inout) :: this !< this instance + type(ParticleType), pointer, intent(inout) :: particle !< the particle + integer(I4B), intent(in) :: ip !< particle index + real(DP), intent(in) :: trelease !< release time + ! local + integer(I4B) :: irow, icol, ilay, icpl + integer(I4B) :: ic, icu, ic_old + real(DP) :: x, y, z + real(DP) :: top, bot, hds + + ic = this%rptnode(ip) + icu = this%dis%get_nodeuser(ic) - ! Initialize the particle call create_particle(particle) + if (size(this%boundname) /= 0) then particle%name = this%boundname(ip) else particle%name = '' end if + particle%irpt = ip particle%istopweaksink = this%istopweaksink particle%istopzone = this%istopzone + particle%idrymeth = this%idrymeth particle%icu = icu + select type (dis => this%dis) type is (DisType) call get_ijk(icu, dis%nrow, dis%ncol, dis%nlay, irow, icol, ilay) @@ -462,28 +472,49 @@ subroutine release(this, ip, trelease) end select particle%ilay = ilay particle%izone = this%rptzone(ic) + particle%istatus = 0 - ! Handle inactive cells + ! If the cell is inactive, either drape the particle + ! to the top-most active cell beneath it if drape is + ! enabled, or else terminate permanently unreleased. if (this%ibound(ic) == 0) then - ! If drape option activated, release in highest active - ! cell vertically below release point. - if (this%idrape /= 0) & + ic_old = ic + if (this%idrape > 0) then call this%dis%highest_active(ic, this%ibound) - ! If returned cell is inactive, do not release particle - if (this%ibound(ic) == 0) & - particle%istatus = 8 ! permanently unreleased + if (ic == ic_old .or. this%ibound(ic) == 0) & + particle%istatus = 8 + else + particle%istatus = 8 + end if + end if + + ! Load coordinates and transform if needed + x = this%rptx(ip) + y = this%rpty(ip) + if (this%ilocalz > 0) then + top = this%fmi%dis%top(ic) + bot = this%fmi%dis%bot(ic) + hds = this%fmi%gwfhead(ic) + z = bot + this%rptz(ip) * (hds - bot) + else + z = this%rptz(ip) end if + + call this%validate_release_point(ic, x, y, z) + particle%x = x particle%y = y particle%z = z particle%trelease = trelease - ! Set stopping time to earlier of times specified by STOPTIME and STOPTRAVELTIME + + ! Set stop time to earlier of STOPTIME and STOPTRAVELTIME if (this%stoptraveltime == huge(1d0)) then particle%tstop = this%stoptime else particle%tstop = particle%trelease + this%stoptraveltime if (this%stoptime < particle%tstop) particle%tstop = this%stoptime end if + particle%ttrack = particle%trelease particle%idomain(1) = 0 particle%iboundary(1) = 0 @@ -495,15 +526,7 @@ subroutine release(this, ip, trelease) particle%iexmeth = this%iexmeth particle%iextend = this%iextend particle%extol = this%extol - - ! Store the particle's state and deallocate it - call this%particles%save_particle(particle, np) - deallocate (particle) - - ! Accumulate particle mass for release point - this%rptm(ip) = this%rptm(ip) + DONE - - end subroutine release + end subroutine initialize_particle !> @ brief Read and prepare period data for particle input subroutine prp_rp(this) @@ -691,6 +714,25 @@ subroutine prp_options(this, option, found) case ('DRAPE') this%idrape = 1 found = .true. + case ('DRY_TRACKING_METHOD') + call this%parser%GetStringCaps(keyword) + select case (keyword) + case ('DROP') + this%idrymeth = 0 + case ('STOP') + this%idrymeth = 1 + case ('STAY') + this%idrymeth = 2 + case default + write (errmsg, '(a, a)') & + 'Unknown dry tracking method: ', trim(keyword) + call store_error(errmsg) + write (errmsg, '(a, a)') & + 'DRY must be "DROP", "STOP" or "STAY"' + call store_error(errmsg) + call this%parser%StoreErrorUnit() + end select + found = .true. case ('TRACK') call this%parser%GetStringCaps(keyword) if (keyword == 'FILEOUT') then @@ -891,7 +933,7 @@ subroutine prp_read_packagedata(this) if (n < 1 .or. n > this%nreleasepoints) then write (errmsg, '(a,1x,i0,a)') & - 'Release point number must be greater than 0 and less than ', & + 'Release point number must be greater than 0 and less than', & 'or equal to', this%nreleasepoints, '.' call store_error(errmsg) cycle diff --git a/src/Model/ParticleTracking/prt.f90 b/src/Model/ParticleTracking/prt.f90 index e47f8ff6a0b..0730d8f6b02 100644 --- a/src/Model/ParticleTracking/prt.f90 +++ b/src/Model/ParticleTracking/prt.f90 @@ -932,7 +932,7 @@ subroutine prt_solve(this) ! -- If particle is permanently unreleased, record its initial/terminal state if (particle%istatus == 8) & - call this%method%save(particle, reason=3) ! reason=3: termination + call this%method%save(particle, reason=3) ! If particle is inactive or not yet to be released, cycle if (particle%istatus > 1) cycle @@ -940,7 +940,7 @@ subroutine prt_solve(this) ! If particle released this time step, record its initial state particle%istatus = 1 if (particle%trelease >= totimc) & - call this%method%save(particle, reason=0) ! reason=0: release + call this%method%save(particle, reason=0) ! Maximum time is the end of the time step or the particle ! stop time, whichever comes first, unless it's the final @@ -957,6 +957,10 @@ subroutine prt_solve(this) ! Get and apply the tracking method call this%method%apply(particle, tmax) + ! Reset previous cell, exit face, and zone numbers + particle%icp = 0 + particle%izp = 0 + ! Update particle storage call packobj%particles%save_particle(particle, np) end do diff --git a/src/Solution/ParticleTracker/CellDefn.f90 b/src/Solution/ParticleTracker/CellDefn.f90 index 6075a38a399..836f0421633 100644 --- a/src/Solution/ParticleTracker/CellDefn.f90 +++ b/src/Solution/ParticleTracker/CellDefn.f90 @@ -1,5 +1,7 @@ module CellDefnModule use KindModule, only: DP, I4B, LGP + use ConstantsModule, only: DZERO + use MathUtilModule, only: is_close implicit none private @@ -15,6 +17,7 @@ module CellDefnModule integer(I4B), public :: npolyverts !< number of vertices for cell polygon real(DP), public :: porosity !< cell porosity real(DP), public :: retfactor !< cell retardation factor + integer(I4B), public :: ilay !< layer number integer(I4B), public :: izone !< cell zone number integer(I4B), public :: iweaksink !< weak sink indicator integer(I4B), public :: inoexitface !< no exit face indicator diff --git a/src/Solution/ParticleTracker/Method.f90 b/src/Solution/ParticleTracker/Method.f90 index 38049b5d51a..061a627e1ad 100644 --- a/src/Solution/ParticleTracker/Method.f90 +++ b/src/Solution/ParticleTracker/Method.f90 @@ -2,6 +2,7 @@ module MethodModule use KindModule, only: DP, I4B, LGP + use ConstantsModule, only: DZERO use ErrorUtilModule, only: pstop use SubcellModule, only: SubcellType use ParticleModule @@ -11,6 +12,7 @@ module MethodModule use CellDefnModule, only: CellDefnType use TrackControlModule, only: TrackControlType use TimeSelectModule, only: TimeSelectType + use MathUtilModule, only: is_close implicit none private @@ -27,7 +29,7 @@ module MethodModule !! depending on cell geometry (implementing the strategy pattern). !< type, abstract :: MethodType - character(len=40), pointer, public :: type !< method name + character(len=40), pointer, public :: name !< method name logical(LGP), public :: delegates !< whether the method delegates type(PrtFmiType), pointer, public :: fmi => null() !< ptr to fmi class(CellType), pointer, public :: cell => null() !< ptr to the current cell @@ -50,7 +52,7 @@ module MethodModule procedure :: save procedure :: track procedure :: try_pass - procedure :: update + procedure :: check end type MethodType abstract interface @@ -94,7 +96,8 @@ subroutine init(this, fmi, cell, subcell, trackctl, tracktimes, & if (present(retfactor)) this%retfactor => retfactor end subroutine init - !> @brief Track particle through subdomains + !> @brief Track the particle over domains of the given + ! level until the particle terminates or tmax elapses. recursive subroutine track(this, particle, level, tmax) ! dummy class(MethodType), intent(inout) :: this @@ -106,6 +109,7 @@ recursive subroutine track(this, particle, level, tmax) integer(I4B) :: nextlevel class(methodType), pointer :: submethod + ! Advance the particle over subdomains advancing = .true. nextlevel = level + 1 do while (advancing) @@ -115,28 +119,28 @@ recursive subroutine track(this, particle, level, tmax) end do end subroutine track - !> @brief Try passing the particle to the next subdomain + !> @brief Try passing the particle to the next subdomain. subroutine try_pass(this, particle, nextlevel, advancing) class(MethodType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle integer(I4B) :: nextlevel logical(LGP) :: advancing - ! tracking submethod marked tracking complete? - ! reset domain boundary flag and don't advance + ! if the particle is done advancing, reset the domain boundary flag. if (.not. particle%advancing) then particle%iboundary = 0 advancing = .false. else - ! otherwise pass particle to next subdomain - ! and if it's on a boundary, stop advancing + ! otherwise pass the particle to the next subdomain. + ! if that leaves it on a boundary, stop advancing. call this%pass(particle) - if (particle%iboundary(nextlevel - 1) .ne. 0) & + if (particle%iboundary(nextlevel - 1) .ne. 0) then advancing = .false. + end if end if end subroutine try_pass - !> @brief Load subdomain tracking method (submethod) + !> @brief Load the subdomain tracking method (submethod). subroutine load(this, particle, next_level, submethod) class(MethodType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle @@ -145,14 +149,14 @@ subroutine load(this, particle, next_level, submethod) call pstop(1, "load must be overridden") end subroutine load - !> @brief Pass a particle to the next subdomain, internal use only + !> @brief Pass the particle to the next subdomain. subroutine pass(this, particle) class(MethodType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle call pstop(1, "pass must be overridden") end subroutine pass - !> @brief Save a particle's current state. + !> @brief Save the particle's state to output files. subroutine save(this, particle, reason) use TdisModule, only: kper, kstp, totimc ! dummy @@ -179,50 +183,106 @@ subroutine save(this, particle, reason) end if end if - ! Save the particle's state to any registered tracking output files - call this%trackctl%save(particle, kper=per, & - kstp=stp, reason=reason) + call this%trackctl%save(particle, kper=per, kstp=stp, reason=reason) end subroutine save - !> @brief Update particle state and check termination conditions + !> @brief Check reporting/terminating conditions before tracking. !! - !! Update the particle's properties (e.g. advancing flag, zone number, - !! status). If any termination conditions apply, the particle's status - !! will be set to the appropriate termination value. If any reporting - !! conditions apply, save particle state with the proper reason code. + !! Check a number of conditions determining whether to continue + !! tracking the particle or terminate it, as well as whether to + !! record any output data as per selected reporting conditions. !< - subroutine update(this, particle, cell_defn) + subroutine check(this, particle, cell_defn) + ! modules + use TdisModule, only: endofsimulation, totim ! dummy class(MethodType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle type(CellDefnType), pointer, intent(inout) :: cell_defn + ! local + logical(LGP) :: dry_cell, dry_particle, no_exit_face, stop_zone, weak_sink + + dry_cell = this%fmi%ibdgwfsat0(cell_defn%icell) == 0 + dry_particle = particle%z > cell_defn%top + no_exit_face = cell_defn%inoexitface > 0 + stop_zone = cell_defn%izone > 0 .and. particle%istopzone == cell_defn%izone + weak_sink = cell_defn%iweaksink > 0 particle%izone = cell_defn%izone - if (cell_defn%izone .ne. 0) then - if (particle%istopzone .eq. cell_defn%izone) then - particle%advancing = .false. - particle%istatus = 6 - call this%save(particle, reason=3) ! particle terminated - return - end if + if (stop_zone) then + particle%advancing = .false. + particle%istatus = 6 + call this%save(particle, reason=3) + return end if - if (cell_defn%inoexitface .ne. 0) then + + if (no_exit_face .and. .not. dry_cell) then particle%advancing = .false. particle%istatus = 5 - call this%save(particle, reason=3) ! particle terminated + call this%save(particle, reason=3) return end if - if (cell_defn%iweaksink .ne. 0) then - if (particle%istopweaksink .ne. 0) then + + if (weak_sink) then + if (particle%istopweaksink > 0) then particle%advancing = .false. particle%istatus = 3 - call this%save(particle, reason=3) ! particle terminated + call this%save(particle, reason=3) return else - call this%save(particle, reason=4) ! particle exited weak sink + call this%save(particle, reason=4) + end if + end if + + if (dry_cell) then + if (particle%idrymeth == 0) then + no_exit_face = .false. + else if (particle%idrymeth == 1) then + ! stop + particle%advancing = .false. + particle%istatus = 7 + call this%save(particle, reason=3) + return + else if (particle%idrymeth == 2) then + ! stay + no_exit_face = .false. + particle%advancing = .false. + particle%ttrack = totim + ! terminate if last period/step + if (endofsimulation) then + particle%istatus = 5 + call this%save(particle, reason=3) + return + end if + call this%save(particle, reason=2) + end if + else if (dry_particle .and. this%name /= "passtobottom") then + ! dry particle + if (particle%idrymeth == 0) then + ! drop to water table + particle%z = cell_defn%top + call this%save(particle, reason=1) + else if (particle%idrymeth == 1) then + ! terminate + particle%advancing = .false. + particle%istatus = 7 + call this%save(particle, reason=3) + return + else if (particle%idrymeth == 2) then + ! stay + no_exit_face = .false. + particle%advancing = .false. return end if end if - end subroutine update + + if (no_exit_face) then + particle%advancing = .false. + particle%istatus = 5 + call this%save(particle, reason=3) + return + end if + + end subroutine check end module MethodModule diff --git a/src/Solution/ParticleTracker/MethodCellPassToBot.f90 b/src/Solution/ParticleTracker/MethodCellPassToBot.f90 index 33825b5b7b6..433b3ca9650 100644 --- a/src/Solution/ParticleTracker/MethodCellPassToBot.f90 +++ b/src/Solution/ParticleTracker/MethodCellPassToBot.f90 @@ -28,15 +28,15 @@ module MethodCellPassToBotModule subroutine create_method_cell_ptb(method) type(MethodCellPassToBotType), pointer :: method allocate (method) - allocate (method%type) - method%type = "passtobottom" + allocate (method%name) + method%name = "passtobottom" method%delegates = .false. end subroutine create_method_cell_ptb !> @brief Deallocate the pass-to-bottom tracking method subroutine deallocate (this) class(MethodCellPassToBotType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine deallocate !> @brief Pass particle vertically and instantaneously to the cell bottom @@ -46,10 +46,15 @@ subroutine apply_ptb(this, particle, tmax) type(ParticleType), pointer, intent(inout) :: particle real(DP), intent(in) :: tmax - call this%update(particle, this%cell%defn) + ! Check termination/reporting conditions + call this%check(particle, this%cell%defn) if (.not. particle%advancing) return + + ! Pass to bottom face particle%z = this%cell%defn%bot particle%iboundary(2) = this%cell%defn%npolyverts + 2 + + ! Save datum call this%save(particle, reason=1) end subroutine apply_ptb diff --git a/src/Solution/ParticleTracker/MethodCellPollock.f90 b/src/Solution/ParticleTracker/MethodCellPollock.f90 index 71fc327fef7..86b1d09f4ca 100644 --- a/src/Solution/ParticleTracker/MethodCellPollock.f90 +++ b/src/Solution/ParticleTracker/MethodCellPollock.f90 @@ -36,7 +36,7 @@ subroutine create_method_cell_pollock(method) allocate (method) call create_cell_rect(cell) method%cell => cell - method%type => method%cell%type + method%name => method%cell%type method%delegates = .true. call create_subcell_rect(subcell) method%subcell => subcell @@ -45,7 +45,7 @@ end subroutine create_method_cell_pollock !> @brief Destroy the tracking method subroutine destroy_mcp(this) class(MethodCellPollockType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine destroy_mcp !> @brief Load subcell tracking method @@ -129,19 +129,12 @@ subroutine apply_mcp(this, particle, tmax) select type (cell => this%cell) type is (CellRectType) - ! Update particle state, return early if done advancing - call this%update(particle, cell%defn) + ! Check termination/reporting conditions + call this%check(particle, cell%defn) if (.not. particle%advancing) return - ! If the particle is above the top of the cell (presumed water table) - ! pass it vertically and instantaneously to the top - if (particle%z > cell%defn%top) then - particle%z = cell%defn%top - call this%save(particle, reason=1) - end if - - ! Transform particle location into local cell coordinates - ! (translated and rotated but not scaled relative to model). + ! Transform model coordinates to local cell coordinates + ! (translated/rotated but not scaled relative to model) xOrigin = cell%xOrigin yOrigin = cell%yOrigin zOrigin = cell%zOrigin @@ -150,14 +143,13 @@ subroutine apply_mcp(this, particle, tmax) call particle%transform(xOrigin, yOrigin, zOrigin, & sinrot, cosrot) - ! Track the particle across the cell. + ! Track the particle over the cell call this%track(particle, 2, tmax) - ! Transform particle location back to model coordinates, then - ! reset transformation and eliminate accumulated roundoff error. + ! Transform cell coordinates back to model coordinates call particle%transform(xOrigin, yOrigin, zOrigin, & sinrot, cosrot, invert=.true.) - call particle%transform(reset=.true.) + call particle%reset_transform() end select end subroutine apply_mcp diff --git a/src/Solution/ParticleTracker/MethodCellPollockQuad.f90 b/src/Solution/ParticleTracker/MethodCellPollockQuad.f90 index 8fe7e41525f..c5f61f2c221 100644 --- a/src/Solution/ParticleTracker/MethodCellPollockQuad.f90 +++ b/src/Solution/ParticleTracker/MethodCellPollockQuad.f90 @@ -37,7 +37,7 @@ subroutine create_method_cell_quad(method) allocate (method) call create_cell_rect_quad(cell) method%cell => cell - method%type => method%cell%type + method%name => method%cell%type method%delegates = .true. call create_subcell_rect(subcell) method%subcell => subcell @@ -46,7 +46,7 @@ end subroutine create_method_cell_quad !> @brief Deallocate the Pollock quad-refined cell method subroutine deallocate (this) class(MethodCellPollockQuadType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine deallocate !> @brief Load subcell into tracking method @@ -208,19 +208,12 @@ subroutine apply_mcpq(this, particle, tmax) select type (cell => this%cell) type is (CellRectQuadType) - ! Update particle state, return early if done advancing - call this%update(particle, cell%defn) + ! Check termination/reporting conditions + call this%check(particle, cell%defn) if (.not. particle%advancing) return - ! If the particle is above the top of the cell (presumed water table) - ! pass it vertically and instantaneously to the top - if (particle%z > cell%defn%top) then - particle%z = cell%defn%top - call this%save(particle, reason=1) ! reason=1: cell transition - end if - - ! Transform particle location into local cell coordinates - ! (translated and rotated but not scaled relative to model). + ! Transform model coordinates to local cell coordinates + ! (translated/rotated but not scaled relative to model) xOrigin = cell%xOrigin yOrigin = cell%yOrigin zOrigin = cell%zOrigin @@ -229,14 +222,13 @@ subroutine apply_mcpq(this, particle, tmax) call particle%transform(xOrigin, yOrigin, zOrigin, & sinrot, cosrot) - ! Track the particle across the cell. + ! Track the particle over the cell call this%track(particle, 2, tmax) - ! Transform particle location back to model coordinates, then - ! reset transformation and eliminate accumulated roundoff error. + ! Transform cell coordinates back to model coordinates call particle%transform(xOrigin, yOrigin, zOrigin, & sinrot, cosrot, invert=.true.) - call particle%transform(reset=.true.) + call particle%reset_transform() end select end subroutine apply_mcpq @@ -299,7 +291,7 @@ subroutine load_subcell(this, particle, subcell) qintl2 = cell%qintl(isc + 1) qextl1 = cell%qextl1(isc) qextl2 = cell%qextl2(isc) - ! + subcell%dx = dx subcell%dy = dy subcell%dz = dz diff --git a/src/Solution/ParticleTracker/MethodCellTernary.f90 b/src/Solution/ParticleTracker/MethodCellTernary.f90 index fac00b00845..a04a65d231f 100644 --- a/src/Solution/ParticleTracker/MethodCellTernary.f90 +++ b/src/Solution/ParticleTracker/MethodCellTernary.f90 @@ -62,7 +62,7 @@ subroutine create_method_cell_ternary(method) allocate (method) call create_cell_poly(cell) method%cell => cell - method%type => method%cell%type + method%name => method%cell%type method%delegates = .true. call create_subcell_tri(subcell) method%subcell => subcell @@ -71,7 +71,7 @@ end subroutine create_method_cell_ternary !> @brief Destroy the tracking method subroutine destroy_mct(this) class(MethodCellTernaryType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine destroy_mct !> @brief Load subcell into tracking method @@ -156,22 +156,16 @@ subroutine apply_mct(this, particle, tmax) ! local integer(I4B) :: i - ! Update particle state, return early if done advancing - call this%update(particle, this%cell%defn) - if (.not. particle%advancing) return - - ! If the particle is above the top of the cell (presumed water table) - ! pass it vertically and instantaneously to the top - if (particle%z > this%cell%defn%top) then - particle%z = this%cell%defn%top - call this%save(particle, reason=1) - end if - ! (Re)allocate type-bound arrays select type (cell => this%cell) type is (CellPolyType) + ! Check termination/reporting conditions + call this%check(particle, this%cell%defn) + if (.not. particle%advancing) return + ! Number of vertices this%nverts = cell%defn%npolyverts + ! (Re)allocate type-bound arrays if (allocated(this%xvert)) then deallocate (this%xvert) @@ -185,6 +179,7 @@ subroutine apply_mct(this, particle, tmax) deallocate (this%xvertnext) deallocate (this%yvertnext) end if + allocate (this%xvert(this%nverts)) allocate (this%yvert(this%nverts)) allocate (this%vne(this%nverts)) @@ -195,15 +190,18 @@ subroutine apply_mct(this, particle, tmax) allocate (this%iprev(this%nverts)) allocate (this%xvertnext(this%nverts)) allocate (this%yvertnext(this%nverts)) + ! Cell vertices do i = 1, this%nverts this%xvert(i) = cell%defn%polyvert(1, i) this%yvert(i) = cell%defn%polyvert(2, i) end do + ! Top, bottom, and thickness this%ztop = cell%defn%top this%zbot = cell%defn%bot this%dz = this%ztop - this%zbot + ! Shifted arrays do i = 1, this%nverts this%iprev(i) = i diff --git a/src/Solution/ParticleTracker/MethodDis.f90 b/src/Solution/ParticleTracker/MethodDis.f90 index ad427c9e411..6167837dc19 100644 --- a/src/Solution/ParticleTracker/MethodDis.f90 +++ b/src/Solution/ParticleTracker/MethodDis.f90 @@ -1,6 +1,6 @@ module MethodDisModule - use KindModule, only: DP, I4B + use KindModule, only: DP, I4B, LGP use ConstantsModule, only: DONE, DZERO use MethodModule, only: MethodType use MethodCellPoolModule @@ -11,6 +11,7 @@ module MethodDisModule use PrtFmiModule, only: PrtFmiType use DisModule, only: DisType use GeomUtilModule, only: get_ijk, get_jk + use MathUtilModule, only: is_close implicit none private @@ -40,23 +41,23 @@ module MethodDisModule !> @brief Create a new structured grid (DIS) tracking method subroutine create_method_dis(method) - ! -- dummy + ! dummy type(MethodDisType), pointer :: method - ! -- local + ! local type(CellRectType), pointer :: cell allocate (method) - allocate (method%type) + allocate (method%name) call create_cell_rect(cell) method%cell => cell - method%type = "dis" + method%name = "dis" method%delegates = .true. end subroutine create_method_dis !> @brief Destructor the tracking method subroutine deallocate (this) class(MethodDisType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine deallocate subroutine load_cell(this, ic, cell) @@ -81,11 +82,14 @@ subroutine load_cell(this, ic, cell) select type (dis => this%fmi%dis) type is (DisType) icu = dis%get_nodeuser(ic) + call get_ijk(icu, dis%nrow, dis%ncol, dis%nlay, & irow, jcol, klay) + dx = dis%delr(jcol) dy = dis%delc(irow) dz = cell%defn%top - cell%defn%bot + cell%dx = dx cell%dy = dy cell%dz = dz @@ -95,41 +99,54 @@ subroutine load_cell(this, ic, cell) cell%yOrigin = cell%defn%polyvert(2, 1) cell%zOrigin = cell%defn%bot cell%ipvOrigin = 1 - areax = dy * dz - areay = dx * dz - areaz = dx * dy + factor = DONE / cell%defn%retfactor factor = factor / cell%defn%porosity + + areaz = dx * dy + term = factor / areaz + + cell%vz1 = cell%defn%faceflow(6) * term + cell%vz2 = -cell%defn%faceflow(7) * term + + if (this%fmi%ibdgwfsat0(ic) == 0) then + cell%vx1 = DZERO + cell%vx2 = DZERO + cell%vy1 = DZERO + cell%vy2 = DZERO + cell%vz1 = DZERO + cell%vz2 = DZERO + return + end if + + areax = dy * dz term = factor / areax cell%vx1 = cell%defn%faceflow(1) * term cell%vx2 = -cell%defn%faceflow(3) * term + + areay = dx * dz term = factor / areay cell%vy1 = cell%defn%faceflow(4) * term cell%vy2 = -cell%defn%faceflow(2) * term - term = factor / areaz - cell%vz1 = cell%defn%faceflow(6) * term - cell%vz2 = -cell%defn%faceflow(7) * term + end select end subroutine load_cell !> @brief Load the cell geometry and tracking method subroutine load_dis(this, particle, next_level, submethod) - ! -- dummy + ! dummy class(MethodDisType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle integer(I4B), intent(in) :: next_level class(MethodType), pointer, intent(inout) :: submethod - ! -- local + ! local integer(I4B) :: ic select type (cell => this%cell) type is (CellRectType) - ! -- Load cell definition and geometry ic = particle%idomain(next_level) call this%load_celldefn(ic, cell%defn) call this%load_cell(ic, cell) - - ! -- If cell is active but dry, Initialize instant pass-to-bottom method if (this%fmi%ibdgwfsat0(ic) == 0) then call method_cell_ptb%init( & fmi=this%fmi, & @@ -138,7 +155,6 @@ subroutine load_dis(this, particle, next_level, submethod) tracktimes=this%tracktimes) submethod => method_cell_ptb else - ! -- Otherwise initialize Pollock's method call method_cell_plck%init( & fmi=this%fmi, & cell=this%cell, & @@ -176,7 +192,7 @@ subroutine load_particle(this, cell, particle) select type (dis => this%fmi%dis) type is (DisType) - ! compute and set reduced/user node numbers and layer + ! compute reduced/user node numbers and layer inface = particle%iboundary(2) inbr = cell%defn%facenbr(inface) idiag = this%fmi%dis%con%ia(cell%defn%icell) @@ -185,6 +201,25 @@ subroutine load_particle(this, cell, particle) icu = dis%get_nodeuser(ic) call get_ijk(icu, dis%nrow, dis%ncol, dis%nlay, & irow, icol, ilay) + + ! if returning to a cell through the bottom + ! face after previously leaving it through + ! that same face, we've entered a cycle + ! as can occur e.g. in wells. terminate + ! in the previous cell. + if (ic == particle%icp .and. inface == 7 .and. ilay < particle%ilay) then + particle%advancing = .false. + particle%idomain(2) = particle%icp + particle%istatus = 2 + particle%izone = particle%izp + call this%save(particle, reason=3) + return + else + particle%icp = particle%idomain(2) + particle%izp = particle%izone + end if + + ! update node numbers and layer particle%idomain(2) = ic particle%icu = icu particle%ilay = ilay @@ -218,7 +253,6 @@ subroutine load_particle(this, cell, particle) end if particle%z = z end select - end subroutine load_particle !> @brief Update cell-cell flows of particle mass. @@ -239,13 +273,12 @@ subroutine update_flowja(this, cell, particle) idiag = this%fmi%dis%con%ia(cell%defn%icell) ipos = idiag + inbr - ! -- leaving old cell + ! leaving old cell this%flowja(ipos) = this%flowja(ipos) - DONE - ! -- entering new cell + ! entering new cell this%flowja(this%fmi%dis%con%isym(ipos)) & = this%flowja(this%fmi%dis%con%isym(ipos)) + DONE - end subroutine update_flowja !> @brief Pass a particle to the next cell, if there is one @@ -267,15 +300,17 @@ subroutine pass_dis(this, particle) particle%advancing = .false. call this%save(particle, reason=3) else - ! Otherwise, load cell properties into the - ! particle and update intercell mass flows + ! Update old to new cell properties call this%load_particle(cell, particle) + if (.not. particle%advancing) return + + ! Update intercell mass flows call this%update_flowja(cell, particle) end if end select end subroutine pass_dis - !> @brief Apply the method to a particle + !> @brief Apply the structured tracking method to a particle. subroutine apply_dis(this, particle, tmax) class(MethodDisType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle @@ -299,36 +334,35 @@ end function get_top !> @brief Loads cell definition from the grid subroutine load_celldefn(this, ic, defn) - ! -- dummy + ! dummy class(MethodDisType), intent(inout) :: this integer(I4B), intent(in) :: ic type(CellDefnType), pointer, intent(inout) :: defn - ! -- Load basic cell properties + ! Load basic cell properties call this%load_properties(ic, defn) - ! -- Load cell polygon vertices + ! Load cell polygon vertices call this%fmi%dis%get_polyverts( & defn%icell, & defn%polyvert, & closed=.true.) - - ! -- Load cell face neighbors call this%load_neighbors(defn) - ! -- Load 180 degree face indicators + ! Load 180 degree face indicators defn%ispv180(1:defn%npolyverts + 1) = .false. - ! -- Load face flows (assumes face neighbors already loaded) call this%load_flows(defn) end subroutine load_celldefn subroutine load_properties(this, ic, defn) - ! -- dummy + ! dummy class(MethodDisType), intent(inout) :: this integer(I4B), intent(in) :: ic type(CellDefnType), pointer, intent(inout) :: defn + ! local + integer(I4B) :: irow, icol, ilay, icu defn%icell = ic defn%npolyverts = 4 ! rectangular cell always has 4 vertices @@ -341,19 +375,24 @@ subroutine load_properties(this, ic, defn) defn%sat = this%fmi%gwfsat(ic) defn%porosity = this%porosity(ic) defn%retfactor = this%retfactor(ic) + select type (dis => this%fmi%dis) + type is (DisType) + icu = dis%get_nodeuser(ic) + call get_ijk(icu, dis%nrow, dis%ncol, dis%nlay, irow, icol, ilay) + defn%ilay = ilay + end select defn%izone = this%izone(ic) defn%can_be_rect = .true. defn%can_be_quad = .false. - end subroutine load_properties !> @brief Loads face neighbors to cell definition from the grid. !! Assumes cell index and number of vertices are already loaded. subroutine load_neighbors(this, defn) - ! -- dummy + ! dummy class(MethodDisType), intent(inout) :: this type(CellDefnType), pointer, intent(inout) :: defn - ! -- local + ! local integer(I4B) :: ic1 integer(I4B) :: ic2 integer(I4B) :: icu1 @@ -371,7 +410,7 @@ subroutine load_neighbors(this, defn) select type (dis => this%fmi%dis) type is (DisType) - ! -- Load face neighbors + ! Load face neighbors defn%facenbr = 0 ic1 = defn%icell icu1 = dis%get_nodeuser(ic1) @@ -387,7 +426,7 @@ subroutine load_neighbors(this, defn) call get_ijk(icu2, dis%nrow, dis%ncol, dis%nlay, & irow2, jcol2, klay2) if (klay2 == klay1) then - ! -- Edge (polygon) face neighbor + ! Edge (polygon) face neighbor if (irow2 > irow1) then ! Neighbor to the S iedgeface = 4 @@ -403,15 +442,15 @@ subroutine load_neighbors(this, defn) end if defn%facenbr(iedgeface) = int(iloc, 1) else if (klay2 > klay1) then - ! -- Bottom face neighbor + ! Bottom face neighbor defn%facenbr(defn%npolyverts + 2) = int(iloc, 1) else - ! -- Top face neighbor + ! Top face neighbor defn%facenbr(defn%npolyverts + 3) = int(iloc, 1) end if end do end select - ! -- List of edge (polygon) faces wraps around + ! List of edge (polygon) faces wraps around defn%facenbr(defn%npolyverts + 1) = defn%facenbr(1) end subroutine load_neighbors @@ -442,7 +481,6 @@ subroutine load_flows(this, defn) else defn%iweaksink = 0 end if - end subroutine load_flows subroutine load_face_flows_to_defn(this, defn) @@ -467,10 +505,10 @@ end subroutine load_face_flows_to_defn !> @brief Add boundary flows to the cell definition faceflow array. !! Assumes cell index and number of vertices are already loaded. subroutine load_boundary_flows_to_defn(this, defn) - ! -- dummy + ! dummy class(MethodDisType), intent(inout) :: this type(CellDefnType), intent(inout) :: defn - ! -- local + ! local integer(I4B) :: ioffset ioffset = (defn%icell - 1) * MAX_POLY_CELLS diff --git a/src/Solution/ParticleTracker/MethodDisv.f90 b/src/Solution/ParticleTracker/MethodDisv.f90 index 4303594af9f..de65bbbf134 100644 --- a/src/Solution/ParticleTracker/MethodDisv.f90 +++ b/src/Solution/ParticleTracker/MethodDisv.f90 @@ -46,16 +46,16 @@ module MethodDisvModule !> @brief Create a new vertex grid (DISV) tracking method subroutine create_method_disv(method) - ! -- dummy + ! dummy type(MethodDisvType), pointer :: method - ! -- local + ! local type(CellPolyType), pointer :: cell allocate (method) - allocate (method%type) + allocate (method%name) call create_cell_poly(cell) method%cell => cell - method%type = "disv" + method%name = "disv" method%delegates = .true. call create_defn(method%neighbor) end subroutine create_method_disv @@ -63,7 +63,7 @@ end subroutine create_method_disv !> @brief Destroy the tracking method subroutine deallocate (this) class(MethodDisvType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine deallocate !> @brief Load the cell and the tracking method @@ -72,12 +72,12 @@ subroutine load_disv(this, particle, next_level, submethod) use CellRectModule use CellRectQuadModule use CellUtilModule - ! -- dummy + ! dummy class(MethodDisvType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle integer(I4B), intent(in) :: next_level class(MethodType), pointer, intent(inout) :: submethod - ! -- local + ! local integer(I4B) :: ic class(CellType), pointer :: base type(CellRectType), pointer :: rect @@ -85,53 +85,47 @@ subroutine load_disv(this, particle, next_level, submethod) select type (cell => this%cell) type is (CellPolyType) - ! load cell definition ic = particle%idomain(next_level) call this%load_cell_defn(ic, cell%defn) if (this%fmi%ibdgwfsat0(ic) == 0) then - ! -- Cell is active but dry, so select and initialize pass-to-bottom - ! -- cell method and set cell method pointer call method_cell_ptb%init( & fmi=this%fmi, & cell=this%cell, & trackctl=this%trackctl, & tracktimes=this%tracktimes) submethod => method_cell_ptb + else if (particle%ifrctrn > 0) then + call method_cell_tern%init( & + fmi=this%fmi, & + cell=this%cell, & + trackctl=this%trackctl, & + tracktimes=this%tracktimes) + submethod => method_cell_tern + else if (cell%defn%can_be_rect) then + call cell_poly_to_rect(cell, rect) + base => rect + call method_cell_plck%init( & + fmi=this%fmi, & + cell=base, & + trackctl=this%trackctl, & + tracktimes=this%tracktimes) + submethod => method_cell_plck + else if (cell%defn%can_be_quad) then + call cell_poly_to_quad(cell, quad) + base => quad + call method_cell_quad%init( & + fmi=this%fmi, & + cell=base, & + trackctl=this%trackctl, & + tracktimes=this%tracktimes) + submethod => method_cell_quad else - ! -- Select and initialize cell method and set cell method pointer - if (particle%ifrctrn > 0) then - call method_cell_tern%init( & - fmi=this%fmi, & - cell=this%cell, & - trackctl=this%trackctl, & - tracktimes=this%tracktimes) - submethod => method_cell_tern - else if (cell%defn%can_be_rect) then - call cell_poly_to_rect(cell, rect) - base => rect - call method_cell_plck%init( & - fmi=this%fmi, & - cell=base, & - trackctl=this%trackctl, & - tracktimes=this%tracktimes) - submethod => method_cell_plck - else if (cell%defn%can_be_quad) then - call cell_poly_to_quad(cell, quad) - base => quad - call method_cell_quad%init( & - fmi=this%fmi, & - cell=base, & - trackctl=this%trackctl, & - tracktimes=this%tracktimes) - submethod => method_cell_quad - else - call method_cell_tern%init( & - fmi=this%fmi, & - cell=this%cell, & - trackctl=this%trackctl, & - tracktimes=this%tracktimes) - submethod => method_cell_tern - end if + call method_cell_tern%init( & + fmi=this%fmi, & + cell=this%cell, & + trackctl=this%trackctl, & + tracktimes=this%tracktimes) + submethod => method_cell_tern end if end select end subroutine load_disv @@ -156,7 +150,6 @@ subroutine load_particle(this, cell, particle) select type (dis => this%fmi%dis) type is (DisvType) - ! compute and set reduced/user node numbers and layer inface = particle%iboundary(2) idiag = dis%con%ia(cell%defn%icell) inbr = cell%defn%facenbr(inface) @@ -164,13 +157,31 @@ subroutine load_particle(this, cell, particle) ic = dis%con%ja(ipos) icu = dis%get_nodeuser(ic) call get_jk(icu, dis%ncpl, dis%nlay, icpl, ilay) + + ! if returning to a cell through the bottom + ! face after previously leaving it through + ! that same face, we've entered a cycle + ! as can occur e.g. in wells. terminate + ! in the previous cell. + if (ic == particle%icp .and. inface == 7 .and. ilay < particle%ilay) then + particle%advancing = .false. + particle%idomain(2) = particle%icp + particle%istatus = 2 + particle%izone = particle%izp + call this%save(particle, reason=3) + return + else + particle%icp = particle%idomain(2) + particle%izp = particle%izone + end if + particle%idomain(2) = ic particle%icu = icu particle%ilay = ilay - ! map/set particle entry face and z coordinate z = particle%z call this%map_neighbor(cell%defn, inface, z) + particle%iboundary(2) = inface particle%idomain(3:) = 0 particle%iboundary(3:) = 0 @@ -221,9 +232,11 @@ subroutine pass_disv(this, particle) particle%advancing = .false. call this%save(particle, reason=3) else - ! Otherwise, load cell properties into the - ! particle and update intercell mass flows + ! Update old to new cell properties call this%load_particle(cell, particle) + if (.not. particle%advancing) return + + ! Update intercell mass flows call this%update_flowja(cell, particle) end if end select @@ -252,14 +265,14 @@ subroutine map_neighbor(this, defn, inface, z) real(DP) :: bot real(DP) :: sat - ! -- Map to shared cell face of neighbor + ! Map to shared cell face of neighbor inbr = defn%facenbr(inface) if (inbr .eq. 0) then - ! -- Exterior face; no neighbor to map to + ! Exterior face; no neighbor to map to ! todo AMP: reconsider when multiple models allowed inface = -1 else - ! -- Load definition for neighbor cell (neighbor with shared face) + ! Load definition for neighbor cell (neighbor with shared face) icin = defn%icell j = this%fmi%dis%con%ia(icin) ic = this%fmi%dis%con%ja(j + inbr) @@ -268,13 +281,13 @@ subroutine map_neighbor(this, defn, inface, z) npolyvertsin = defn%npolyverts npolyverts = this%neighbor%npolyverts if (inface .eq. npolyvertsin + 2) then - ! -- Exits through bot, enters through top + ! Exits through bot, enters through top inface = npolyverts + 3 else if (inface .eq. npolyvertsin + 3) then - ! -- Exits through top, enters through bot + ! Exits through top, enters through bot inface = npolyverts + 2 else - ! -- Exits and enters through shared polygon face + ! Exits and enters through shared polygon face j = this%fmi%dis%con%ia(ic) do m = 1, npolyverts + 3 inbrnbr = this%neighbor%facenbr(m) @@ -283,7 +296,7 @@ subroutine map_neighbor(this, defn, inface, z) exit end if end do - ! -- Map z between cells + ! Map z between cells topfrom = defn%top botfrom = defn%bot zrel = (z - botfrom) / (topfrom - botfrom) @@ -295,34 +308,26 @@ subroutine map_neighbor(this, defn, inface, z) end if end subroutine map_neighbor - !> @brief Apply the DISV-grid method + !> @brief Apply the DISV tracking method to a particle. subroutine apply_disv(this, particle, tmax) class(MethodDisvType), intent(inout) :: this type(ParticleType), pointer, intent(inout) :: particle real(DP), intent(in) :: tmax + call this%track(particle, 1, tmax) end subroutine apply_disv !> @brief Load cell definition from the grid subroutine load_cell_defn(this, ic, defn) - ! -- dummy + ! dummy class(MethodDisvType), intent(inout) :: this integer(I4B), intent(in) :: ic type(CellDefnType), pointer, intent(inout) :: defn - ! -- Load basic cell properties call this%load_properties(ic, defn) - - ! -- Load polygon vertices call this%load_polygon(defn) - - ! -- Load face neighbors call this%load_neighbors(defn) - - ! -- Load 180-degree indicator call this%load_indicators(defn) - - ! -- Load flows (assumes face neighbors already loaded) call this%load_flows(defn) end subroutine load_cell_defn @@ -336,6 +341,7 @@ subroutine load_properties(this, ic, defn) real(DP) :: top real(DP) :: bot real(DP) :: sat + integer(I4B) :: icu, icpl, ilay defn%icell = ic defn%iatop = get_iatop(this%fmi%dis%get_ncpl(), & @@ -350,7 +356,12 @@ subroutine load_properties(this, ic, defn) defn%porosity = this%porosity(ic) defn%retfactor = this%retfactor(ic) defn%izone = this%izone(ic) - + select type (dis => this%fmi%dis) + type is (DisvType) + icu = dis%get_nodeuser(ic) + call get_jk(icu, dis%ncpl, dis%nlay, icpl, ilay) + defn%ilay = ilay + end select end subroutine load_properties subroutine load_polygon(this, defn) @@ -363,16 +374,15 @@ subroutine load_polygon(this, defn) defn%polyvert, & closed=.true.) defn%npolyverts = size(defn%polyvert, dim=2) - 1 - end subroutine load_polygon !> @brief Loads face neighbors to cell definition from the grid !! Assumes cell index and number of vertices are already loaded. subroutine load_neighbors(this, defn) - ! -- dummy + ! dummy class(MethodDisvType), intent(inout) :: this type(CellDefnType), pointer, intent(inout) :: defn - ! -- local + ! local integer(I4B) :: ic1 integer(I4B) :: ic2 integer(I4B) :: icu1 @@ -392,14 +402,14 @@ subroutine load_neighbors(this, defn) integer(I4B) :: nfaces integer(I4B) :: nslots - ! -- expand facenbr array if needed + ! expand facenbr array if needed nfaces = defn%npolyverts + 3 nslots = size(defn%facenbr) if (nslots < nfaces) call ExpandArray(defn%facenbr, nfaces - nslots) select type (dis => this%fmi%dis) type is (DisvType) - ! -- Load face neighbors. + ! Load face neighbors. defn%facenbr = 0 ic1 = defn%icell icu1 = dis%get_nodeuser(ic1) @@ -420,14 +430,14 @@ subroutine load_neighbors(this, defn) dis%javert(istart2:istop2), & isharedface) if (isharedface /= 0) then - ! -- Edge (polygon) face neighbor + ! Edge (polygon) face neighbor defn%facenbr(isharedface) = int(iloc, 1) else if (k2 > k1) then - ! -- Bottom face neighbor + ! Bottom face neighbor defn%facenbr(defn%npolyverts + 2) = int(iloc, 1) else if (k2 < k1) then - ! -- Top face neighbor + ! Top face neighbor defn%facenbr(defn%npolyverts + 3) = int(iloc, 1) else call pstop(1, "k2 should be <> k1, since no shared edge face") @@ -435,9 +445,8 @@ subroutine load_neighbors(this, defn) end if end do end select - ! -- List of edge (polygon) faces wraps around + ! List of edge (polygon) faces wraps around defn%facenbr(defn%npolyverts + 1) = defn%facenbr(1) - end subroutine load_neighbors !> @brief Load flows into the cell definition. @@ -476,7 +485,6 @@ subroutine load_flows(this, defn) else defn%iweaksink = 0 end if - end subroutine load_flows subroutine load_face_flows_to_defn_poly(this, defn) @@ -501,10 +509,10 @@ end subroutine load_face_flows_to_defn_poly !> @brief Load boundary flows from the grid into a rectangular cell. !! Assumes cell index and number of vertices are already loaded. subroutine load_boundary_flows_to_defn_rect(this, defn) - ! -- dummy + ! dummy class(MethodDisvType), intent(inout) :: this type(CellDefnType), intent(inout) :: defn - ! -- local + ! local integer(I4B) :: ioffset ! assignment of BoundaryFlows to faceflow below assumes clockwise @@ -523,16 +531,15 @@ subroutine load_boundary_flows_to_defn_rect(this, defn) this%fmi%BoundaryFlows(ioffset + 9) defn%faceflow(7) = defn%faceflow(7) + & this%fmi%BoundaryFlows(ioffset + 10) - end subroutine load_boundary_flows_to_defn_rect !> @brief Load boundary flows from the grid into rectangular quadcell. !! Assumes cell index and number of vertices are already loaded. subroutine load_boundary_flows_to_defn_rect_quad(this, defn) - ! -- dummy + ! dummy class(MethodDisvType), intent(inout) :: this type(CellDefnType), intent(inout) :: defn - ! -- local + ! local integer(I4B) :: m integer(I4B) :: n integer(I4B) :: nn @@ -546,7 +553,7 @@ subroutine load_boundary_flows_to_defn_rect_quad(this, defn) ioffset = (defn%icell - 1) * 10 - ! -- Polygon faces in positions 1 through npolyverts + ! Polygon faces in positions 1 through npolyverts do n = 1, 4 if (n .eq. 2) then nbf = 4 @@ -569,23 +576,23 @@ subroutine load_boundary_flows_to_defn_rect_quad(this, defn) if (m2 .lt. m1) m2 = m2 + defn%npolyverts mdiff = m2 - m1 if (mdiff .eq. 1) then - ! -- Assign BoundaryFlow to corresponding polygon face + ! Assign BoundaryFlow to corresponding polygon face defn%faceflow(m1) = defn%faceflow(m1) + qbf else - ! -- Split BoundaryFlow between two faces on quad-refined edge + ! Split BoundaryFlow between two faces on quad-refined edge qbf = 5d-1 * qbf defn%faceflow(m1) = defn%faceflow(m1) + qbf defn%faceflow(m1 + 1) = defn%faceflow(m1 + 1) + qbf end if end do - ! -- Wrap around to 1 in position npolyverts+1 + ! Wrap around to 1 in position npolyverts+1 m = defn%npolyverts + 1 defn%faceflow(m) = defn%faceflow(1) - ! -- Bottom in position npolyverts+2 + ! Bottom in position npolyverts+2 m = m + 1 defn%faceflow(m) = defn%faceflow(m) + & this%fmi%BoundaryFlows(ioffset + 9) - ! -- Top in position npolyverts+3 + ! Top in position npolyverts+3 m = m + 1 defn%faceflow(m) = defn%faceflow(m) + & this%fmi%BoundaryFlows(ioffset + 10) @@ -595,10 +602,10 @@ end subroutine load_boundary_flows_to_defn_rect_quad !> @brief Load boundary flows from the grid into a polygonal cell. !! Assumes cell index and number of vertices are already loaded. subroutine load_boundary_flows_to_defn_poly(this, defn) - ! -- dummy + ! dummy class(MethodDisvType), intent(inout) :: this type(CellDefnType), intent(inout) :: defn - ! -- local + ! local integer(I4B) :: ic integer(I4B) :: npolyverts integer(I4B) :: ioffset @@ -624,13 +631,14 @@ subroutine load_boundary_flows_to_defn_poly(this, defn) end subroutine load_boundary_flows_to_defn_poly !> @brief Load 180-degree vertex indicator array and set flags - !! indicating how cell can be represented. - !! Assumes cell index and number of vertices are already loaded. + !! indicating how cell can be represented. Assumes cell index + !! and number of vertices are already loaded. + !< subroutine load_indicators(this, defn) - ! -- dummy + ! dummy class(MethodDisvType), intent(inout) :: this type(CellDefnType), pointer, intent(inout) :: defn - ! -- local + ! local integer(I4B) :: npolyverts integer(I4B) :: m integer(I4B) :: m0 @@ -655,12 +663,9 @@ subroutine load_indicators(this, defn) ic = defn%icell npolyverts = defn%npolyverts - ! -- expand ispv180 array if needed if (size(defn%ispv180) < npolyverts + 3) & call ExpandArray(defn%ispv180, npolyverts + 1) - ! -- Load 180-degree indicator. - ! -- Also, set flags that indicate how cell can be represented. defn%ispv180(1:npolyverts + 1) = .false. defn%can_be_rect = .false. defn%can_be_quad = .false. @@ -668,6 +673,7 @@ subroutine load_indicators(this, defn) num90 = 0 num180 = 0 last180 = .false. + ! assumes non-self-intersecting polygon do m = 1, npolyverts m1 = m @@ -708,9 +714,10 @@ subroutine load_indicators(this, defn) end if end do - ! -- List of 180-degree indicators wraps around for convenience + ! List of 180-degree indicators wraps around for convenience defn%ispv180(npolyverts + 1) = defn%ispv180(1) + ! Set rect/quad flags if (num90 .eq. 4) then if (num180 .eq. 0) then defn%can_be_rect = .true. @@ -718,7 +725,6 @@ subroutine load_indicators(this, defn) defn%can_be_quad = .true. end if end if - end subroutine load_indicators end module MethodDisvModule diff --git a/src/Solution/ParticleTracker/MethodSubcellPollock.f90 b/src/Solution/ParticleTracker/MethodSubcellPollock.f90 index de9157b8b9e..1cbd0cce45b 100644 --- a/src/Solution/ParticleTracker/MethodSubcellPollock.f90 +++ b/src/Solution/ParticleTracker/MethodSubcellPollock.f90 @@ -36,14 +36,14 @@ subroutine create_method_subcell_pollock(method) allocate (method) call create_subcell_rect(subcell) method%subcell => subcell - method%type => method%subcell%type + method%name => method%subcell%type method%delegates = .false. end subroutine create_method_subcell_pollock !> @brief Deallocate the Pollock's subcell method subroutine deallocate (this) class(MethodSubcellPollockType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine deallocate !> @brief Apply Pollock's method to a rectangular subcell @@ -73,6 +73,7 @@ subroutine apply_msp(this, particle, tmax) call particle%transform(xOrigin, yOrigin) call this%track_subcell(subcell, particle, tmax) call particle%transform(xOrigin, yOrigin, invert=.true.) + call particle%reset_transform() end select end subroutine apply_msp diff --git a/src/Solution/ParticleTracker/MethodSubcellTernary.f90 b/src/Solution/ParticleTracker/MethodSubcellTernary.f90 index 057eca00050..97637b4c59b 100644 --- a/src/Solution/ParticleTracker/MethodSubcellTernary.f90 +++ b/src/Solution/ParticleTracker/MethodSubcellTernary.f90 @@ -38,14 +38,14 @@ subroutine create_method_subcell_ternary(method) allocate (method) call create_subcell_tri(subcell) method%subcell => subcell - method%type => method%subcell%type + method%name => method%subcell%type method%delegates = .false. end subroutine create_method_subcell_ternary !> @brief Deallocate the ternary subcell tracking method. subroutine deallocate (this) class(MethodSubcellTernaryType), intent(inout) :: this - deallocate (this%type) + deallocate (this%name) end subroutine deallocate !> @brief Apply the ternary subcell tracking method. diff --git a/src/Solution/ParticleTracker/Particle.f90 b/src/Solution/ParticleTracker/Particle.f90 index 3bb1471345c..347ae4b1a2b 100644 --- a/src/Solution/ParticleTracker/Particle.f90 +++ b/src/Solution/ParticleTracker/Particle.f90 @@ -40,12 +40,15 @@ module ParticleModule ! stop criteria integer(I4B), public :: istopweaksink !< weak sink option (0: do not stop, 1: stop) integer(I4B), public :: istopzone !< stop zone number + integer(I4B), public :: idrymeth !< dry tracking method ! state - integer(I4B), allocatable, public :: idomain(:) !< tracking domain hierarchy + integer(I4B), allocatable, public :: idomain(:) !< tracking domain hierarchy ! TODO: rename to itdomain? idomain integer(I4B), allocatable, public :: iboundary(:) !< tracking domain boundaries - integer(I4B), public :: icu !< user cell (node) number + integer(I4B), public :: icp !< previous cell number (reduced) + integer(I4B), public :: icu !< user cell number integer(I4B), public :: ilay !< grid layer - integer(I4B), public :: izone !< zone number + integer(I4B), public :: izone !< current zone number + integer(I4B), public :: izp !< previous zone number integer(I4B), public :: istatus !< tracking status real(DP), public :: x !< x coordinate real(DP), public :: y !< y coordinate @@ -68,6 +71,7 @@ module ParticleModule procedure, public :: get_model_coords procedure, public :: load_particle procedure, public :: transform => transform_coords + procedure, public :: reset_transform end type ParticleType !> @brief Structure of arrays to store particles. @@ -81,12 +85,15 @@ module ParticleModule ! stopping criteria integer(I4B), dimension(:), pointer, public, contiguous :: istopweaksink !< weak sink option: 0 = do not stop, 1 = stop integer(I4B), dimension(:), pointer, public, contiguous :: istopzone !< stop zone number + integer(I4B), dimension(:), pointer, public, contiguous :: idrymeth !< stop in dry cells ! state integer(I4B), dimension(:, :), pointer, public, contiguous :: idomain !< array of indices for domains in the tracking domain hierarchy integer(I4B), dimension(:, :), pointer, public, contiguous :: iboundary !< array of indices for tracking domain boundaries - integer(I4B), dimension(:), pointer, public, contiguous :: icu !< cell number (user, not reduced) + integer(I4B), dimension(:), pointer, public, contiguous :: icp !< previous cell number (reduced) + integer(I4B), dimension(:), pointer, public, contiguous :: icu !< cell number (user) integer(I4B), dimension(:), pointer, public, contiguous :: ilay !< layer integer(I4B), dimension(:), pointer, public, contiguous :: izone !< current zone number + integer(I4B), dimension(:), pointer, public, contiguous :: izp !< previous zone number integer(I4B), dimension(:), pointer, public, contiguous :: istatus !< particle status real(DP), dimension(:), pointer, public, contiguous :: x !< model x coord of particle real(DP), dimension(:), pointer, public, contiguous :: y !< model y coord of particle @@ -126,9 +133,11 @@ subroutine allocate_particle_store(this, np, mempath) call mem_allocate(this%irpt, np, 'PLIRPT', mempath) call mem_allocate(this%iprp, np, 'PLIPRP', mempath) call mem_allocate(this%name, LENBOUNDNAME, np, 'PLNAME', mempath) + call mem_allocate(this%icp, np, 'PLICP', mempath) call mem_allocate(this%icu, np, 'PLICU', mempath) call mem_allocate(this%ilay, np, 'PLILAY', mempath) call mem_allocate(this%izone, np, 'PLIZONE', mempath) + call mem_allocate(this%izp, np, 'PLIZP', mempath) call mem_allocate(this%istatus, np, 'PLISTATUS', mempath) call mem_allocate(this%x, np, 'PLX', mempath) call mem_allocate(this%y, np, 'PLY', mempath) @@ -138,6 +147,7 @@ subroutine allocate_particle_store(this, np, mempath) call mem_allocate(this%ttrack, np, 'PLTTRACK', mempath) call mem_allocate(this%istopweaksink, np, 'PLISTOPWEAKSINK', mempath) call mem_allocate(this%istopzone, np, 'PLISTOPZONE', mempath) + call mem_allocate(this%idrymeth, np, 'PLIDRYMETH', mempath) call mem_allocate(this%ifrctrn, np, 'PLIFRCTRN', mempath) call mem_allocate(this%iexmeth, np, 'PLIEXMETH', mempath) call mem_allocate(this%extol, np, 'PLEXTOL', mempath) @@ -155,9 +165,11 @@ subroutine deallocate (this, mempath) call mem_deallocate(this%iprp, 'PLIPRP', mempath) call mem_deallocate(this%irpt, 'PLIRPT', mempath) call mem_deallocate(this%name, 'PLNAME', mempath) + call mem_deallocate(this%icp, 'PLICP', mempath) call mem_deallocate(this%icu, 'PLICU', mempath) call mem_deallocate(this%ilay, 'PLILAY', mempath) call mem_deallocate(this%izone, 'PLIZONE', mempath) + call mem_deallocate(this%izp, 'PLIZP', mempath) call mem_deallocate(this%istatus, 'PLISTATUS', mempath) call mem_deallocate(this%x, 'PLX', mempath) call mem_deallocate(this%y, 'PLY', mempath) @@ -167,6 +179,7 @@ subroutine deallocate (this, mempath) call mem_deallocate(this%ttrack, 'PLTTRACK', mempath) call mem_deallocate(this%istopweaksink, 'PLISTOPWEAKSINK', mempath) call mem_deallocate(this%istopzone, 'PLISTOPZONE', mempath) + call mem_deallocate(this%idrymeth, 'PLIDRYMETH', mempath) call mem_deallocate(this%ifrctrn, 'PLIFRCTRN', mempath) call mem_deallocate(this%iexmeth, 'PLIEXMETH', mempath) call mem_deallocate(this%extol, 'PLEXTOL', mempath) @@ -177,7 +190,7 @@ end subroutine deallocate !> @brief Reallocate particle arrays subroutine resize(this, np, mempath) - ! -- dummy + ! dummy class(ParticleStoreType), intent(inout) :: this !< particle store integer(I4B), intent(in) :: np !< number of particles character(*), intent(in) :: mempath !< path to memory @@ -187,9 +200,11 @@ subroutine resize(this, np, mempath) call mem_reallocate(this%iprp, np, 'PLIPRP', mempath) call mem_reallocate(this%irpt, np, 'PLIRPT', mempath) call mem_reallocate(this%name, LENBOUNDNAME, np, 'PLNAME', mempath) + call mem_reallocate(this%icp, np, 'PLICP', mempath) call mem_reallocate(this%icu, np, 'PLICU', mempath) call mem_reallocate(this%ilay, np, 'PLILAY', mempath) call mem_reallocate(this%izone, np, 'PLIZONE', mempath) + call mem_reallocate(this%izp, np, 'PLIZP', mempath) call mem_reallocate(this%istatus, np, 'PLISTATUS', mempath) call mem_reallocate(this%x, np, 'PLX', mempath) call mem_reallocate(this%y, np, 'PLY', mempath) @@ -199,6 +214,7 @@ subroutine resize(this, np, mempath) call mem_reallocate(this%ttrack, np, 'PLTTRACK', mempath) call mem_reallocate(this%istopweaksink, np, 'PLISTOPWEAKSINK', mempath) call mem_reallocate(this%istopzone, np, 'PLISTOPZONE', mempath) + call mem_reallocate(this%idrymeth, np, 'PLIDRYMETH', mempath) call mem_reallocate(this%ifrctrn, np, 'PLIFRCTRN', mempath) call mem_reallocate(this%iexmeth, np, 'PLIEXMETH', mempath) call mem_reallocate(this%extol, np, 'PLEXTOL', mempath) @@ -219,7 +235,7 @@ subroutine load_particle(this, store, imdl, iprp, ip) integer(I4B), intent(in) :: iprp !< index of particle release package particle originated in integer(I4B), intent(in) :: ip !< index into the particle list - call this%transform(reset=.true.) + call this%reset_transform() this%imdl = imdl this%iprp = iprp this%irpt = store%irpt(ip) @@ -227,9 +243,12 @@ subroutine load_particle(this, store, imdl, iprp, ip) this%name = store%name(ip) this%istopweaksink = store%istopweaksink(ip) this%istopzone = store%istopzone(ip) + this%idrymeth = store%idrymeth(ip) + this%icp = store%icp(ip) this%icu = store%icu(ip) this%ilay = store%ilay(ip) this%izone = store%izone(ip) + this%izp = store%izp(ip) this%istatus = store%istatus(ip) this%x = store%x(ip) this%y = store%y(ip) @@ -261,9 +280,12 @@ subroutine save_particle(this, particle, ip) this%name(ip) = particle%name this%istopweaksink(ip) = particle%istopweaksink this%istopzone(ip) = particle%istopzone + this%idrymeth(ip) = particle%idrymeth + this%icp(ip) = particle%icp this%icu(ip) = particle%icu this%ilay(ip) = particle%ilay this%izone(ip) = particle%izone + this%izp(ip) = particle%izp this%istatus(ip) = particle%istatus this%x(ip) = particle%x this%y(ip) = particle%y @@ -285,9 +307,14 @@ subroutine save_particle(this, particle, ip) this%extend(ip) = particle%iextend end subroutine save_particle - !> @brief Apply the given global-to-local transformation to the particle. + !> @brief Transform particle coordinates. + !! + !! Apply a translation and/or rotation to particle coordinates. + !! No rescaling. It's also possible to invert a transformation. + !! Be sure to reset the transformation after using it. + !< subroutine transform_coords(this, xorigin, yorigin, zorigin, & - sinrot, cosrot, invert, reset) + sinrot, cosrot, invert) use GeomUtilModule, only: transform, compose class(ParticleType), intent(inout) :: this !< particle real(DP), intent(in), optional :: xorigin !< x coordinate of origin @@ -296,44 +323,33 @@ subroutine transform_coords(this, xorigin, yorigin, zorigin, & real(DP), intent(in), optional :: sinrot !< sine of rotation angle real(DP), intent(in), optional :: cosrot !< cosine of rotation angle logical(LGP), intent(in), optional :: invert !< whether to invert - logical(LGP), intent(in), optional :: reset !< whether to reset - - ! Reset if requested - if (present(reset)) then - if (reset) then - this%xorigin = DZERO - this%yorigin = DZERO - this%zorigin = DZERO - this%sinrot = DZERO - this%cosrot = DONE - this%cosrot = DONE - this%transformed = .false. - return - end if - end if - ! Otherwise, transform coordinates call transform(this%x, this%y, this%z, & this%x, this%y, this%z, & xorigin, yorigin, zorigin, & sinrot, cosrot, invert) - ! Modify transformation from model coordinates to particle's new - ! local coordinates by incorporating this latest transformation call compose(this%xorigin, this%yorigin, this%zorigin, & this%sinrot, this%cosrot, & xorigin, yorigin, zorigin, & sinrot, cosrot, invert) - ! Set isTransformed flag to true. Note that there is no check - ! to see whether the modification brings the coordinates back - ! to model coordinates (in which case the origin would be very - ! close to zero and sinrot and cosrot would be very close to 0. - ! and 1., respectively, allowing for roundoff error). this%transformed = .true. end subroutine transform_coords - !> @brief Return the particle's model (global) coordinates. + subroutine reset_transform(this) + class(ParticleType), intent(inout) :: this !< particle + + this%xorigin = DZERO + this%yorigin = DZERO + this%zorigin = DZERO + this%sinrot = DZERO + this%cosrot = DONE + this%cosrot = DONE + this%transformed = .false. + end subroutine reset_transform + + !> @brief Return the particle's model coordinates. subroutine get_model_coords(this, x, y, z) use GeomUtilModule, only: transform, compose class(ParticleType), intent(inout) :: this !< particle @@ -342,12 +358,11 @@ subroutine get_model_coords(this, x, y, z) real(DP), intent(out) :: z !< z coordinate if (this%transformed) then - ! Transform back from local to model coordinates + ! Untransform coordinates call transform(this%x, this%y, this%z, x, y, z, & this%xorigin, this%yorigin, this%zorigin, & - this%sinrot, this%cosrot, .true.) + this%sinrot, this%cosrot, invert=.true.) else - ! Already in model coordinates x = this%x y = this%y z = this%z diff --git a/src/Solution/ParticleTracker/vertical.md b/src/Solution/ParticleTracker/vertical.md new file mode 100644 index 00000000000..f0afbc109ba --- /dev/null +++ b/src/Solution/ParticleTracker/vertical.md @@ -0,0 +1,133 @@ +# Vertical tracking + +This document describes the approach PRT takes to vertical particle motion. + +## 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. + +```mermaid +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 +``` + +## 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. + +Particles may traverse an arbitrary number of cells in a time step, in the lateral as well as vertical dimensions. + +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. + +```mermaid +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 +``` + +### 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. + +```mermaid +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 +``` + +#### 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. \ No newline at end of file