Skip to content

Commit

Permalink
Merge pull request #134 from USEPA/5-2-3_Bug_Fixes
Browse files Browse the repository at this point in the history
Bug fixes for release 5.2.3
  • Loading branch information
LRossman authored Jul 27, 2023
2 parents 48ca812 + d4236a9 commit 890ce92
Show file tree
Hide file tree
Showing 10 changed files with 105 additions and 72 deletions.
4 changes: 2 additions & 2 deletions src/solver/consts.h
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 02/12/23 (Build 5.2.3)
// Date: 07/15/23 (Build 5.2.4)
// Author: L. Rossman
//
// Various Constants
Expand All @@ -16,7 +16,7 @@
// General Constants
//------------------

#define VERSION 52003
#define VERSION 52004
#define MAGICNUMBER 516114522
#define EOFMARK 0x1A // Use 0x04 for UNIX systems
#define MAXTITLE 3 // Max. # title lines
Expand Down
4 changes: 2 additions & 2 deletions src/solver/dynwave.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 06/12/23 (Build 5.2.4)
// Date: 07/13/23 (Build 5.2.4)
// Author: L. Rossman
// M. Tryby (EPA)
// R. Dickinson (CDM)
Expand Down Expand Up @@ -599,7 +599,7 @@ int findNodeDepths(double dt)
//
{
int i;
double yOld; // previous node depth (ft)
double yOld = 0.0; // previous node depth (ft)

// --- compute outfall depths based on flow in connecting link
for ( i = 0; i < Nobjects[LINK]; i++ ) link_setOutfallDepth(i);
Expand Down
12 changes: 8 additions & 4 deletions src/solver/inlet.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 10/08/22 (Build 5.2.2)
// Date: 07/13/23 (Build 5.2.4)
// Author: L. Rossman
//
// Street/Channel Inlet Functions
Expand All @@ -18,6 +18,10 @@
// - Substitutes the constant BIG for HUGE.
// Build 5.2.2:
// - Additional statistics added to Street Flow Summary table.
// Build 5.2.4:
// - Fixed expression for equivalent gutter slope in getCurbInletCapture.
// - Corrected sign in equation for effective head in a curb inlet
// with an inclined throat opening in getCurbOrificeFlow.
//-----------------------------------------------------------------------------
#define _CRT_SECURE_NO_DEPRECATE

Expand Down Expand Up @@ -1475,7 +1479,7 @@ double getCurbInletCapture(double Q, double L)
// Input: Q = flow rate seen by inlet (cfs)
// L = length of inlet opening (ft)
// Output: returns captured flow rate (cfs)
// Purpose: finds the flow captured by an on-sag inlet.
// Purpose: finds the flow captured by an on-grade curb opening inlet.
//
{
double Se = Sx, // equivalent gutter slope
Expand All @@ -1491,7 +1495,7 @@ double getCurbInletCapture(double Q, double L)
{
Sr = Sw / Sx;
Eo = getEo(Sr, T-W, W);
Se = Sx + Sw * Eo; //HEC-22 Eq(4-24)
Se = Sx + (a/W) * Eo; //HEC-22 Eq(4-24)
}

// --- opening length for full capture
Expand Down Expand Up @@ -1772,7 +1776,7 @@ double getCurbOrificeFlow(double di, double h, double L, int throatAngle)
if (throatAngle == HORIZONTAL_THROAT)
d = di - h / 2.0;
else if (throatAngle == INCLINED_THROAT)
d = di + (h / 2.0) * 0.7071;
d = di - (h / 2.0) * 0.7071;
return 0.67 * h * L * sqrt(2.0 * 32.16 * d); //HEC-22 Eq(4-31a)
}

Expand Down
6 changes: 4 additions & 2 deletions src/solver/lid.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 02/10/23 (Build 5.2.3)
// Date: 07/13/23 (Build 5.2.4)
// Author: L. Rossman
//
// This module handles all data processing involving LID (Low Impact
Expand Down Expand Up @@ -70,6 +70,8 @@
// - Covered property added to RAIN_BARREL parameters
// Build 5.2.3
// - Fixed double counting of initial water volume in green roof drain mat.
// Build 5.2.4
// - Fixed test for invalid data in readDrainData function.
//-----------------------------------------------------------------------------
#define _CRT_SECURE_NO_DEPRECATE

Expand Down Expand Up @@ -765,7 +767,7 @@ int readDrainData(int j, char* toks[], int ntoks)
for (i = 0; i < 6; i++) x[i] = 0.0;
for (i = 2; i < 8; i++)
{
if ( (ntoks > i) && (! getDouble(toks[i], &x[i-2]) || x[i-2]) < 0.0 )
if ( (ntoks > i) && (! getDouble(toks[i], &x[i-2]) || x[i-2] < 0.0) )
return error_setInpError(ERR_NUMBER, toks[i]);
}

Expand Down
111 changes: 63 additions & 48 deletions src/solver/lidproc.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 11/01/21 (Build 5.2.0)
// Date: 07/13/23 (Build 5.2.4)
// Author: L. Rossman
//
// This module computes the hydrologic performance of an LID (Low Impact
Expand Down Expand Up @@ -53,6 +53,11 @@
// - Fixed failure to account for effect of Impervious Surface Fraction on
// pavement permeability for Permeable Pavement LID
// - Fixed units conversion for pavement depth in detailed report file.
// Build 5.2.4:
// - Modified flux limits in biocellFluxRates, pavementFluxRates and
// trenchFluxRates.
// - Corrected head calculation in getStorageDrainRate when unit has both
// a soil and pavement layer.
//-----------------------------------------------------------------------------
#define _CRT_SECURE_NO_DEPRECATE

Expand Down Expand Up @@ -654,61 +659,64 @@ void biocellFluxRates(double x[], double f[])
SurfaceInfil = MIN(SurfaceInfil, maxRate);
}

