separating functionality for more flexibility

This commit is contained in:
Martin Diehl 2022-02-27 17:54:11 +01:00
parent efa30d1f3a
commit 7ee440c1b1
3 changed files with 151 additions and 82 deletions

View File

@ -440,7 +440,7 @@ subroutine IO_error(error_ID,el,ip,g,instance,ext_msg)
case (155) case (155)
msg = 'material index out of bounds' msg = 'material index out of bounds'
case (180) case (180)
msg = 'missing/invalid material definition via State Variable 2' msg = 'missing/invalid material definition'
case (190) case (190)
msg = 'unknown element type:' msg = 'unknown element type:'
case (191) case (191)

View File

@ -11,24 +11,117 @@ module VTK
implicit none implicit none
private private
public :: VTK_readVTI public :: &
VTK_readVTIdataset_int, &
VTK_readVTI_cellsSizeOrigin
contains contains
!-------------------------------------------------------------------------------------------------- !--------------------------------------------------------------------------------------------------
!> @brief Parse vtk image data (.vti) !> @brief Read integer dataset from a VTK image data (*.vti) file.
!> @details https://vtk.org/Wiki/VTK_XML_Formats !> @details https://vtk.org/Wiki/VTK_XML_Formats
!-------------------------------------------------------------------------------------------------- !--------------------------------------------------------------------------------------------------
subroutine VTK_readVTI(cells,geomSize,origin,material, & function VTK_readVTIdataset_int(fileContent,label) result(dataset)
fileContent)
character(len=*), intent(in) :: &
label, &
fileContent
integer, dimension(:), allocatable :: &
dataset
character(len=:), allocatable :: dataType, headerType, base64_str
logical :: compressed
call VTK_readVTIdataset_raw(base64_str,dataType,headerType,compressed, &
fileContent,label)
dataset = as_Int(base64_str,headerType,compressed,dataType)
if (.not. allocated(dataset)) call IO_error(error_ID = 844, ext_msg='dataset "'//label//'" not found')
end function VTK_readVTIdataset_int
!--------------------------------------------------------------------------------------------------
!> @brief Read dataset as raw data (base64 string) from a VTK image data (*.vti) file.
!> @details https://vtk.org/Wiki/VTK_XML_Formats
!--------------------------------------------------------------------------------------------------
subroutine VTK_readVTIdataset_raw(base64_str,dataType,headerType,compressed, &
fileContent,label)
character(len=*), intent(in) :: &
label, &
fileContent
character(len=:), allocatable, intent(out) :: dataType, headerType, base64_str
logical, intent(out) :: compressed
logical :: inFile,inImage,gotCellData
integer(pI64) :: &
startPos, endPos, &
s
inFile = .false.
inImage = .false.
startPos = 1_pI64
outer: do while (startPos < len(fileContent,kind=pI64))
endPos = startPos + index(fileContent(startPos:),IO_EOL,kind=pI64) - 2_pI64
if (endPos < startPos) endPos = len(fileContent,kind=pI64) ! end of file without new line
if (.not. inFile) then
if (index(fileContent(startPos:endPos),'<VTKFile',kind=pI64) /= 0_pI64) then
inFile = .true.
if (.not. fileFormatOk(fileContent(startPos:endPos))) call IO_error(error_ID = 844, ext_msg='file format')
headerType = merge('UInt64','UInt32',getXMLValue(fileContent(startPos:endPos),'header_type')=='UInt64')
compressed = getXMLValue(fileContent(startPos:endPos),'compressor') == 'vtkZLibDataCompressor'
end if
else
if (.not. inImage) then
if (index(fileContent(startPos:endPos),'<ImageData',kind=pI64) /= 0_pI64) then
inImage = .true.
end if
else
if (index(fileContent(startPos:endPos),'<CellData',kind=pI64) /= 0_pI64) then
do while (index(fileContent(startPos:endPos),'</CellData>',kind=pI64) == 0_pI64)
if (index(fileContent(startPos:endPos),'<DataArray',kind=pI64) /= 0_pI64 .and. &
getXMLValue(fileContent(startPos:endPos),'Name') == label ) then
if (getXMLValue(fileContent(startPos:endPos),'format') /= 'binary') &
call IO_error(error_ID = 844, ext_msg='format ('//label//')')
dataType = getXMLValue(fileContent(startPos:endPos),'type')
startPos = endPos + 2_pI64
endPos = startPos + index(fileContent(startPos:),IO_EOL,kind=pI64) - 2_pI64
s = startPos + verify(fileContent(startPos:endPos),IO_WHITESPACE,kind=pI64) -1_pI64 ! start (no leading whitespace)
base64_str = fileContent(s:endPos)
exit outer
end if
startPos = endPos + 2_pI64
endPos = startPos + index(fileContent(startPos:),IO_EOL,kind=pI64) - 2_pI64
end do
end if
end if
end if
startPos = endPos + 2_pI64
end do outer
end subroutine VTK_readVTIdataset_raw
!--------------------------------------------------------------------------------------------------
!> @brief Read cells, size, and origin of an VTK image data (*.vti) file.
!> @details https://vtk.org/Wiki/VTK_XML_Formats
!--------------------------------------------------------------------------------------------------
subroutine VTK_readVTI_cellsSizeOrigin(cells,geomSize,origin, &
fileContent)
integer, dimension(3), intent(out) :: & integer, dimension(3), intent(out) :: &
cells ! # of cells (across all processes!) cells ! # of cells (across all processes!)
real(pReal), dimension(3), intent(out) :: & real(pReal), dimension(3), intent(out) :: &
geomSize, & ! physical size (across all processes!) geomSize, & ! size (across all processes!)
origin ! origin (across all processes!) origin ! origin (across all processes!)
integer, dimension(:), intent(out), allocatable :: &
material
character(len=*), intent(in) :: & character(len=*), intent(in) :: &
fileContent fileContent
@ -42,14 +135,10 @@ subroutine VTK_readVTI(cells,geomSize,origin,material, &
cells = -1 cells = -1
geomSize = -1.0_pReal geomSize = -1.0_pReal
inFile = .false. inFile = .false.
inImage = .false. inImage = .false.
gotCelldata = .false.
!--------------------------------------------------------------------------------------------------
! parse XML file
startPos = 1_pI64 startPos = 1_pI64
do while (startPos < len(fileContent,kind=pI64)) outer: do while (startPos < len(fileContent,kind=pI64))
endPos = startPos + index(fileContent(startPos:),IO_EOL,kind=pI64) - 2_pI64 endPos = startPos + index(fileContent(startPos:),IO_EOL,kind=pI64) - 2_pI64
if (endPos < startPos) endPos = len(fileContent,kind=pI64) ! end of file without new line if (endPos < startPos) endPos = len(fileContent,kind=pI64) ! end of file without new line
@ -65,49 +154,24 @@ subroutine VTK_readVTI(cells,geomSize,origin,material, &
if (index(fileContent(startPos:endPos),'<ImageData',kind=pI64) /= 0_pI64) then if (index(fileContent(startPos:endPos),'<ImageData',kind=pI64) /= 0_pI64) then
inImage = .true. inImage = .true.
call cellsSizeOrigin(cells,geomSize,origin,fileContent(startPos:endPos)) call cellsSizeOrigin(cells,geomSize,origin,fileContent(startPos:endPos))
end if exit outer
else
if (index(fileContent(startPos:endPos),'<CellData',kind=pI64) /= 0_pI64) then
gotCellData = .true.
do while (index(fileContent(startPos:endPos),'</CellData>',kind=pI64) == 0_pI64)
if (index(fileContent(startPos:endPos),'<DataArray',kind=pI64) /= 0_pI64 .and. &
getXMLValue(fileContent(startPos:endPos),'Name') == 'material' ) then
if (getXMLValue(fileContent(startPos:endPos),'format') /= 'binary') &
call IO_error(error_ID = 844, ext_msg='format (material)')
dataType = getXMLValue(fileContent(startPos:endPos),'type')
startPos = endPos + 2_pI64
endPos = startPos + index(fileContent(startPos:),IO_EOL,kind=pI64) - 2_pI64
s = startPos + verify(fileContent(startPos:endPos),IO_WHITESPACE,kind=pI64) -1_pI64 ! start (no leading whitespace)
material = as_Int(fileContent(s:endPos),headerType,compressed,dataType)
exit
end if
startPos = endPos + 2_pI64
endPos = startPos + index(fileContent(startPos:),IO_EOL,kind=pI64) - 2_pI64
end do
end if end if
end if end if
end if end if
if (gotCellData) exit
startPos = endPos + 2_pI64 startPos = endPos + 2_pI64
end do end do outer
if (.not. allocated(material)) call IO_error(error_ID = 844, ext_msg='material data not found')
if (size(material) /= product(cells)) call IO_error(error_ID = 844, ext_msg='size(material)')
if (any(geomSize<=0)) call IO_error(error_ID = 844, ext_msg='size') if (any(geomSize<=0)) call IO_error(error_ID = 844, ext_msg='size')
if (any(cells<1)) call IO_error(error_ID = 844, ext_msg='cells') if (any(cells<1)) call IO_error(error_ID = 844, ext_msg='cells')
material = material + 1
if (any(material<1)) call IO_error(error_ID = 844, ext_msg='material ID < 0')
end subroutine VTK_readVTI end subroutine VTK_readVTI_cellsSizeOrigin
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief determine size and origin from coordinates !> @brief Determine size and origin from coordinates.
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
subroutine cellsSizeOrigin(c,s,o,header) subroutine cellsSizeOrigin(c,s,o,header)
integer, dimension(3), intent(out) :: c integer, dimension(3), intent(out) :: c
@ -120,7 +184,7 @@ subroutine cellsSizeOrigin(c,s,o,header)
temp = getXMLValue(header,'Direction') temp = getXMLValue(header,'Direction')
if (temp /= '1 0 0 0 1 0 0 0 1' .and. temp /= '') & ! https://discourse.vtk.org/t/vti-specification/6526 if (temp /= '1 0 0 0 1 0 0 0 1' .and. temp /= '') & ! https://discourse.vtk.org/t/vti-specification/6526
call IO_error(error_ID = 844, ext_msg = 'coordinate order') call IO_error(error_ID = 844, ext_msg = 'coordinate order')
temp = getXMLValue(header,'WholeExtent') temp = getXMLValue(header,'WholeExtent')
@ -138,15 +202,15 @@ subroutine cellsSizeOrigin(c,s,o,header)
end subroutine end subroutine
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief Interpret Base64 string in vtk XML file as integer of default kind !> @brief Interpret Base64 string in vtk XML file as integer of default kind.
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
function as_Int(base64_str,headerType,compressed,dataType) function as_Int(base64_str,headerType,compressed,dataType)
character(len=*), intent(in) :: base64_str, & ! base64 encoded string character(len=*), intent(in) :: base64_str, & ! base64 encoded string
headerType, & ! header type (UInt32 or Uint64) headerType, & ! header type (UInt32 or Uint64)
dataType ! data type (Int32, Int64, Float32, Float64) dataType ! data type (Int32, Int64, Float32, Float64)
logical, intent(in) :: compressed ! indicate whether data is zlib compressed logical, intent(in) :: compressed ! indicate whether data is zlib compressed
integer, dimension(:), allocatable :: as_Int integer, dimension(:), allocatable :: as_Int
@ -167,15 +231,15 @@ function as_Int(base64_str,headerType,compressed,dataType)
end function as_Int end function as_Int
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief Interpret Base64 string in vtk XML file as integer of pReal kind !> @brief Interpret Base64 string in vtk XML file as real of kind pReal.
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
function as_pReal(base64_str,headerType,compressed,dataType) function as_pReal(base64_str,headerType,compressed,dataType)
character(len=*), intent(in) :: base64_str, & ! base64 encoded string character(len=*), intent(in) :: base64_str, & ! base64 encoded string
headerType, & ! header type (UInt32 or Uint64) headerType, & ! header type (UInt32 or Uint64)
dataType ! data type (Int32, Int64, Float32, Float64) dataType ! data type (Int32, Int64, Float32, Float64)
logical, intent(in) :: compressed ! indicate whether data is zlib compressed logical, intent(in) :: compressed ! indicate whether data is zlib compressed
real(pReal), dimension(:), allocatable :: as_pReal real(pReal), dimension(:), allocatable :: as_pReal
@ -196,14 +260,14 @@ function as_pReal(base64_str,headerType,compressed,dataType)
end function as_pReal end function as_pReal
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief Interpret Base64 string in vtk XML file as bytes !> @brief Interpret Base64 string in vtk XML file as bytes.
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
function asBytes(base64_str,headerType,compressed) result(bytes) function asBytes(base64_str,headerType,compressed) result(bytes)
character(len=*), intent(in) :: base64_str, & ! base64 encoded string character(len=*), intent(in) :: base64_str, & ! base64 encoded string
headerType ! header type (UInt32 or Uint64) headerType ! header type (UInt32 or Uint64)
logical, intent(in) :: compressed ! indicate whether data is zlib compressed logical, intent(in) :: compressed ! indicate whether data is zlib compressed
integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes
@ -217,19 +281,19 @@ function asBytes(base64_str,headerType,compressed) result(bytes)
end function asBytes end function asBytes
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief Interpret compressed Base64 string in vtk XML file as bytes !> @brief Interpret compressed Base64 string in vtk XML file as bytes.
!> @details A compressed Base64 string consists of a header block and a data block !> @details A compressed Base64 string consists of a header block and a data block
! [#blocks/#u-size/#p-size/#c-size-1/#c-size-2/.../#c-size-#blocks][DATA-1/DATA-2...] ! [#blocks/#u-size/#p-size/#c-size-1/#c-size-2/.../#c-size-#blocks][DATA-1/DATA-2...]
! #blocks = Number of blocks ! #blocks = Number of blocks
! #u-size = Block size before compression ! #u-size = Block size before compression
! #p-size = Size of last partial block (zero if it not needed) ! #p-size = Size of last partial block (zero if it not needed)
! #c-size-i = Size in bytes of block i after compression ! #c-size-i = Size in bytes of block i after compression
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
function asBytes_compressed(base64_str,headerType) result(bytes) function asBytes_compressed(base64_str,headerType) result(bytes)
character(len=*), intent(in) :: base64_str, & ! base64 encoded string character(len=*), intent(in) :: base64_str, & ! base64 encoded string
headerType ! header type (UInt32 or Uint64) headerType ! header type (UInt32 or Uint64)
integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes
integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes_inflated integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes_inflated
@ -265,15 +329,15 @@ function asBytes_compressed(base64_str,headerType) result(bytes)
end function asBytes_compressed end function asBytes_compressed
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief Interprete uncompressed Base64 string in vtk XML file as bytes !> @brief Interprete uncompressed Base64 string in vtk XML file as bytes.
!> @details An uncompressed Base64 string consists of N headers blocks and a N data blocks !> @details An uncompressed Base64 string consists of N headers blocks and a N data blocks
![#bytes-1/DATA-1][#bytes-2/DATA-2]... ![#bytes-1/DATA-1][#bytes-2/DATA-2]...
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
function asBytes_uncompressed(base64_str,headerType) result(bytes) function asBytes_uncompressed(base64_str,headerType) result(bytes)
character(len=*), intent(in) :: base64_str, & ! base64 encoded string character(len=*), intent(in) :: base64_str, & ! base64 encoded string
headerType ! header type (UInt32 or Uint64) headerType ! header type (UInt32 or Uint64)
integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes integer(C_SIGNED_CHAR), dimension(:), allocatable :: bytes
integer(pI64) :: s integer(pI64) :: s
@ -300,9 +364,9 @@ function asBytes_uncompressed(base64_str,headerType) result(bytes)
end function asBytes_uncompressed end function asBytes_uncompressed
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief Get XML string value for given key. !> @brief Get XML string value for given key.
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
pure function getXMLValue(line,key) pure function getXMLValue(line,key)
character(len=*), intent(in) :: line, key character(len=*), intent(in) :: line, key
@ -337,9 +401,9 @@ pure function getXMLValue(line,key)
end function end function
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
!> @brief Check for supported file format variants. !> @brief Check for supported file format variants.
!------------------------------------------------------------------------------------------------ !--------------------------------------------------------------------------------------------------
pure function fileFormatOk(line) pure function fileFormatOk(line)
character(len=*),intent(in) :: line character(len=*),intent(in) :: line

View File

@ -76,7 +76,12 @@ subroutine discretization_grid_init(restart)
if (worldrank == 0) then if (worldrank == 0) then
fileContent = IO_read(interface_geomFile) fileContent = IO_read(interface_geomFile)
call VTK_readVTI(cells,geomSize,origin,materialAt_global,fileContent) call VTK_readVTI_cellsSizeOrigin(cells,geomSize,origin,fileContent)
materialAt_global = VTK_readVTIdataset_int(fileContent,'material') + 1
if (any(materialAt_global < 1)) &
call IO_error(180,ext_msg='material ID < 1')
if (size(materialAt_global) /= product(cells)) &
call IO_error(180,ext_msg='mismatch of material IDs and # cells')
fname = interface_geomFile fname = interface_geomFile
if (scan(fname,'/') /= 0) fname = fname(scan(fname,'/',.true.)+1:) if (scan(fname,'/') /= 0) fname = fname(scan(fname,'/',.true.)+1:)
call results_openJobFile(parallel=.false.) call results_openJobFile(parallel=.false.)