28 character(len=LENFTYPE) ::
ftype =
'SSM'
29 character(len=LENPACKAGENAME) ::
text =
' SOURCE-SINK MIX'
39 integer(I4B),
pointer :: nbound
40 integer(I4B),
dimension(:),
pointer,
contiguous :: isrctype => null()
41 integer(I4B),
dimension(:),
pointer,
contiguous :: iauxpak => null()
42 integer(I4B),
dimension(:),
pointer,
contiguous :: ibound => null()
43 real(dp),
dimension(:),
pointer,
contiguous :: cnew => null()
46 type(
tspspctype),
dimension(:),
pointer :: ssmivec => null()
47 real(dp),
pointer :: eqnsclfac => null()
48 character(len=LENVARNAME) :: depvartype =
''
82 subroutine ssm_cr(ssmobj, name_model, inunit, iout, fmi, eqnsclfac, &
86 character(len=*),
intent(in) :: name_model
87 integer(I4B),
intent(in) :: inunit
88 integer(I4B),
intent(in) :: iout
90 real(dp),
intent(in),
pointer :: eqnsclfac
91 character(len=LENVARNAME),
intent(in) :: depvartype
97 call ssmobj%set_names(1, name_model,
'SSM',
'SSM')
100 call ssmobj%allocate_scalars()
103 ssmobj%inunit = inunit
106 ssmobj%eqnsclfac => eqnsclfac
109 call ssmobj%parser%Initialize(ssmobj%inunit, ssmobj%iout)
113 ssmobj%depvartype = depvartype
134 subroutine ssm_ar(this, dis, ibound, cnew)
140 integer(I4B),
dimension(:),
pointer,
contiguous :: ibound
141 real(DP),
dimension(:),
pointer,
contiguous :: cnew
143 character(len=*),
parameter :: fmtssm = &
144 "(1x,/1x,'SSM -- SOURCE-SINK MIXING PACKAGE, VERSION 1, 8/25/2017', &
145 &' INPUT READ FROM UNIT ', i0, //)"
148 write (this%iout, fmtssm) this%inunit
152 this%ibound => ibound
156 if (this%fmi%nflowpack == 0)
then
157 write (
errmsg,
'(a)')
'SSM package does not detect any boundary flows &
158 &that require SSM terms. Activate GWF-GWT (or &
159 &GWF-GWE, as appropriate) exchange or activate &
160 &FMI package and provide a budget file that &
161 &contains boundary flows. If no boundary flows &
162 &are present in corresponding GWF model then this &
163 &SSM package should be removed.'
165 call this%parser%StoreErrorUnit()
169 call this%allocate_arrays()
172 call this%read_options()
175 call this%read_data()
178 call this%pak_setup_outputtab()
197 do ip = 1, this%fmi%nflowpack
198 if (this%fmi%iatp(ip) /= 0) cycle
199 if (this%isrctype(ip) == 3 .or. this%isrctype(ip) == 4)
then
200 ssmiptr => this%ssmivec(ip)
201 call ssmiptr%spc_rp()
228 do ip = 1, this%fmi%nflowpack
229 if (this%fmi%iatp(ip) /= 0) cycle
230 do i = 1, this%fmi%gwfpackages(ip)%nbound
231 node = this%fmi%gwfpackages(ip)%nodelist(i)
233 this%nbound = this%nbound + 1
240 do ip = 1, this%fmi%nflowpack
241 if (this%fmi%iatp(ip) /= 0) cycle
242 if (this%isrctype(ip) == 3 .or. this%isrctype(ip) == 4)
then
243 ssmiptr => this%ssmivec(ip)
244 call ssmiptr%spc_ad(this%fmi%gwfpackages(ip)%nbound, &
245 this%fmi%gwfpackages(ip)%budtxt)
257 subroutine ssm_term(this, ipackage, ientry, rrate, rhsval, hcofval, &
261 integer(I4B),
intent(in) :: ipackage
262 integer(I4B),
intent(in) :: ientry
263 real(DP),
intent(out),
optional :: rrate
264 real(DP),
intent(out),
optional :: rhsval
265 real(DP),
intent(out),
optional :: hcofval
266 real(DP),
intent(out),
optional :: cssm
267 real(DP),
intent(out),
optional :: qssm
269 logical(LGP) :: lauxmixed
271 integer(I4B) :: nbound_flow
283 nbound_flow = this%fmi%gwfpackages(ipackage)%nbound
284 n = this%fmi%gwfpackages(ipackage)%nodelist(ientry)
287 if (this%ibound(n) > 0)
then
290 qbnd = this%fmi%gwfpackages(ipackage)%get_flow(ientry)
291 call this%get_ssm_conc(ipackage, ientry, nbound_flow, ctmp, lauxmixed)
295 if (.not. lauxmixed)
then
301 if (qbnd >=
dzero)
then
306 if (ctmp <
dzero)
then
318 if (qbnd >=
dzero)
then
321 if (ctmp < this%cnew(n))
then
331 if (qbnd <=
dzero)
then
332 hcoftmp = qbnd * omega * this%eqnsclfac
334 rhstmp = -qbnd * ctmp * (
done - omega) * this%eqnsclfac
341 if (
present(hcofval)) hcofval = hcoftmp
342 if (
present(rhsval)) rhsval = rhstmp
343 if (
present(rrate)) rrate = hcoftmp * ctmp - rhstmp
344 if (
present(cssm)) cssm = ctmp
345 if (
present(qssm)) qssm = qbnd
361 integer(I4B),
intent(in) :: ipackage
362 integer(I4B),
intent(in) :: ientry
363 integer(I4B),
intent(in) :: nbound_flow
364 real(DP),
intent(out) :: conc
365 logical(LGP),
intent(out) :: lauxmixed
367 integer(I4B) :: isrctype
368 integer(I4B) :: iauxpos
372 isrctype = this%isrctype(ipackage)
374 select case (isrctype)
376 iauxpos = this%iauxpak(ipackage)
377 conc = this%fmi%gwfpackages(ipackage)%auxvar(iauxpos, ientry)
378 if (isrctype == 2) lauxmixed = .true.
380 conc = this%ssmivec(ipackage)%get_value(ientry, nbound_flow)
381 if (isrctype == 4) lauxmixed = .true.
390 subroutine ssm_fc(this, matrix_sln, idxglo, rhs)
394 integer(I4B),
intent(in),
dimension(:) :: idxglo
395 real(DP),
intent(inout),
dimension(:) :: rhs
400 integer(I4B) :: idiag
401 integer(I4B) :: nflowpack
402 integer(I4B) :: nbound
407 nflowpack = this%fmi%nflowpack
409 if (this%fmi%iatp(ip) /= 0) cycle
412 nbound = this%fmi%gwfpackages(ip)%nbound
414 n = this%fmi%gwfpackages(ip)%nodelist(i)
416 call this%ssm_term(ip, i, rhsval=rhsval, hcofval=hcofval)
417 idiag = idxglo(this%dis%con%ia(n))
418 call matrix_sln%add_value_pos(idiag, hcofval)
419 rhs(n) = rhs(n) + rhsval
435 real(DP),
dimension(:),
contiguous,
intent(inout) :: flowja
440 integer(I4B) :: idiag
444 do ip = 1, this%fmi%nflowpack
447 if (this%fmi%iatp(ip) /= 0) cycle
450 do i = 1, this%fmi%gwfpackages(ip)%nbound
451 n = this%fmi%gwfpackages(ip)%nodelist(i)
453 call this%ssm_term(ip, i, rrate=rate)
454 idiag = this%dis%con%ia(n)
455 flowja(idiag) = flowja(idiag) + rate
467 subroutine ssm_bd(this, isuppress_output, model_budget)
473 integer(I4B),
intent(in) :: isuppress_output
474 type(
budgettype),
intent(inout) :: model_budget
476 character(len=LENBUDROWLABEL) :: rowlabel
486 do ip = 1, this%fmi%nflowpack
489 if (this%fmi%iatp(ip) /= 0) cycle
496 do i = 1, this%fmi%gwfpackages(ip)%nbound
497 n = this%fmi%gwfpackages(ip)%nodelist(i)
499 call this%ssm_term(ip, i, rrate=rate)
500 if (rate <
dzero)
then
508 rowlabel =
'SSM_'//adjustl(trim(this%fmi%flowpacknamearray(ip)))
509 call model_budget%addentry(rin, rout,
delt, &
510 this%fmi%gwfpackages(ip)%budtxt, &
511 isuppress_output, rowlabel=rowlabel)
527 integer(I4B),
intent(in) :: icbcfl
528 integer(I4B),
intent(in) :: ibudfl
529 integer(I4B),
intent(in) :: icbcun
531 character(len=LINELENGTH) :: title
532 integer(I4B) :: node, nodeu
533 character(len=20) :: nodestr
534 integer(I4B) :: maxrows
536 integer(I4B) :: i, n2, ibinun
541 real(DP),
dimension(0) :: auxrow
542 character(len=LENAUXNAME),
dimension(0) :: auxname
544 character(len=LENBOUNDNAME) :: bname
546 character(len=*),
parameter :: fmttkk = &
547 &
"(1X,/1X,A,' PERIOD ',I0,' STEP ',I0)"
552 if (ibudfl /= 0 .and. this%iprflow /= 0)
then
553 call this%outputtab%set_kstpkper(
kstp,
kper)
554 do ip = 1, this%fmi%nflowpack
555 if (this%fmi%iatp(ip) /= 0) cycle
558 do i = 1, this%fmi%gwfpackages(ip)%nbound
559 node = this%fmi%gwfpackages(ip)%nodelist(i)
561 maxrows = maxrows + 1
565 if (maxrows > 0)
then
566 call this%outputtab%set_maxbound(maxrows)
568 title =
'SSM PACKAGE ('//trim(this%packName)// &
570 call this%outputtab%set_title(title)
574 if (this%ipakcb < 0)
then
576 else if (this%ipakcb == 0)
then
581 if (icbcfl == 0) ibinun = 0
584 if (ibinun /= 0)
then
585 call this%dis%record_srcdst_list_header(
text, this%name_model, &
586 this%name_model, this%name_model, &
587 this%packName, naux, auxname, &
588 ibinun, this%nbound, this%iout)
592 if (this%nbound > 0)
then
595 do ip = 1, this%fmi%nflowpack
596 if (this%fmi%iatp(ip) /= 0) cycle
599 do i = 1, this%fmi%gwfpackages(ip)%nbound
602 node = this%fmi%gwfpackages(ip)%nodelist(i)
604 call this%ssm_term(ip, i, rrate=rrate, qssm=qssm, cssm=cssm)
608 if (ibudfl /= 0)
then
609 if (this%iprflow /= 0)
then
612 nodeu = this%dis%get_nodeuser(node)
613 call this%dis%nodeu_to_string(nodeu, nodestr)
614 bname = this%fmi%gwfpackages(ip)%name
615 call this%outputtab%add_term(i)
616 call this%outputtab%add_term(trim(adjustl(nodestr)))
617 call this%outputtab%add_term(qssm)
618 call this%outputtab%add_term(cssm)
619 call this%outputtab%add_term(rrate)
620 call this%outputtab%add_term(bname)
625 if (ibinun /= 0)
then
627 call this%dis%record_mf6_list_entry(ibinun, node, n2, rrate, &
636 if (ibudfl /= 0)
then
637 if (this%iprflow /= 0)
then
638 write (this%iout,
'(1x)')
657 if (this%inunit > 0)
then
658 do ip = 1,
size(this%ssmivec)
659 if (this%isrctype(ip) == 3 .or. this%isrctype(ip) == 4)
then
660 ssmiptr => this%ssmivec(ip)
661 call ssmiptr%spc_da()
664 deallocate (this%ssmivec)
668 if (this%inunit > 0)
then
671 this%ibound => null()
676 if (
associated(this%outputtab))
then
677 call this%outputtab%table_da()
678 deallocate (this%outputtab)
679 nullify (this%outputtab)
686 call this%NumericalPackageType%da()
700 call this%NumericalPackageType%allocate_scalars()
703 call mem_allocate(this%nbound,
'NBOUND', this%memoryPath)
719 integer(I4B) :: nflowpack
723 nflowpack = this%fmi%nflowpack
724 call mem_allocate(this%iauxpak, nflowpack,
'IAUXPAK', this%memoryPath)
725 call mem_allocate(this%isrctype, nflowpack,
'ISRCTYPE', this%memoryPath)
734 allocate (this%ssmivec(nflowpack))
745 character(len=LINELENGTH) :: keyword
747 logical :: isfound, endOfBlock
749 character(len=*),
parameter :: fmtiprflow = &
750 "(4x,'SSM FLOW INFORMATION WILL BE PRINTED TO LISTING FILE &
751 &WHENEVER ICBCFL IS NOT ZERO.')"
752 character(len=*),
parameter :: fmtisvflow = &
753 "(4x,'CELL-BY-CELL FLOW INFORMATION WILL BE SAVED TO BINARY FILE &
754 &WHENEVER ICBCFL IS NOT ZERO.')"
757 call this%parser%GetBlock(
'OPTIONS', isfound, ierr, blockrequired=.false., &
758 supportopenclose=.true.)
762 write (this%iout,
'(1x,a)')
'PROCESSING SSM OPTIONS'
764 call this%parser%GetNextLine(endofblock)
766 call this%parser%GetStringCaps(keyword)
767 select case (keyword)
770 write (this%iout, fmtiprflow)
773 write (this%iout, fmtisvflow)
775 write (
errmsg,
'(a,a)')
'Unknown SSM option: ', trim(keyword)
777 call this%parser%StoreErrorUnit()
780 write (this%iout,
'(1x,a)')
'END OF SSM OPTIONS'
793 call this%read_sources_aux()
796 call this%read_sources_fileinput()
808 character(len=LINELENGTH) :: keyword
809 character(len=20) :: srctype
812 integer(I4B) :: nflowpack
813 integer(I4B) :: isrctype
814 logical :: isfound, endOfBlock
821 nflowpack = this%fmi%nflowpack
824 call this%parser%GetBlock(
'SOURCES', isfound, ierr, &
825 supportopenclose=.true., &
826 blockrequired=.true.)
828 write (this%iout,
'(1x,a)')
'PROCESSING SOURCES'
830 call this%parser%GetNextLine(endofblock)
834 call this%parser%GetStringCaps(keyword)
837 if (trim(adjustl(this%fmi%gwfpackages(ip)%name)) == keyword)
then
842 if (.not. pakfound)
then
843 write (
errmsg,
'(a,a)')
'Flow package cannot be found: ', &
846 call this%parser%StoreErrorUnit()
850 if (this%isrctype(ip) /= 0)
then
851 write (
errmsg,
'(a, a)') &
852 'A package cannot be specified more than once in the SSM SOURCES &
853 &block. The following package was specified more than once: ', &
856 call this%parser%StoreErrorUnit()
860 call this%parser%GetStringCaps(srctype)
861 select case (srctype)
863 write (this%iout,
'(1x,a)')
'AUX SOURCE DETECTED.'
866 write (this%iout,
'(1x,a)')
'AUXMIXED SOURCE DETECTED.'
870 write (
errmsg,
'(a, a)') &
871 'SRCTYPE must be AUX or AUXMIXED. Found: ', trim(srctype)
873 call this%parser%StoreErrorUnit()
877 this%isrctype(ip) = isrctype
880 call this%set_iauxpak(ip, trim(keyword))
883 write (this%iout,
'(1x,a)')
'END PROCESSING SOURCES'
885 write (
errmsg,
'(a)')
'Required SOURCES block not found.'
887 call this%parser%StoreErrorUnit()
892 call this%parser%StoreErrorUnit()
905 character(len=LINELENGTH) :: keyword
906 character(len=LINELENGTH) :: keyword2
907 character(len=20) :: srctype
910 integer(I4B) :: nflowpack
911 integer(I4B) :: isrctype
912 logical :: isfound, endOfBlock
919 nflowpack = this%fmi%nflowpack
922 call this%parser%GetBlock(
'FILEINPUT', isfound, ierr, &
923 supportopenclose=.true., &
924 blockrequired=.false.)
926 write (this%iout,
'(1x,a)')
'PROCESSING FILEINPUT'
928 call this%parser%GetNextLine(endofblock)
932 call this%parser%GetStringCaps(keyword)
935 if (trim(adjustl(this%fmi%gwfpackages(ip)%name)) == keyword)
then
940 if (.not. pakfound)
then
941 write (
errmsg,
'(a,a)')
'Flow package cannot be found: ', &
944 call this%parser%StoreErrorUnit()
948 if (this%isrctype(ip) /= 0)
then
949 write (
errmsg,
'(a, a)') &
950 'A package cannot be specified more than once in the SSM SOURCES &
951 &and SOURCES_FILES blocks. The following package was specified &
952 &more than once: ', &
955 call this%parser%StoreErrorUnit()
959 call this%parser%GetStringCaps(srctype)
960 select case (srctype)
962 write (this%iout,
'(1x,a)')
'SPC6 SOURCE DETECTED.'
966 call this%parser%GetStringCaps(keyword2)
967 if (trim(adjustl(keyword2)) /=
'FILEIN')
then
968 errmsg =
'SPC6 keyword must be followed by "FILEIN" '// &
969 'then by filename and optionally by <MIXED>.'
971 call this%parser%StoreErrorUnit()
976 call this%set_ssmivec(ip, trim(keyword))
979 call this%parser%GetStringCaps(keyword2)
980 if (trim(keyword2) ==
'MIXED')
then
982 write (this%iout,
'(1x,a,a)')
'ASSIGNED MIXED SSM TYPE TO PACKAGE ', &
987 'SRCTYPE must be SPC6. Found: ', trim(srctype)
989 call this%parser%StoreErrorUnit()
993 this%isrctype(ip) = isrctype
996 write (this%iout,
'(1x,a)')
'END PROCESSING FILEINPUT'
998 write (this%iout,
'(1x,a)') &
999 'OPTIONAL FILEINPUT BLOCK NOT FOUND. CONTINUING.'
1004 call this%parser%StoreErrorUnit()
1019 integer(I4B),
intent(in) :: ip
1020 character(len=*),
intent(in) :: packname
1022 character(len=LENAUXNAME) :: auxname
1024 integer(I4B) :: iaux
1027 call this%parser%GetStringCaps(auxname)
1029 do iaux = 1, this%fmi%gwfpackages(ip)%naux
1030 if (trim(this%fmi%gwfpackages(ip)%auxname(iaux)) == &
1036 if (.not. auxfound)
then
1037 write (
errmsg,
'(a, a)') &
1038 'Auxiliary name cannot be found: ', trim(auxname)
1040 call this%parser%StoreErrorUnit()
1044 this%iauxpak(ip) = iaux
1045 write (this%iout,
'(4x, a, i0, a, a)')
'USING AUX COLUMN ', &
1046 iaux,
' IN PACKAGE ', trim(packname)
1060 integer(I4B),
intent(in) :: ip
1061 character(len=*),
intent(in) :: packname
1063 character(len=LINELENGTH) :: filename
1065 integer(I4B) :: inunit
1068 call this%parser%GetString(filename)
1070 call openfile(inunit, this%iout, filename,
'SPC', filstat_opt=
'OLD')
1073 ssmiptr => this%ssmivec(ip)
1074 call ssmiptr%initialize(this%dis, ip, inunit, this%iout, this%name_model, &
1075 trim(packname), this%depvartype)
1077 write (this%iout,
'(4x, a, a, a, a, a)')
'USING SPC INPUT FILE ', &
1078 trim(filename),
' TO SET ', trim(this%depvartype), &
1079 'S FOR PACKAGE ', trim(packname)
1090 character(len=LINELENGTH) :: title
1091 character(len=LINELENGTH) :: text
1092 integer(I4B) :: ntabcol
1095 if (this%iprflow /= 0)
then
1104 title =
'SSM PACKAGE ('//trim(this%packName)// &
1106 call table_cr(this%outputtab, this%packName, title)
1107 call this%outputtab%table_df(1, ntabcol, this%iout, transient=.true.)
1109 call this%outputtab%initialize_column(text, 10, alignment=
tabcenter)
1111 call this%outputtab%initialize_column(text, 20, alignment=
tableft)
1113 call this%outputtab%initialize_column(text, 15, alignment=
tabcenter)
1115 call this%outputtab%initialize_column(text, 15, alignment=
tabcenter)
1117 call this%outputtab%initialize_column(text, 15, alignment=
tabcenter)
1118 text =
'PACKAGE NAME'
1119 call this%outputtab%initialize_column(text, 16, alignment=
tabcenter)
This module contains the BudgetModule.
This module contains simulation constants.
integer(i4b), parameter linelength
maximum length of a standard line
@ tabcenter
centered table column
@ tableft
left justified table column
integer(i4b), parameter lenpackagename
maximum length of the package name
integer(i4b), parameter lenbudrowlabel
maximum length of the rowlabel string used in the budget table
integer(i4b), parameter lenvarname
maximum length of a variable name
integer(i4b), parameter lenftype
maximum length of a package type (DIS, WEL, OC, etc.)
integer(i4b), parameter lenauxname
maximum length of a aux variable
integer(i4b), parameter lenboundname
maximum length of a bound name
real(dp), parameter dzero
real constant zero
real(dp), parameter done
real constant 1
This module defines variable data types.
This module contains the base numerical package type.
This module contains simulation methods.
subroutine, public store_error(msg, terminate)
Store an error message.
integer(i4b) function, public count_errors()
Return number of errors.
subroutine, public store_error_unit(iunit, terminate)
Store the file unit number.
This module contains simulation variables.
character(len=maxcharlen) errmsg
error message string
subroutine, public table_cr(this, name, title)
integer(i4b), pointer, public kstp
current time step number
integer(i4b), pointer, public kper
current stress period number
real(dp), pointer, public delt
length of the current time step
This module contains the TspSpc Module.
This module contains the TspSsm Module.
subroutine ssm_fc(this, matrix_sln, idxglo, rhs)
@ brief Fill coefficients
subroutine ssm_ot_flow(this, icbcfl, ibudfl, icbcun)
@ brief Output flows
subroutine ssm_term(this, ipackage, ientry, rrate, rhsval, hcofval, cssm, qssm)
@ brief Calculate the SSM mass flow rate and hcof and rhs values
subroutine allocate_scalars(this)
@ brief Allocate scalars
subroutine ssm_cq(this, flowja)
@ brief Calculate flow
subroutine ssm_bd(this, isuppress_output, model_budget)
@ brief Calculate the global SSM budget terms
subroutine pak_setup_outputtab(this)
@ brief Setup the output table
subroutine ssm_df(this)
@ brief Define SSM Package
subroutine set_ssmivec(this, ip, packname)
@ brief Set ssmivec array value for package ip
subroutine set_iauxpak(this, ip, packname)
@ brief Set iauxpak array value for package ip
subroutine get_ssm_conc(this, ipackage, ientry, nbound_flow, conc, lauxmixed)
@ brief Provide bound concentration (or temperature) and mixed flag
subroutine ssm_rp(this)
@ brief Read and prepare this SSM Package
subroutine read_sources_fileinput(this)
@ brief Read FILEINPUT block
character(len=lenpackagename) text
subroutine ssm_ar(this, dis, ibound, cnew)
@ brief Allocate and read SSM Package
subroutine read_sources_aux(this)
@ brief Read SOURCES block
subroutine, public ssm_cr(ssmobj, name_model, inunit, iout, fmi, eqnsclfac, depvartype)
@ brief Create a new SSM package
subroutine allocate_arrays(this)
@ brief Allocate arrays
character(len=lenftype) ftype
subroutine ssm_ad(this)
@ brief Advance the SSM Package
subroutine read_options(this)
@ brief Read package options
subroutine ssm_da(this)
@ brief Deallocate
subroutine read_data(this)
@ brief Read package data
Derived type for the Budget object.
Derived type for managing SPC input.
Derived type for the SSM Package.