//... storage & soil layers are full
else if ( soilTheta >= soilPorosity && storageDepth >= storageThickness )
else
{
//... limiting rate is smaller of soil perc and storage outflow
maxRate = StorageExfil + StorageDrain;
if ( SoilPerc < maxRate )
//... storage & soil layers are full
if ( soilTheta >= soilPorosity && storageDepth >= storageThickness )
{
maxRate = SoilPerc;
if ( maxRate > StorageExfil ) StorageDrain = maxRate - StorageExfil;
else
//... limiting rate is smaller of soil perc and storage outflow
maxRate = StorageExfil + StorageDrain;
if ( SoilPerc < maxRate )
{
StorageExfil = maxRate;
StorageDrain = 0.0;
maxRate = SoilPerc;
if ( maxRate > StorageExfil ) StorageDrain = maxRate - StorageExfil;
else
{
StorageExfil = maxRate;
StorageDrain = 0.0;
}
}
}
else SoilPerc = maxRate;

//... apply limiting rate to surface infil.
SurfaceInfil = MIN(SurfaceInfil, maxRate);
}
else SoilPerc = maxRate;

//... either layer not full
else if ( storageThickness > 0.0 )
{
//... limit storage exfiltration by available storage volume
maxRate = SoilPerc - StorageEvap + storageDepth*storageVoidFrac/Tstep;
StorageExfil = MIN(StorageExfil, maxRate);
StorageExfil = MAX(StorageExfil, 0.0);
//... apply limiting rate to surface infil.
SurfaceInfil = MIN(SurfaceInfil, maxRate);
}

//... limit underdrain flow by volume above drain offset
if ( StorageDrain > 0.0 )
//... either layer not full
else
{
maxRate = -StorageExfil - StorageEvap;
if ( storageDepth >= storageThickness) maxRate += SoilPerc;
if ( theLidProc->drain.offset <= storageDepth )
//... limit storage exfiltration by available storage volume
maxRate = SoilPerc - StorageEvap + storageDepth*storageVoidFrac/Tstep;
StorageExfil = MIN(StorageExfil, maxRate);
StorageExfil = MAX(StorageExfil, 0.0);

//... limit underdrain flow by volume above drain offset
if ( StorageDrain > 0.0 )
{
maxRate += (storageDepth - theLidProc->drain.offset) *
storageVoidFrac/Tstep;
maxRate = -StorageExfil - StorageEvap;
if ( storageDepth >= storageThickness) maxRate += SoilPerc;
if ( theLidProc->drain.offset <= storageDepth )
{
maxRate += (storageDepth - theLidProc->drain.offset) *
storageVoidFrac/Tstep;
}
maxRate = MAX(maxRate, 0.0);
StorageDrain = MIN(StorageDrain, maxRate);
}
maxRate = MAX(maxRate, 0.0);
StorageDrain = MIN(StorageDrain, maxRate);
}

