Source code for yt.frontends.amrvac.datfile_utils

import struct

import numpy as np

# Size of basic types (in bytes)
SIZE_LOGICAL = 4
SIZE_INT = 4
SIZE_DOUBLE = 8
NAME_LEN = 16

# For un-aligned data, use '=' (for aligned data set to '')
ALIGN = "="


[docs] def get_header(istream): """Read header from an MPI-AMRVAC 2.0 snapshot. istream' should be a file opened in binary mode. """ istream.seek(0) h = {} fmt = ALIGN + "i" [h["datfile_version"]] = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) if h["datfile_version"] < 3: raise OSError("Unsupported AMRVAC .dat file version: %d", h["datfile_version"]) # Read scalar data at beginning of file fmt = ALIGN + 9 * "i" + "d" hdr = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) [ h["offset_tree"], h["offset_blocks"], h["nw"], h["ndir"], h["ndim"], h["levmax"], h["nleafs"], h["nparents"], h["it"], h["time"], ] = hdr # Read min/max coordinates fmt = ALIGN + h["ndim"] * "d" h["xmin"] = np.array(struct.unpack(fmt, istream.read(struct.calcsize(fmt)))) h["xmax"] = np.array(struct.unpack(fmt, istream.read(struct.calcsize(fmt)))) # Read domain and block size (in number of cells) fmt = ALIGN + h["ndim"] * "i" h["domain_nx"] = np.array(struct.unpack(fmt, istream.read(struct.calcsize(fmt)))) h["block_nx"] = np.array(struct.unpack(fmt, istream.read(struct.calcsize(fmt)))) if h["datfile_version"] >= 5: # Read periodicity fmt = ALIGN + h["ndim"] * "i" # Fortran logical is 4 byte int h["periodic"] = np.array( struct.unpack(fmt, istream.read(struct.calcsize(fmt))), dtype=bool ) # Read geometry name fmt = ALIGN + NAME_LEN * "c" hdr = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) h["geometry"] = b"".join(hdr).strip().decode() # Read staggered flag fmt = ALIGN + "i" # Fortran logical is 4 byte int h["staggered"] = bool(struct.unpack(fmt, istream.read(struct.calcsize(fmt)))[0]) # Read w_names w_names = [] for _ in range(h["nw"]): fmt = ALIGN + NAME_LEN * "c" hdr = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) w_names.append(b"".join(hdr).strip().decode()) h["w_names"] = w_names # Read physics type fmt = ALIGN + NAME_LEN * "c" hdr = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) h["physics_type"] = b"".join(hdr).strip().decode() # Read number of physics-defined parameters fmt = ALIGN + "i" [n_pars] = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) # First physics-parameter values are given, then their names fmt = ALIGN + n_pars * "d" vals = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) fmt = ALIGN + n_pars * NAME_LEN * "c" names = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) # Split and join the name strings (from one character array) names = [ b"".join(names[i : i + NAME_LEN]).strip().decode() for i in range(0, len(names), NAME_LEN) ] # Store the values corresponding to the names for val, name in zip(vals, names, strict=True): h[name] = val return h
[docs] def get_tree_info(istream): """ Read levels, morton-curve indices, and byte offsets for each block as stored in the datfile istream is an open datfile buffer with 'rb' mode This can be used as the "first pass" data reading required by YT's interface. """ istream.seek(0) header = get_header(istream) nleafs = header["nleafs"] nparents = header["nparents"] # Read tree info. Skip 'leaf' array istream.seek(header["offset_tree"] + (nleafs + nparents) * SIZE_LOGICAL) # Read block levels fmt = ALIGN + nleafs * "i" block_lvls = np.array(struct.unpack(fmt, istream.read(struct.calcsize(fmt)))) # Read block indices fmt = ALIGN + nleafs * header["ndim"] * "i" block_ixs = np.reshape( struct.unpack(fmt, istream.read(struct.calcsize(fmt))), [nleafs, header["ndim"]] ) # Read block offsets (skip ghost cells !) bcfmt = ALIGN + header["ndim"] * "i" bcsize = struct.calcsize(bcfmt) * 2 fmt = ALIGN + nleafs * "q" block_offsets = ( np.array(struct.unpack(fmt, istream.read(struct.calcsize(fmt)))) + bcsize ) return block_lvls, block_ixs, block_offsets
[docs] def get_single_block_data(istream, byte_offset, block_shape): """retrieve a specific block (all fields) from a datfile""" istream.seek(byte_offset) # Read actual data fmt = ALIGN + np.prod(block_shape) * "d" d = struct.unpack(fmt, istream.read(struct.calcsize(fmt))) # Fortran ordering block_data = np.reshape(d, block_shape, order="F") return block_data
[docs] def get_single_block_field_data(istream, byte_offset, block_shape, field_idx): """retrieve a specific block (ONE field) from a datfile""" # compute byte size of a single field field_shape = block_shape[:-1] fmt = ALIGN + np.prod(field_shape) * "d" byte_size_field = struct.calcsize(fmt) istream.seek(byte_offset + byte_size_field * field_idx) data = np.fromfile(istream, "=f8", count=np.prod(field_shape)) data.shape = field_shape[::-1] return data.T