//... limit soil perc by unused storage volume
maxRate = StorageExfil + StorageDrain + StorageEvap +
(storageThickness - storageDepth) *
storageVoidFrac/Tstep;
SoilPerc = MIN(SoilPerc, maxRate);

//... limit soil perc by unused storage volume
maxRate = StorageExfil + StorageDrain + StorageEvap +
(storageThickness - storageDepth) *
storageVoidFrac/Tstep;
SoilPerc = MIN(SoilPerc, maxRate);

//... limit surface infil. by unused soil volume
maxRate = (soilPorosity - soilTheta) * soilThickness / Tstep +
SoilPerc + SoilEvap;
SurfaceInfil = MIN(SurfaceInfil, maxRate);
//... limit surface infil. by unused soil volume
maxRate = (soilPorosity - soilTheta) * soilThickness / Tstep +
SoilPerc + SoilEvap;
SurfaceInfil = MIN(SurfaceInfil, maxRate);
}
}

//... find surface layer outflow rate
SurfaceOutflow = getSurfaceOutflowRate(surfaceDepth);

Expand Down Expand Up @@ -803,7 +811,7 @@ void trenchFluxRates(double x[], double f[])
SurfaceOutflow = getSurfaceOutflowRate(surfaceDepth);

// ... find net fluxes for each layer
f[SURF] = SurfaceInflow - SurfaceEvap - StorageInflow - SurfaceOutflow /
f[SURF] = (SurfaceInflow - SurfaceEvap - StorageInflow - SurfaceOutflow) /
theLidProc->surface.voidFrac;;
f[STOR] = (StorageInflow - StorageEvap - StorageExfil - StorageDrain) /
theLidProc->storage.voidFrac;
Expand Down Expand Up @@ -966,6 +974,7 @@ void pavementFluxRates(double x[], double f[])
StorageExfil = MIN(StorageExfil, SoilPerc);
StorageDrain = SoilPerc - StorageExfil;
}
PavePerc = MIN(PavePerc, SoilPerc);

//... limit surface infil. by available pavement volume
availVolume = (paveThickness - paveDepth) * paveVoidFrac;
Expand All @@ -981,6 +990,8 @@ void pavementFluxRates(double x[], double f[])
PavePerc = MIN(PavePerc, SoilPerc);
SoilPerc = PavePerc;
SurfaceInfil = MIN(SurfaceInfil,PavePerc);
maxRate = MAX(StorageVolume / Tstep + SoilPerc - StorageEvap, 0.0);
StorageExfil = MIN(StorageExfil, maxRate);
}

//... no adjoining layers are full
Expand Down Expand Up @@ -1370,15 +1381,19 @@ double getStorageDrainRate(double storageDepth, double soilTheta,
// depth in layer above it
if ( soilTheta >= soilPorosity )
{
if ( paveThickness > 0.0 ) head += paveDepth;
if ( paveThickness > 0.0 )
{
head += paveDepth;
if ( paveDepth >= paveThickness ) head += surfaceDepth;
}
else head += surfaceDepth;
}
}
}

// --- no soil layer so increase head by water level in pavement
// layer and possibly surface layer
if ( paveThickness > 0.0 )
else if ( paveThickness > 0.0 )
{
head += paveDepth;
if ( paveDepth >= paveThickness ) head += surfaceDepth;
Expand Down
10 changes: 7 additions & 3 deletions src/solver/project.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 11/21/22 (Build 5.2.2)
// Date: 07/17/23 (Build 5.2.4)
// Author: L. Rossman
//
// Project management functions.
Expand Down Expand Up @@ -52,6 +52,10 @@
// Build 5.2.2:
// - Default number of threads changed from OMP max. number to 1
// to be consistent with User Manual.
// Build 5.2.4:
// - Default Inertial Damping changed from SOME to PARTIAL_DAMPING.
// - Default CourantFactor changed from 0 (fixed routing time step)
// - to 0.75 (variable time step)
//-----------------------------------------------------------------------------
#define _CRT_SECURE_NO_DEPRECATE

Expand Down Expand Up @@ -842,12 +846,12 @@ void setDefaults()
SurchargeMethod = EXTRAN; // Use EXTRAN method for surcharging
CrownCutoff = 0.96; // Fractional pipe crown cutoff
AllowPonding = FALSE; // No ponding at nodes
InertDamping = SOME; // Partial inertial damping
InertDamping = PARTIAL_DAMPING; // Partial inertial damping
NormalFlowLtd = BOTH; // Default normal flow limitation
ForceMainEqn = H_W; // Hazen-Williams eqn. for force mains
LinkOffsets = DEPTH_OFFSET; // Use depth for link offsets
LengtheningStep = 0; // No lengthening of conduits
CourantFactor = 0.0; // No variable time step
CourantFactor = 0.75; // Variable time step reduced to 75%
MinSurfArea = 0.0; // Force use of default min. surface area
MinSlope = 0.0; // No user supplied minimum conduit slope
SkipSteadyState = FALSE; // Do flow routing in steady state periods
Expand Down
9 changes: 6 additions & 3 deletions src/solver/street.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,11 +3,14 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 11/01/21 (Build 5.2.0)
// Date: 07/13/23 (Build 5.2.4)
// Author: L. Rossman
//
// Street Cross-Section Functions
//
// Build 5.2.4:
// - Fixed incorrect index used to retrieve street backing parameters
// in street_readParams.
//-----------------------------------------------------------------------------
#define _CRT_SECURE_NO_DEPRECATE

Expand Down Expand Up @@ -106,8 +109,8 @@ int street_readParams(char* tok[], int ntoks)
// --- read street backing parameters
if (ntoks > 8)
{
if (!getDouble(tok[8], &x[8]) || x[k] < 0.0)
return error_setInpError(ERR_NUMBER, tok[k]);
if (!getDouble(tok[8], &x[8]) || x[8] < 0.0)
return error_setInpError(ERR_NUMBER, tok[8]);
if (x[8] > 0.0)
{
if (ntoks < 11) return error_setInpError(ERR_ITEMS, "");
Expand Down
9 changes: 6 additions & 3 deletions src/solver/surfqual.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//
// Project: EPA SWMM5
// Version: 5.2
// Date: 11/01/21 (Build 5.2.0)
// Date: 07/14/23 (Build 5.2.4)
// Author: L. Rossman
//
// Subcatchment water quality functions.
Expand All @@ -16,6 +16,10 @@
// - Support for separate accounting of LID drain flows included.
// Build 5.1.014:
// - Fixed bug in computing effective BMP removal by LIDs.
// Build 5.2.4:
// - Set low runoff flow concentrations to zero before computing runoff
// mass loads rather than after so that they match wet weather mass
// inflows reported for conveyance system nodes.
//-----------------------------------------------------------------------------
#define _CRT_SECURE_NO_DEPRECATE

Expand Down Expand Up @@ -253,7 +257,7 @@ void surfqual_getWashoff(int j, double runoff, double tStep)
{
// --- convert washoff load to a concentration
cOut = 0.0;
if ( vOut1 > 0.0 ) cOut = OutflowLoad[p] / vOut1;
if ( vOut1 > 0.0 && hasOutflow ) cOut = OutflowLoad[p] / vOut1;

// --- assign any difference between pre- and post-LID
// subcatchment outflow loads to BMP removal
Expand All @@ -279,7 +283,6 @@ void surfqual_getWashoff(int j, double runoff, double tStep)
}

// --- save new washoff concentration
if ( !hasOutflow ) cOut = 0.0;
Subcatch[j].newQual[p] = cOut / LperFT3;
}

Expand Down
Loading

0 comments on commit 890ce92

Please sign in to comment.