From e16f47b486988ee88cceff988c34d7ff45f00163 Mon Sep 17 00:00:00 2001 From: Allen Byrne <50328838+byrnHDF@users.noreply.github.com> Date: Fri, 22 Nov 2024 08:39:03 -0600 Subject: [PATCH 01/12] Add unicode doc to doxygen Tech Notes (#5136) --- doxygen/dox/TechnicalNotes.dox | 147 ++++++++++++++++++++++++++++++++- 1 file changed, 146 insertions(+), 1 deletion(-) diff --git a/doxygen/dox/TechnicalNotes.dox b/doxygen/dox/TechnicalNotes.dox index 3ea6af63a25..61ddfbd3398 100644 --- a/doxygen/dox/TechnicalNotes.dox +++ b/doxygen/dox/TechnicalNotes.dox @@ -11,6 +11,7 @@ \li \ref SWMR \li \ref VDS \li \ref RELVERSION +\li \ref UNICODE \li \ref VFL \li HDF5 Library Architecture Overview \li \ref VOL_Connector @@ -89,7 +90,7 @@ A beneficial side effect of using SWMR access is better fault tolerance. It is m
#H5FD_SEC2 | This is the default driver which uses Posix file-system functions +like read and write to perform I/O to a single file. All I/O requests are unbuffered +although the driver does optimize file seeking operations to some extent. + | +
#H5FD_STDIO | This driver uses functions from 'stdio.h' to perform buffered I/O to a single file. + | +
#H5FD_CORE | This driver performs I/O directly to memory and can be +used to create small temporary files that never exist on permanent storage. This +type of storage is generally very fast since the I/O consists only of memory-to-memory copy operations. + | +
#H5FD_MPIO | This is the driver of choice for accessing files in parallel +using MPI and MPI-IO. It is only predefined if the library is compiled with parallel I/O support. + | +
#H5FD_FAMILY | Large format address spaces are partitioned into more +manageable pieces and sent to separate storage locations using an underlying driver +of the user's choice. \ref H5TOOL_RT_UG can be used to change the sizes of the family +members when stored as files or to convert a family of files to a single file or vice versa. + | +
static H5FD_t * open (const char *name, unsigned flags, hid_t fapl, haddr_t maxaddr) |
+The file name name and file access property list fapl are the same as were specified in the #H5Fcreate +or #H5Fopen call. The flags are the same as in those calls also except the flag #H5F_ACC_CREAT is also +present if the call was to H5Fcreate and they are documented in the 'H5Fpublic.h' file. The maxaddr +argument is the maximum format address that the driver should be prepared to handle (the minimum address is always zero). | +
static herr_t close (H5FD_t *file) |
+The file argument is the handle which was returned by the open function, and the close should +free only memory associated with the driver-specific part of the handle (the public parts will +have already been released by HDF5's virtual file layer). | +
const int cmp (const H5FD_t *f1, const H5FD_t *f2) |
+The driver may provide a function which compares two files f1 and f2 belonging to the same +driver and returns a negative, positive, or zero value a la the strcmp function.(The ordering +is arbitrary as long as it's consistent within a particular file driver.) If this function is +not provided then HDF5 assumes that all calls to the open callback return unique files regardless +of the arguments and it is up to the application to avoid doing this if that assumption is incorrect. | +
Function | +Description | +
---|---|
static hsize_t sb_size (H5FD_t *file) |
+The sb_size function returns the number of bytes necessary to encode +information needed later if the file is reopened. | +
static herr_t sb_encode (H5FD_t *file, char *name, unsigned char *buf) |
+The sb_encode function encodes information from the file into buffer buf +allocated by the caller. It also writes an 8-character (plus null termination) into +the name argument, which should be a unique identification for the driver. | +
static herr_t sb_decode (H5FD_t *file, const char *name, const unsigned char *buf) |
+The sb_decode function looks at the name decodes data from the buffer buf and +updates the file argument with the new information, advancing *p in the process. | +
#H5FD_MEM_SUPER | userblock | +
#H5FD_MEM_BTREE | An allocation request for a node of a B-tree. + | +
#H5FD_MEM_DRAW | An allocation request for the raw data of a dataset. + | +
#H5FD_MEM_GHEAP | An allocation request for a global heap collection. Global +heaps are used to store certain types of references such as dataset region references. +The set of all global heap collections can become quite large. + | +
#H5FD_MEM_LHEAP | An allocation request for a local heap. Local heaps are used +to store the names which are members of a group. The combined size of all local heaps is +a function of the number of object names in the file. + | +
#H5FD_MEM_OHDR | An allocation request for (part of) an object header. Object +headers are relatively small and include meta information about objects (like the data +space and type of a dataset) and attributes. + | +
#H5FD_FLMAP_SINGLE | All memory usage types are mapped to a single free list. + | +
#H5FD_FLMAP_DICHOTOMY | Memory usage is segregated into meta data and raw data +for the purposes of memory management. + | +
#H5FD_FLMAP_DEFAULT | Each memory usage type has its own free list. + | +
static haddr_t alloc (H5FD_t *file, H5MF_type_t type, hsize_t size) |
+The file argument is the file from which space is to be allocated, type is the type of +memory being requested (from the list above) without being mapped according to the freelist +map and size is the number of bytes being requested. The library is allowed to allocate large +chunks of storage and manage them in a layer above the file driver (although the current library +doesn't do that). The allocation function should return a format address for the first byte +allocated. The allocated region extends from that address for size bytes. If the request cannot +be honored then the undefined address value is returned (#HADDR_UNDEF). The first call to this +function for a file which has never had memory allocated must return a format address of zero +or #HADDR_UNDEF since this is how the library allocates space for the userblock and/or superblock. | +
static herr_t free (H5FD_t *file, H5MF_type_t type, haddr_t addr, hsize_t size) |
+The file argument is the file for which space is being freed; type is the type of object being +freed (from the list above) without being mapped according to the freelist map; addr is the first +format address to free; and size is the size in bytes of the region being freed. The region being +freed may refer to just part of the region originally allocated and/or may cross allocation boundaries +provided all regions being freed have the same usage type. However, the library will never attempt +to free regions which have already been freed or which have never been allocated. | +
static haddr_t get_eoa (H5FD_t *file) |
+This function returns the current value of the EOA marker for the specified file. | +
static herr_t read (H5FD_t *file, H5FD_mem_t type, hid_t dxpl, haddr_t addr, hsize_t size, void *buf) |
+The read function reads data from file file beginning at address addr and continuing +for size bytes into the buffer buf supplied by the caller. | +
static herr_t write (H5FD_t *file, H5FD_mem_t type, hid_t dxpl, haddr_t addr, hsize_t size, const void *buf) |
+The write function transfers data +in the opposite direction. | +
static herr_t flush (H5FD_t *file) |
+Flush all data for file file to storage. | +
static herr_t query (const H5FD_t *file, unsigned long *flags) |
+This function is called by the library to query which optimizations to enable for I/O to this driver. | +
H5FD_FEAT_AGGREGATE_METADATA (0x00000001) |
+Defining the H5FD_FEAT_AGGREGATE_METADATA for a VFL driver means that the library will attempt to allocate +a larger block for metadata and then sub-allocate each metadata request from that larger block. | +
H5FD_FEAT_ACCUMULATE_METADATA (0x00000002) |
+Defining the H5FD_FEAT_ACCUMULATE_METADATA for a VFL driver means that the library will attempt to cache +metadata as it is written to the file and build up a larger block of metadata to eventually pass to the +VFL 'write' routine. | +
H5FD_FEAT_DATA_SIEVE (0x00000004) |
+Defining the H5FD_FEAT_DATA_SIEVE for a VFL driver means that the library will attempt to cache raw data + as it is read from/written to a file in a "data sieve" buffer. | +
hid_t H5FDregister (H5FD_class_t *cls) |
+The driver described by struct cls is registered with the library and an ID number for the driver is returned. | +
const char *name |
+A pointer to a constant, null-terminated driver name to be used for debugging purposes. | +
size_t fapl_size |
+The size in bytes of the file access mode structure or zero if the driver supplies a copy function +or doesn't define the structure. | +
void *(*fapl_copy)(const void *fapl) |
+An optional function which copies a driver-defined file access mode structure. This field takes +precedence over fm_size when both are defined. | +
void (*fapl_free)(void *fapl) |
+An optional function to free the driver-defined file access mode structure. If null, then the +library calls the C free function to free the structure. | +
size_t dxpl_size |
+The size in bytes of the data transfer mode structure or zero if the driver supplies a copy +function or doesn't define the structure. | +
void *(*dxpl_copy)(const void *dxpl) |
+An optional function which copies a driver-defined data transfer mode structure. This field +takes precedence over xm_size when both are defined. | +
void (*dxpl_free)(void *dxpl) |
+An optional function to free the driver-defined data transfer mode structure. If null, then +the library calls the C free function to free the structure. | +
H5FD_t *(*open)(const char *name, unsigned flags, hid_t fapl, haddr_t maxaddr) |
+The function which opens or creates a new file. | +
herr_t (*close)(H5FD_t *file) |
+The function which ends access to a file. | +
int (*cmp)(const H5FD_t *f1, const H5FD_t *f2) |
+An optional function to determine whether two open files have the same key. If this function +is not present then the library assumes that two files will never be the same. | +
int (*query)(const H5FD_t *f, unsigned long *flags) |
+An optional function to determine which library optimizations a driver can support. | +
haddr_t (*alloc)(H5FD_t *file, H5FD_mem_t type, hsize_t size) |
+An optional function to allocate space in the file. | +
herr_t (*free)(H5FD_t *file, H5FD_mem_t type, haddr_t addr, hsize_t size) |
+An optional function to free space in the file. | +
haddr_t (*get_eoa)(H5FD_t *file) |
+A function to query how much of the format address space has been allocated. | +
herr_t (*set_eoa)(H5FD_t *file, haddr_t) |
+A function to set the end of address space. | +
haddr_t (*get_eof)(H5FD_t *file) |
+A function to return the current end-of-file marker value. | +
herr_t (*read)(H5FD_t *file, H5FD_mem_t type, hid_t dxpl, haddr_t addr, hsize_t size, void *buffer) |
+A function to read data from a file. | +
herr_t (*write)(H5FD_t *file, H5FD_mem_t type, hid_t dxpl, haddr_t addr, hsize_t size, const void *buffer) |
+A function to write data to a file. | +
herr_t (*flush)(H5FD_t *file) |
+A function which flushes cached data to the file. | +
H5FD_mem_t fl_map[H5FD_MEM_NTYPES] |
+An array which maps a file allocation request type to a free list. | +
herr_t H5Dunregister (hid_t driver) |
+Where driver is the ID number returned when the driver was registered. | +
void * H5Pget_driver_data (hid_t fapl) |
+This function is intended to be used by driver functions, not applications. It returns a pointer +directly into the file access property list fapl which is a copy of the driver's file access mode +originally provided to the H5Pset_driver function. If its argument is a data transfer property list +fxpl then it returns a pointer to the driver-specific data transfer information instead. + | +
Initial document, 18 November 1999.
- -Updated on 10/24/00, Quincey Koziol
- -Added the section “Programming Note for C++ Developers Using C -Functions,” 08/23/2012, Mark Evans - - - -
-
-
- - -
-The HDF5 file format describes how HDF5 data structures and dataset raw -data are mapped to a linear format address space and the HDF5 -library implements that bidirectional mapping in terms of an -API. However, the HDF5 format specifications do not indicate how -the format address space is mapped onto storage and HDF (version 5 and -earlier) simply mapped the format address space directly onto a single -file by convention. - -
--Since early versions of HDF5 it became apparent that users want the ability to -map the format address space onto different types of storage (a single file, -multiple files, local memory, global memory, network distributed global -memory, a network protocol, etc.) with various types of maps. For -instance, some users want to be able to handle very large format address -spaces on operating systems that support only 2GB files by partitioning the -format address space into equal-sized parts each served by a separate -file. Other users want the same multi-file storage capability but want to -partition the address space according to purpose (raw data in one file, object -headers in another, global heap in a third, etc.) in order to improve I/O -speeds. - -
--In fact, the number of storage variations is probably larger than the -number of methods that the HDF5 team is capable of implementing and -supporting. Therefore, a Virtual File Layer API is being -implemented which will allow application teams or departments to design -and implement their own mapping between the HDF5 format address space -and storage, with each mapping being a separate file driver -(possibly written in terms of other file drivers). The HDF5 team will -provide a small set of useful file drivers which will also serve as -examples for those who which to write their own: - -
-H5FD_SEC2
-read
and write
to perform I/O to a single file. All I/O
-requests are unbuffered although the driver does optimize file seeking
-operations to some extent.
-
-H5FD_STDIO
-H5FD_CORE
-H5FD_MPIIO
-H5FD_FAMILY
-h5repart
tool can be used to change the sizes of the
-family members when stored as files or to convert a family of files to a
-single file or vice versa.
-
-H5FD_SPLIT
--Most application writers will use a driver defined by the HDF5 library or -contributed by another programming team. This chapter describes how existing -drivers are used. - -
- - - --Each file driver is defined in its own public header file which should -be included by any application which plans to use that driver. The -predefined drivers are in header files whose names begin with -`H5FD' followed by the driver name and `.h'. The `hdf5.h' -header file includes all the predefined driver header files. - -
-
-Once the appropriate header file is included a symbol of the form
-`H5FD_' followed by the upper-case driver name will be the driver
-identification number.(1) However, the
-value may change if the library is closed (e.g., by calling
-H5close
) and the symbol is referenced again.
-
-
-In order to create or open a file one must define the method by which the
-storage is accessed(2) and does so by creating a file access property list(3) which is passed to the H5Fcreate
or
-H5Fopen
function. A default file access property list is created by
-calling H5Pcreate
and then the file driver information is inserted by
-calling a driver initialization function such as H5Pset_fapl_family
:
-
-
-hid_t fapl = H5Pcreate(H5P_FILE_ACCESS); -size_t member_size = 100*1024*1024; /*100MB*/ -H5Pset_fapl_family(fapl, member_size, H5P_DEFAULT); -hid_t file = H5Fcreate("foo%05d.h5", H5F_ACC_TRUNC, H5P_DEFAULT, fapl); -H5Pclose(fapl); -- -
-Each file driver will have its own initialization function
-whose name is H5Pset_fapl_
followed by the driver name and which
-takes a file access property list as the first argument followed by
-additional driver-dependent arguments.
-
-
-An alternative to using the driver initialization function is to set the
-driver directly using the H5Pset_driver
function.(4) Its second argument is the file driver identifier, which may
-have a different numeric value from run to run depending on the order in which
-the file drivers are registered with the library. The third argument
-encapsulates the additional arguments of the driver initialization
-function. This method only works if the file driver writer has made the
-driver-specific property list structure a public datatype, which is
-often not the case.
-
-
-hid_t fapl = H5Pcreate(H5P_FILE_ACCESS); -static H5FD_family_fapl_t fa = {100*1024*1024, H5P_DEFAULT}; -H5Pset_driver(fapl, H5FD_FAMILY, &fa); -hid_t file = H5Fcreate("foo.h5", H5F_ACC_TRUNC, H5P_DEFAULT, fapl); -H5Pclose(fapl); -- -
-It is also possible to query the file driver information from a file access
-property list by calling H5Pget_driver
to determine the driver and then
-calling a driver-defined query function to obtain the driver information:
-
-
-hid_t driver = H5Pget_driver(fapl); -if (H5FD_SEC2==driver) { - /*nothing further to get*/ -} else if (H5FD_FAMILY==driver) { - hid_t member_fapl; - haddr_t member_size; - H5Pget_fapl_family(fapl, &member_size, &member_fapl); -} else if (....) { - .... -} -- - - -
-The H5Dread
and H5Dwrite
functions transfer data between
-application memory and the file. They both take an optional data transfer
-property list which has some general driver-independent properties and
-optional driver-defined properties. An application will typically perform I/O
-in one of three styles via the H5Dread
or H5Dwrite
function:
-
-
-Like file access properties in the previous section, data transfer properties -can be set using a driver initialization function or a general purpose -function. For example, to set the MPI-IO driver to use independent access for -I/O operations one would say: - -
- --hid_t dxpl = H5Pcreate(H5P_DATA_XFER); -H5Pset_dxpl_mpio(dxpl, H5FD_MPIO_INDEPENDENT); -H5Dread(dataset, type, mspace, fspace, buffer, dxpl); -H5Pclose(dxpl); -- -
-The alternative is to initialize a driver defined C struct
and pass it
-to the H5Pset_driver
function:
-
-
-hid_t dxpl = H5Pcreate(H5P_DATA_XFER); -static H5FD_mpio_dxpl_t dx = {H5FD_MPIO_INDEPENDENT}; -H5Pset_driver(dxpl, H5FD_MPIO, &dx); -H5Dread(dataset, type, mspace, fspace, buffer, dxpl); -- -
-The transfer property list can be queried in a manner similar to the file -access property list: the driver provides a function (or functions) to return -various information about the transfer property list: - -
- --hid_t driver = H5Pget_driver(dxpl); -if (H5FD_MPIO==driver) { - H5FD_mpio_xfer_t xfer_mode; - H5Pget_dxpl_mpio(dxpl, &xfer_mode); -} else { - .... -} -- - - -
-The HDF5 specifications describe two things: the mapping of data onto a linear -format address space and the C API which performs the mapping. -However, the mapping of the format address space onto storage intentionally -falls outside the scope of the HDF5 specs. This is a direct result of the fact -that it is not generally possible to store information about how to access -storage inside the storage itself. For instance, given only the file name -`/arborea/1225/work/f%03d' the HDF5 library is unable to tell whether the -name refers to a file on the local file system, a family of files on the local -file system, a file on host `arborea' port 1225, a family of files on a -remote system, etc. - -
--Two ways which library could figure out where the storage is located are: -storage access information can be provided by the user, or the library can try -all known file access methods. This implementation uses the former method. - -
--In general, if a file was created with one driver then it isn't possible to -open it with another driver. There are of course exceptions: a file created -with MPIO could probably be opened with the sec2 driver, any file created -by the sec2 driver could be opened as a family of files with one member, -etc. In fact, sometimes a file must not only be opened with the same -driver but also with the same driver properties. The predefined drivers are -written in such a way that specifying the correct driver is sufficient for -opening a file. - -
- - --A driver is simply a collection of functions and data structures which are -registered with the HDF5 library at runtime. The functions fall into these -categories: - -
- --Some drivers need information about file access and data transfers which are -very specific to the driver. The information is usually implemented as a pair -of pointers to C structs which are allocated and initialized as part of an -HDF5 property list and passed down to various driver functions. There are two -classes of settings: file access modes that describe how to access the file -through the driver, and data transfer modes which are settings that control -I/O operations. Each file opened by a particular driver may have a different -access mode; each dataset I/O request for a particular file may have a -different data transfer mode. - -
--Since each driver has its own particular requirements for various settings, -each driver is responsible for defining the mode structures that it -needs. Higher layers of the library treat the structures as opaque but must be -able to copy and free them. Thus, the driver provides either the size of the -structure or a pair of function pointers for each of the mode types. - -
--Example: The family driver needs to know how the format address -space is partitioned and the file access property list to use for the -family members. - -
- --/* Driver-specific file access properties */ -typedef struct H5FD_family_fapl_t { - hsize_t memb_size; /*size of each member */ - hid_t memb_fapl_id; /*file access property list of each memb*/ -} H5FD_family_fapl_t; - -/* Driver specific data transfer properties */ -typedef struct H5FD_family_dxpl_t { - hid_t memb_dxpl_id; /*data xfer property list of each memb */ -} H5FD_family_dxpl_t; -- -
-In order to copy or free one of these structures the member file access -or data transfer properties must also be copied or freed. This is done -by providing a copy and close function for each structure: - -
--Example: The file access property list copy and close functions -for the family driver: - -
- --static void * -H5FD_family_fapl_copy(const void *_old_fa) -{ - const H5FD_family_fapl_t *old_fa = (const H5FD_family_fapl_t*)_old_fa; - H5FD_family_fapl_t *new_fa = malloc(sizeof(H5FD_family_fapl_t)); - assert(new_fa); - - memcpy(new_fa, old_fa, sizeof(H5FD_family_fapl_t)); - new_fa->memb_fapl_id = H5Pcopy(old_fa->memb_fapl_id); - return new_fa; -} - -static herr_t -H5FD_family_fapl_free(void *_fa) -{ - H5FD_family_fapl_t *fa = (H5FD_family_fapl_t*)_fa; - H5Pclose(fa->memb_fapl_id); - free(fa); - return 0; -} -- -
-Generally when a file is created or opened the file access properties
-for the driver are copied into the file pointer which is returned and
-they may be modified from their original value (for instance, the file
-family driver modifies the member size property when opening an existing
-family). In order to support the H5Fget_access_plist
function the
-driver must provide a fapl_get
callback which creates a copy of
-the driver-specific properties based on a particular file.
-
-
-Example: The file family driver copies the member size file -access property list into the return value: - -
- --static void * -H5FD_family_fapl_get(H5FD_t *_file) -{ - H5FD_family_t *file = (H5FD_family_t*)_file; - H5FD_family_fapl_t *fa = calloc(1, sizeof(H5FD_family_fapl_t*)); - - fa->memb_size = file->memb_size; - fa->memb_fapl_id = H5Pcopy(file->memb_fapl_id); - return fa; -} -- - - -
-The higher layers of the library expect files to have a name and allow the
-file to be accessed in various modes. The driver must be able to create a new
-file, replace an existing file, or open an existing file. Opening or creating
-a file should return a handle, a pointer to a specialization of the
-H5FD_t
struct, which allows read-only or read-write access and which
-will be passed to the other driver functions as they are
-called.(5)
-
-
-typedef struct { - /* Public fields */ - H5FD_class_t *cls; /*class data defined below*/ - - /* Private fields -- driver-defined */ - -} H5FD_t; -- -
-Example: The family driver requires handles to the underlying
-storage, the size of the members for this particular file (which might be
-different than the member size specified in the file access property list if
-an existing file family is being opened), the name used to open the file in
-case additional members must be created, and the flags to use for creating
-those additional members. The eoa
member caches the size of the format
-address space so the family members don't have to be queried in order to find
-it.
-
-
-/* The description of a file belonging to this driver. */ -typedef struct H5FD_family_t { - H5FD_t pub; /*public stuff, must be first */ - hid_t memb_fapl_id; /*file access property list for members */ - hsize_t memb_size; /*maximum size of each member file */ - int nmembs; /*number of family members */ - int amembs; /*number of member slots allocated */ - H5FD_t **memb; /*dynamic array of member pointers */ - haddr_t eoa; /*end of allocated addresses */ - char *name; /*name generator printf format */ - unsigned flags; /*flags for opening additional members */ -} H5FD_family_t; -- -
-Example: The sec2 driver needs to keep track of the underlying Unix
-file descriptor and also the end of format address space and current Unix file
-size. It also keeps track of the current file position and last operation
-(read, write, or unknown) in order to optimize calls to lseek
. The
-device
and inode
fields are defined on Unix in order to uniquely
-identify the file and will be discussed below.
-
-
-typedef struct H5FD_sec2_t { - H5FD_t pub; /*public stuff, must be first */ - int fd; /*the unix file */ - haddr_t eoa; /*end of allocated region */ - haddr_t eof; /*end of file; current file size*/ - haddr_t pos; /*current file I/O position */ - int op; /*last operation */ - dev_t device; /*file device number */ - ino_t inode; /*file i-node number */ -} H5FD_sec2_t; -- - - -
-All drivers must define a function for opening/creating a file. This -function should have a prototype which is: - -
--
-The file name name and file access property list fapl are
-the same as were specified in the H5Fcreate
or H5Fopen
-call. The flags are the same as in those calls also except the
-flag H5F_ACC_CREATE
is also present if the call was to
-H5Fcreate
and they are documented in the `H5Fpublic.h'
-file. The maxaddr argument is the maximum format address that the
-driver should be prepared to handle (the minimum address is always
-zero).
-
-Example: The sec2 driver opens a Unix file with the requested name -and saves information which uniquely identifies the file (the Unix device -number and inode). - -
- --static H5FD_t * -H5FD_sec2_open(const char *name, unsigned flags, hid_t fapl_id/*unused*/, - haddr_t maxaddr) -{ - unsigned o_flags; - int fd; - struct stat sb; - H5FD_sec2_t *file=NULL; - - /* Check arguments */ - if (!name || !*name) return NULL; - if (0==maxaddr || HADDR_UNDEF==maxaddr) return NULL; - if (ADDR_OVERFLOW(maxaddr)) return NULL; - - /* Build the open flags */ - o_flags = (H5F_ACC_RDWR & flags) ? O_RDWR : O_RDONLY; - if (H5F_ACC_TRUNC & flags) o_flags |= O_TRUNC; - if (H5F_ACC_CREAT & flags) o_flags |= O_CREAT; - if (H5F_ACC_EXCL & flags) o_flags |= O_EXCL; - - /* Open the file */ - if ((fd=open(name, o_flags, 0666))<0) return NULL; - if (fstat(fd, &sb)<0) { - close(fd); - return NULL; - } - - /* Create the new file struct */ - file = calloc(1, sizeof(H5FD_sec2_t)); - file->fd = fd; - file->eof = sb.st_size; - file->pos = HADDR_UNDEF; - file->op = OP_UNKNOWN; - file->device = sb.st_dev; - file->inode = sb.st_ino; - - return (H5FD_t*)file; -} -- - - -
-Closing a file simply means that all cached data should be flushed to the next -lower layer, the file should be closed at the next lower layer, and all -file-related data structures should be freed. All information needed by the -close function is already present in the file handle. - -
--
-The file argument is the handle which was returned by the open
-function, and the close
should free only memory associated with the
-driver-specific part of the handle (the public parts will have already been released by HDF5's virtual file layer).
-
-Example: The sec2 driver just closes the underlying Unix file, -making sure that the actual file size is the same as that known to the -library by writing a zero to the last file position it hasn't been -written by some previous operation (which happens in the same code which -flushes the file contents and is shown below). - -
- --static herr_t -H5FD_sec2_close(H5FD_t *_file) -{ - H5FD_sec2_t *file = (H5FD_sec2_t*)_file; - - if (H5FD_sec2_flush(_file)<0) return -1; - if (close(file->fd)<0) return -1; - free(file); - return 0; -} -- - - -
-Occasionally an application will attempt to open a single file more than one -time in order to obtain multiple handles to the file. HDF5 allows the files to -share information(6) but in order to -accomplish this HDF5 must be able to tell when two names refer to the same -file. It does this by associating a driver-defined key with each file opened -by a driver and comparing the key for an open request with the keys for all -other files currently open by the same driver. - -
--
-The driver may provide a function which compares two files f1 and
-f2 belonging to the same driver and returns a negative, positive, or
-zero value a la the strcmp
function.(7) If this
-function is not provided then HDF5 assumes that all calls to the open
-callback return unique files regardless of the arguments and it is up to the
-application to avoid doing this if that assumption is incorrect.
-
-Each time a file is opened the library calls the cmp
function to
-compare that file with all other files currently open by the same driver and
-if one of them matches (at most one can match) then the file which was just
-opened is closed and the previously opened file is used instead.
-
-
-Opening a file twice with incompatible flags will result in failure. For -instance, opening a file with the truncate flag is a two step process which -first opens the file without truncation so keys can be compared, and if no -matching file is found already open then the file is closed and immediately -reopened with the truncation flag set (if a matching file is already open then -the truncating open will fail). - -
--Example: The sec2 driver uses the Unix device and i-node as the -key. They were initialized when the file was opened. - -
- --static int -H5FD_sec2_cmp(const H5FD_t *_f1, const H5FD_t *_f2) -{ - const H5FD_sec2_t *f1 = (const H5FD_sec2_t*)_f1; - const H5FD_sec2_t *f2 = (const H5FD_sec2_t*)_f2; - - if (f1->device < f2->device) return -1; - if (f1->device > f2->device) return 1; - - if (f1->inode < f2->inode) return -1; - if (f1->inode > f2->inode) return 1; - - return 0; -} -- - - -
-Some drivers may also need to store certain information in the file superblock -in order to be able to reliably open the file at a later date. This is done by -three functions: one to determine how much space will be necessary to store -the information in the superblock, one to encode the information, and one to -decode the information. These functions are optional, but if any one is -defined then the other two must also be defined. - -
--
-The sb_size
function returns the number of bytes necessary to encode
-information needed later if the file is reopened. The sb_encode
-function encodes information from the file into buffer buf
-allocated by the caller. It also writes an 8-character (plus null
-termination) into the name
argument, which should be a unique
-identification for the driver. The sb_decode
function looks at
-the name
-
-
- decodes -data from the buffer buf and updates the file argument with the new information, -advancing *p in the process. -
-The part of this which is somewhat tricky is that the file must be readable -before the superblock information is decoded. File access modes fall outside -the scope of the HDF5 file format, but they are placed inside the boot block -for convenience.(8) - -
--Example: To be written later. - -
- - --HDF5 does not assume that a file is a linear address space of bytes. Instead, -the library will call functions to allocate and free portions of the HDF5 -format address space, which in turn map onto functions in the file driver to -allocate and free portions of file address space. The library tells the file -driver how much format address space it wants to allocate and the driver -decides what format address to use and how that format address is mapped onto -the file address space. Usually the format address is chosen so that the file -address can be calculated in constant time for data I/O operations (which are -always specified by format addresses). - -
- - - --The HDF5 format allows an optional userblock to appear before the actual HDF5 -data in such a way that if the userblock is sucked out of the file and -everything remaining is shifted downward in the file address space, then the -file is still a valid HDF5 file. The userblock size can be zero or any -multiple of two greater than or equal to 512 and the file superblock begins -immediately after the userblock. - -
--HDF5 allocates space for the userblock and superblock by calling an -allocation function defined below, which must return a chunk of memory at -format address zero on the first call. - -
- - --The library makes many types of allocation requests: - -
-H5FD_MEM_SUPER
-H5FD_MEM_BTREE
-H5FD_MEM_DRAW
-H5FD_MEM_META
-H5FD_MEM_GROUP
-H5FD_MEM_GHEAP
-H5FD_MEM_LHEAP
-H5FD_MEM_OHDR
-
-When a chunk of memory is freed the library adds it to a free list and
-allocation requests are satisfied from the free list before requesting memory
-from the file driver. Each type of allocation request enumerated above has its
-own free list, but the file driver can specify that certain object types can
-share a free list. It does so by providing an array which maps a request type
-to a free list. If any value of the map is H5MF_DEFAULT
(zero) then the
-object's own free list is used. The special value H5MF_NOLIST
indicates
-that the library should not attempt to maintain a free list for that
-particular object type, instead calling the file driver each time an object of
-that type is freed.
-
-
-Mappings predefined in the `H5FDpublic.h' file are: -
H5FD_FLMAP_SINGLE
-H5FD_FLMAP_DICHOTOMY
-H5FD_FLMAP_DEFAULT
-
-Example: To make a map that manages object headers on one free list
-and everything else on another free list one might initialize the map with the
-following code: (the use of H5FD_MEM_SUPER
is arbitrary)
-
-
-H5FD_mem_t mt, map[H5FD_MEM_NTYPES]; - -for (mt=0; mt<H5FD_MEM_NTYPES; mt++) { - map[mt] = (H5FD_MEM_OHDR==mt) ? mt : H5FD_MEM_SUPER; -} -- -
-If an allocation request cannot be satisfied from the free list then one of -two things happen. If the driver defines an allocation callback then it is -used to allocate space; otherwise new memory is allocated from the end of the -format address space by incrementing the end-of-address marker. - -
--
-The file argument is the file from which space is to be allocated,
-type is the type of memory being requested (from the list above) without
-being mapped according to the freelist map and size is the number of
-bytes being requested. The library is allowed to allocate large chunks of
-storage and manage them in a layer above the file driver (although the current
-library doesn't do that). The allocation function should return a format
-address for the first byte allocated. The allocated region extends from that
-address for size bytes. If the request cannot be honored then the
-undefined address value is returned (HADDR_UNDEF
). The first call to
-this function for a file which has never had memory allocated must
-return a format address of zero or HADDR_UNDEF
since this is how the
-library allocates space for the userblock and/or superblock.
-
-Example: To be written later. - -
- - -
-When the library is finished using a certain region of the format address
-space it will return the space to the free list according to the type of
-memory being freed and the free list map described above. If the free list has
-been disabled for a particular memory usage type (according to the free list
-map) and the driver defines a free
callback then it will be
-invoked. The free
callback is also invoked for all entries on the free
-list when the file is closed.
-
-
-
-The file argument is the file for which space is being freed; type -is the type of object being freed (from the list above) without being mapped -according to the freelist map; addr is the first format address to free; -and size is the size in bytes of the region being freed. The region -being freed may refer to just part of the region originally allocated and/or -may cross allocation boundaries provided all regions being freed have the same -usage type. However, the library will never attempt to free regions which have -already been freed or which have never been allocated. -
-A driver may choose to not define the free
function, in which case
-format addresses will be leaked. This isn't normally a huge problem since the
-library contains a simple free list of its own and freeing parts of the format
-address space is not a common occurrence.
-
-
-Example: To be written later. - -
- - --Each file driver must have some mechanism for setting and querying the end of -address, or EOA, marker. The EOA marker is the first format address -after the last format address ever allocated. If the last part of the -allocated address range is freed then the driver may optionally decrease the -eoa marker. - -
--
-This function returns the current value of the EOA marker for the specified -file. -
-Example: The sec2 driver just returns the current eoa marker value -which is cached in the file structure: - -
- --static haddr_t -H5FD_sec2_get_eoa(H5FD_t *_file) -{ - H5FD_sec2_t *file = (H5FD_sec2_t*)_file; - return file->eoa; -} -- -
-The eoa marker is initially zero when a file is opened and the library may set
-it to some other value shortly after the file is opened (after the superblock
-is read and the saved eoa marker is determined) or when allocating additional
-memory in the absence of an alloc
callback (described above).
-
-
-Example: The sec2 driver simply caches the eoa marker in the file -structure and does not extend the underlying Unix file. When the file is -flushed or closed then the Unix file size is extended to match the eoa marker. - -
- --static herr_t -H5FD_sec2_set_eoa(H5FD_t *_file, haddr_t addr) -{ - H5FD_sec2_t *file = (H5FD_sec2_t*)_file; - file->eoa = addr; - return 0; -} -- - - -
-These functions operate on data, transferring a region of the format address -space between memory and files. - -
- - - --A driver must specify two functions to transfer data from the library to the -file and vice versa. - -
--
-The read
function reads data from file file beginning at address
-addr and continuing for size bytes into the buffer buf
-supplied by the caller. The write
function transfers data in the
-opposite direction. Both functions take a data transfer property list
-dxpl which indicates the fine points of how the data is to be
-transferred and which comes directly from the H5Dread
or
-H5Dwrite
function. Both functions receive type of
-data being written, which may allow a driver to tune it's behavior for
-different kinds of data.
-
-Both functions should return a negative value if they fail to transfer the -requested data, or non-negative if they succeed. The library will never -attempt to read from unallocated regions of the format address space. - -
-
-Example: The sec2 driver just makes system calls. It tries not to
-call lseek
if the current operation is the same as the previous
-operation and the file position is correct. It also fills the output buffer
-with zeros when reading between the current EOF and EOA markers and restarts
-system calls which were interrupted.
-
-
-static herr_t -H5FD_sec2_read(H5FD_t *_file, H5FD_mem_t type/*unused*/, hid_t dxpl_id/*unused*/, - haddr_t addr, hsize_t size, void *buf/*out*/) -{ - H5FD_sec2_t *file = (H5FD_sec2_t*)_file; - ssize_t nbytes; - - assert(file && file->pub.cls); - assert(buf); - - /* Check for overflow conditions */ - if (REGION_OVERFLOW(addr, size)) return -1; - if (addr+size>file->eoa) return -1; - - /* Seek to the correct location */ - if ((addr!=file->pos || OP_READ!=file->op) && - file_seek(file->fd, (file_offset_t)addr, SEEK_SET)<0) { - file->pos = HADDR_UNDEF; - file->op = OP_UNKNOWN; - return -1; - } - - /* - * Read data, being careful of interrupted system calls, partial results, - * and the end of the file. - */ - while (size>0) { - do nbytes = read(file->fd, buf, size); - while (-1==nbytes && EINTR==errno); - if (-1==nbytes) { - /* error */ - file->pos = HADDR_UNDEF; - file->op = OP_UNKNOWN; - return -1; - } - if (0==nbytes) { - /* end of file but not end of format address space */ - memset(buf, 0, size); - size = 0; - } - assert(nbytes>=0); - assert((hsize_t)nbytes<=size); - size -= (hsize_t)nbytes; - addr += (haddr_t)nbytes; - buf = (char*)buf + nbytes; - } - - /* Update current position */ - file->pos = addr; - file->op = OP_READ; - return 0; -} -- -
-Example: The sec2 write
callback is similar except it updates
-the file EOF marker when extending the file.
-
-
-Some drivers may desire to cache data in memory in order to make larger I/O
-requests to the underlying file and thus improving bandwidth. Such drivers
-should register a cache flushing function so that the library can insure that
-data has been flushed out of the drivers in response to the application
-calling H5Fflush
.
-
-
-
-Flush all data for file file to storage. -
-Example: The sec2 driver doesn't cache any data but it also doesn't -extend the Unix file as aggressively as it should. Therefore, when finalizing a -file it should write a zero to the last byte of the allocated region so that -when reopening the file later the EOF marker will be at least as large as the -EOA marker saved in the superblock (otherwise HDF5 will refuse to open the -file, claiming that the data appears to be truncated). - -
- --static herr_t -H5FD_sec2_flush(H5FD_t *_file) -{ - H5FD_sec2_t *file = (H5FD_sec2_t*)_file; - - if (file->eoa>file->eof) { - if (-1==file_seek(file->fd, file->eoa-1, SEEK_SET)) return -1; - if (write(file->fd, "", 1)!=1) return -1; - file->eof = file->eoa; - file->pos = file->eoa; - file->op = OP_WRITE; - } - - return 0; -} -- - - -
-The library is capable of performing several generic optimizations on I/O, but -these types of optimizations may not be appropriate for a given VFL driver. -
- --Each driver may provide a query function to allow the library to query whether -to enable these optimizations. If a driver lacks a query function, the library -will disable all types of optimizations which can be queried. -
- --
-This function is called by the library to query which optimizations to enable -for I/O to this driver. These are the flags which are currently defined: - -
-Before a driver can be used the HDF5 library needs to be told of its -existence. This is done by registering the driver, which results in a driver -identification number. Instead of passing many arguments to the registration -function, the driver information is entered into a structure and the address -of the structure is passed to the registration function where it is -copied. This allows the HDF5 API to be extended while providing backward -compatibility at the source level. - -
--
-The driver described by struct cls is registered with the library and an -ID number for the driver is returned. -
-The H5FD_class_t
type is a struct with the following fields:
-
-
const char *name
-size_t fapl_size
-void *(*fapl_copy)(const void *fapl)
-fm_size
when both are defined.
-void (*fapl_free)(void *fapl)
-free
function to free the
-structure.
-size_t dxpl_size
-void *(*dxpl_copy)(const void *dxpl)
-xm_size
when both are
-defined.
-void (*dxpl_free)(void *dxpl)
-free
function to
-free the structure.
-H5FD_t *(*open)(const char *name, unsigned flags, hid_t fapl, haddr_t maxaddr)
-herr_t (*close)(H5FD_t *file)
-int (*cmp)(const H5FD_t *f1, const H5FD_t *f2)
-int (*query)(const H5FD_t *f, unsigned long *flags)
-haddr_t (*alloc)(H5FD_t *file, H5FD_mem_t type, hsize_t size)
-herr_t (*free)(H5FD_t *file, H5FD_mem_t type, haddr_t addr, hsize_t size)
-haddr_t (*get_eoa)(H5FD_t *file)
-herr_t (*set_eoa)(H5FD_t *file, haddr_t)
-haddr_t (*get_eof)(H5FD_t *file)
-herr_t (*read)(H5FD_t *file, H5FD_mem_t type, hid_t dxpl, haddr_t addr, hsize_t size, void *buffer)
-herr_t (*write)(H5FD_t *file, H5FD_mem_t type, hid_t dxpl, haddr_t addr, hsize_t size, const void *buffer)
-herr_t (*flush)(H5FD_t *file)
-H5FD_mem_t fl_map[H5FD_MEM_NTYPES]
--Example: The sec2 driver would be registered as: - -
- --static const H5FD_class_t H5FD_sec2_g = { - "sec2", /*name */ - MAXADDR, /*maxaddr */ - NULL, /*sb_size */ - NULL, /*sb_encode */ - NULL, /*sb_decode */ - 0, /*fapl_size */ - NULL, /*fapl_get */ - NULL, /*fapl_copy */ - NULL, /*fapl_free */ - 0, /*dxpl_size */ - NULL, /*dxpl_copy */ - NULL, /*dxpl_free */ - H5FD_sec2_open, /*open */ - H5FD_sec2_close, /*close */ - H5FD_sec2_cmp, /*cmp */ - H5FD_sec2_query, /*query */ - NULL, /*alloc */ - NULL, /*free */ - H5FD_sec2_get_eoa, /*get_eoa */ - H5FD_sec2_set_eoa, /*set_eoa */ - H5FD_sec2_get_eof, /*get_eof */ - H5FD_sec2_read, /*read */ - H5FD_sec2_write, /*write */ - H5FD_sec2_flush, /*flush */ - H5FD_FLMAP_SINGLE, /*fl_map */ -}; - -hid_t -H5FD_sec2_init(void) -{ - if (!H5FD_SEC2_g) { - H5FD_SEC2_g = H5FDregister(&H5FD_sec2_g); - } - return H5FD_SEC2_g; -} -- -
-A driver can be removed from the library by unregistering it - -
--
-Unregistering a driver makes it unusable for creating new file access or data -transfer property lists but doesn't affect any property lists or files that -already use that driver. - -
- - - - -If a C routine that takes a function pointer as an argument is -called from within C++ code, the C routine should be returned from -normally.
- -Examples of this kind of routine include callbacks such as
-H5Pset_elink_cb
and H5Pset_type_conv_cb
-and functions such as H5Tconvert
and
-H5Ewalk2
.
Exiting the routine in its normal fashion allows the HDF5 C -Library to clean up its work properly. In other words, if the C++ -application jumps out of the routine back to the C++ -“catch” statement, the library is not given the -opportunity to close any temporary data structures that were set -up when the routine was called. The C++ application should save -some state as the routine is started so that any problem that -occurs might be diagnosed.
- - - - - - - --
-This function is intended to be used by driver functions, not applications.
-It returns a pointer directly into the file access property list
-fapl
which is a copy of the driver's file access mode originally
-provided to the H5Pset_driver
function. If its argument is a data
-transfer property list fxpl
then it returns a pointer to the
-driver-specific data transfer information instead.
-
-The various private H5F_low_*
functions will be replaced by public
-H5FD*
functions so they can be called from drivers.
-
-
-All private functions H5F_addr_*
which operate on addresses will be
-renamed as public functions by removing the first underscore so they can be
-called by drivers.
-
-
-The haddr_t
address data type will be passed by value throughout the
-library. The original intent was that this type would eventually be a union of
-file address types for the various drivers and may become quite large, but
-that was back when drivers were part of HDF5. It will become an alias for an
-unsigned integer type (32 or 64 bits depending on how the library was
-configured).
-
-
-The various H5F*.c
driver files will be renamed H5FD*.c
and each
-will have a corresponding header file. All driver functions except the
-initializer and API will be declared static.
-
-
-This documentation didn't cover optimization functions which would be useful -to drivers like MPI-IO. Some drivers may be able to perform data pipeline -operations more efficiently than HDF5 and need to be given a chance to -override those parts of the pipeline. The pipeline would be designed to call -various H5FD optimization functions at various points which return one of -three values: the operation is not implemented by the driver, the operation is -implemented but failed in a non-recoverable manner, the operation is -implemented and succeeded. - -
--Various parts of HDF5 check the only the top-level file driver and do -something special if it is the MPI-IO driver. However, we might want to be -able to put the MPI-IO driver under other drivers such as the raw part of a -split driver or under a debug driver whose sole purpose is to accumulate -statistics as it passes all requests through to the MPI-IO driver. Therefore -we will probably need a function which takes a format address and or object -type and returns the driver which would have been used at the lowest level to -process the request. - -
- --
The driver name is by convention and might -not apply to drivers which are not distributed with HDF5. -
The access method also indicates how to translate -the storage name to a storage server such as a file, network protocol, or -memory. -
The term -"file access property list" is a misnomer since storage isn't -required to be a file. -
This -function is overloaded to operate on data transfer property lists also, as -described below. -
Read-only access is only appropriate when opening an existing -file. -
For instance, writing data to one handle will cause -the data to be immediately visible on the other handle. -
The ordering is -arbitrary as long as it's consistent within a particular file driver. -
File access modes do not describe data, but rather -describe how the HDF5 format address space is mapped to the underlying -file(s). Thus, in general the mapping must be known before the file superblock -can be read. However, the user usually knows enough about the mapping for the -superblock to be readable and once the superblock is read the library can fill -in the missing parts of the mapping. -
- - - - - From 778646f5061bd719b872a0a77e0b7c005a27cec4 Mon Sep 17 00:00:00 2001 From: Allen Byrne <50328838+byrnHDF@users.noreply.github.com> Date: Fri, 22 Nov 2024 15:44:40 -0600 Subject: [PATCH 03/12] Update zfp compression lib references (#5141) --- CMakePresets.json | 2 +- HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.ddl | 2 +- HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.tst | 2 +- config/cmake/cacheinit.cmake | 2 +- release_docs/INSTALL_CMake.txt | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/CMakePresets.json b/CMakePresets.json index 24f89c628dd..18ae431dbbe 100644 --- a/CMakePresets.json +++ b/CMakePresets.json @@ -68,7 +68,7 @@ "LZF_PACKAGE_NAME": {"type": "STRING", "value": "lzf"}, "SZ_TGZ_NAME": {"type": "STRING", "value": "SZ-2.1.12.5.tar.gz"}, "SZ_PACKAGE_NAME": {"type": "STRING", "value": "SZ"}, - "ZFP_TGZ_NAME": {"type": "STRING", "value": "zfp-1.0.0.tar.gz"}, + "ZFP_TGZ_NAME": {"type": "STRING", "value": "zfp-1.0.1.tar.gz"}, "ZFP_PACKAGE_NAME": {"type": "STRING", "value": "zfp"}, "ZSTD_TGZ_NAME": {"type": "STRING", "value": "zstd-1.5.6.tar.gz"}, "ZSTD_PACKAGE_NAME": {"type": "STRING", "value": "zstd"} diff --git a/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.ddl b/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.ddl index a8e66c08054..8dadf939143 100644 --- a/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.ddl +++ b/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.ddl @@ -10,7 +10,7 @@ GROUP "/" { FILTERS { USER_DEFINED_FILTER { FILTER_ID 32013 - COMMENT H5Z-ZFP-1.1.1 (ZFP-1.0.0) github.com/LLNL/H5Z-ZFP + COMMENT H5Z-ZFP-1.1.1 (ZFP-1.0.1) github.com/LLNL/H5Z-ZFP PARAMS { XXXX } } } diff --git a/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.tst b/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.tst index dd7197cda78..2bec93cce2f 100644 --- a/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.tst +++ b/HDF5Examples/C/H5FLT/tfiles/h5ex_d_zfp.tst @@ -5,7 +5,7 @@ zfp filter is available for encoding and decoding. Filter info is available from the dataset creation property Filter identifier is 32013 Number of parameters is 6 with the value 268456209 - To find more about the filter check H5Z-ZFP-1.1.1 (ZFP-1.0.0) github.com/LLNL/H5Z-ZFP + To find more about the filter check H5Z-ZFP-1.1.1 (ZFP-1.0.1) github.com/LLNL/H5Z-ZFP ....Reading zfp compressed data ................ Maximum value in DS1 is 1890.0000 zfp filter is available now since H5Dread triggered loading of the filter. diff --git a/config/cmake/cacheinit.cmake b/config/cmake/cacheinit.cmake index 93cb9332e94..851e7c5987e 100644 --- a/config/cmake/cacheinit.cmake +++ b/config/cmake/cacheinit.cmake @@ -229,7 +229,7 @@ set (ZFP_GIT_URL "https://github.com/LLNL/zfp.git" CACHE STRING "Use ZFP from G set (ZFP_GIT_BRANCH "develop" CACHE STRING "" FORCE) set (ZFP_TGZ_ORIGPATH "https://github.com/LLNL/zfp/releases/download/1.0.0" CACHE STRING "Use PLUGINS from original location" FORCE) -set (ZFP_TGZ_NAME "zfp-1.0.0.tar.gz" CACHE STRING "Use ZFP from compressed file" FORCE) +set (ZFP_TGZ_NAME "zfp-1.0.1.tar.gz" CACHE STRING "Use ZFP from compressed file" FORCE) set (ZFP_PACKAGE_NAME "zfp" CACHE STRING "Name of ZFP package" FORCE) diff --git a/release_docs/INSTALL_CMake.txt b/release_docs/INSTALL_CMake.txt index 149c7ba6295..28b7195695b 100644 --- a/release_docs/INSTALL_CMake.txt +++ b/release_docs/INSTALL_CMake.txt @@ -657,7 +657,7 @@ These five steps are described in detail below. set (ZFP_GIT_URL "https://github.com/LLNL/zfp.git" CACHE STRING "Use ZFP from GitHub repository" FORCE) set (ZFP_GIT_BRANCH "develop" CACHE STRING "" FORCE) set (ZFP_TGZ_ORIGPATH "https://github.com/LLNL/zfp/releases/download/1.0.0" CACHE STRING "Use PLUGINS from original location" FORCE) - set (ZFP_TGZ_NAME "zfp-1.0.0.tar.gz" CACHE STRING "Use ZFP from compressed file" FORCE) + set (ZFP_TGZ_NAME "zfp-1.0.1.tar.gz" CACHE STRING "Use ZFP from compressed file" FORCE) set (ZFP_PACKAGE_NAME "zfp" CACHE STRING "Name of ZFP package" FORCE) ###### # zstd From 2ce2fb0054d4d774b1e762c747b83c12b34428be Mon Sep 17 00:00:00 2001 From: Allen Byrne <50328838+byrnHDF@users.noreply.github.com> Date: Sat, 23 Nov 2024 14:00:16 -0600 Subject: [PATCH 04/12] Convert file image ops file to doxygen (#5134) --- doxygen/aliases | 2 - doxygen/dox/RFC.dox | 1 - doxygen/dox/TechnicalNotes.dox | 8 +- doxygen/dox/UsersGuide.dox | 46 +- doxygen/dox/ViewTools.dox | 4 + doxygen/examples/FileImageOps.html | 1611 ---------------------------- doxygen/img/DOChunks_fig1.png | Bin 0 -> 22612 bytes doxygen/img/DOChunks_fig2.png | Bin 0 -> 208502 bytes hl/src/H5DOpublic.h | 215 +++- hl/src/H5LTpublic.h | 3 +- src/H5Dmodule.h | 2 +- src/H5Dpublic.h | 2 +- src/H5Fmodule.h | 1448 ++++++++++++++++++++++++- src/H5Fpublic.h | 2 +- src/H5PLmodule.h | 2 +- src/H5Pmodule.h | 2 +- src/H5Ppublic.h | 19 +- src/H5module.h | 2 +- 18 files changed, 1708 insertions(+), 1661 deletions(-) delete mode 100644 doxygen/examples/FileImageOps.html create mode 100644 doxygen/img/DOChunks_fig1.png create mode 100644 doxygen/img/DOChunks_fig2.png diff --git a/doxygen/aliases b/doxygen/aliases index 0d355c001f0..fffec1cc2aa 100644 --- a/doxygen/aliases +++ b/doxygen/aliases @@ -260,7 +260,6 @@ ALIASES += sa_metadata_ops="\sa \li H5Pget_all_coll_metadata_ops() \li H5Pget_co ################################################################################ ALIASES += ref_cons_semantics="Enabling a Strict Consistency Semantics Model in Parallel HDF5" -ALIASES += ref_file_image_ops="HDF5 File Image Operations" ALIASES += ref_filter_pipe="Data Flow Pipeline for H5Dread()" ALIASES += ref_group_impls="Group implementations in HDF5" ALIASES += ref_h5lib_relver="HDF5 Library Release Version Numbers" @@ -327,7 +326,6 @@ ALIASES += ref_rfc20121114="HDF5 File Space Management" ALIASES += ref_rfc20120828="New HDF5 API Routines for HPC Applications - Read/Write Multiple Datasets in an HDF5 file" ALIASES += ref_rfc20120523="HDF5 File Space Management: Paged Aggregation" -ALIASES += ref_rfc20120501="HDF5 File Image Operations" ALIASES += ref_rfc20120305="Enabling a Strict Consistency Semantics Model in Parallel HDF5" ALIASES += ref_rfc20120220="h5repack: Improved Hyperslab selections for Large Chunked Datasets" ALIASES += ref_rfc20120120="A Maintainer's Guide for the Datatype Module in HDF5 Library" diff --git a/doxygen/dox/RFC.dox b/doxygen/dox/RFC.dox index 1b1141b4fd5..0482aee0c98 100644 --- a/doxygen/dox/RFC.dox +++ b/doxygen/dox/RFC.dox @@ -53,7 +53,6 @@
Using the Direct Chunk Write Function
+@ref H5DO_UG
Describes another way that chunks can be written to datasets.
@@ -377,7 +419,7 @@ These documents provide additional information for the use and tuning of specifi\ref H5FIM_UG
Describes how to work with HDF5 files in memory. Disk I/O is not required when file images are opened, created, read from, or written to.
diff --git a/doxygen/dox/ViewTools.dox b/doxygen/dox/ViewTools.dox index 43686751bfb..6bb000eb71b 100644 --- a/doxygen/dox/ViewTools.dox +++ b/doxygen/dox/ViewTools.dox @@ -303,6 +303,8 @@ GROUP "/" { } \endcode +\see H5TOOL_DP_UG for more information. + \subsection subsecViewToolsViewContent_h5ls h5ls The h5ls tool by default just displays the objects in the root group. It will not display items in groups beneath the root group unless specified. Useful h5ls options for viewing @@ -355,6 +357,8 @@ The output is shown below: /HDFEOS\ INFORMATION/StructMetadata.0 Dataset {SCALAR} \endcode +\see H5TOOL_LS_UG for more information. + \section secViewToolsViewDset Datasets and Dataset Properties Both h5dump and h5ls can be used to view specific datasets.File image operations allow users to work with HDF5 files in memory in the same ways that users currently work with HDF5 files on disk. Disk I/O is not required when file images are opened, created, read from, or written to.
-An HDF5 file image is an HDF5 file that is held in a buffer in main memory. Setting up a file image in memory involves using either a buffer in the file access property list or a buffer in the Core (aka Memory) file driver.
-The advantage of working with a file in memory is faster access to the data.
-The challenge of working with files in memory buffers is maximizing performance and minimizing memory footprint while working within the constraints of the property list mechanism. This should be a non-issue for small file images, but may be a major issue for large images.
-If invoked with the appropriate flags, the H5LTopen_file_image() high level library call should deal with these challenges in most cases. However, some applications may require the programmer to address these issues directly.
- -Functions used in file image operations are listed below.
-C Function | -Purpose | -Section | -
---|---|---|
H5Pset_file_image | -Specifies an initial file image | -2.1.1 | -H5Pget_file_image | -Retrieves a copy of the file image designated for a VFD to use as the initial contents of a file | -2.1.2 | -
H5Pset_file_image_callbacks | -Manages file image buffer allocation, copying, reallocation, and release | -2.1.3 | - - -H5Pget_file_image_callbacks | -Obtains the current file image callbacks from a file access property list | -2.1.4 | - - -H5Fget_file_image | -Provides a simple way to retrieve a copy of the image of an existing, open file | -2.1.5 | - - -H5LTopen_file_image | -Provides a convenient way to open an initial file image with the Core VFD | -2.1.6 | - -
The following abbreviations are used in this document:
-Abbreviation | -Explanation | -
---|---|
FAPL or fapl | -File Access Property List. In code samples, fapl is used. | -
VFD | -Virtual File Driver | -
VFL | -Virtual File Layer | -
Developers who use the file image operations described in this document should be proficient and experienced users of the HDF5 C Library APIs. More specifically, developers should have a working knowledge of property lists, callbacks, and virtual file drivers.
- -See the following for more information.
-The “RFC: File Image Operations” is the primary source for the information in this document.
-The “Alternate File Storage Layouts and Low-level File Drivers” section is in “The HDF5 File” chapter of the HDF5 User’s Guide .
-The H5P_SET_FAPL_CORE function call can be used to modify the file access property list so that the Memory virtual file driver, H5FD_CORE, is used. The Memory file driver is also known as the Core file driver.
-Refer to the Virtual File Layer for more detail. A list of VFL Functions is provided below.
-- -
C Function | -Purpose | -
---|---|
H5Pset_driver | -Sets a file driver | -
H5Pget_driver_info | -Returns a pointer to file driver information | -
H5FDregister | -Registers a new file driver as a member of the virtual file driver class | -
H5FDunregister | -Removes a driver ID from the library | -
H5FDopen | -Opens a file | -
H5FDclose | -Closes the file using the driver 'close' callback | -
H5FDcmp | -Compares the keys of two files using the file driver callback if the files belong to the same driver, otherwise sort the files by driver class pointer value | -
H5FDquery | -Queries a VFL driver for its feature flags | -
H5FDalloc | -Allocates memory from the file | -
H5FDfree | -Frees format addresses in the file | -
H5FDset_eoa | -Set the end-of-address marker for the file | -
H5FDget_eoa | -Returns the address of the first byte after the last allocated memory in the file | -
H5FDget_eof | -Returns the end-of-file address, which is the greater of the end-of-format address and the actual EOF marker | -
H5FDread | -Reads bytes from the file beginning at the specified address according to the provided data transfer property list | -
H5FDwrite | -Writes bytes to the file beginning at the specified address according to the provided data transfer property list | -
H5FDflush | -Notifies driver to flush all cached data | -
The C API function calls described in this chapter fall into two categories: low-level routines that are part of the main HDF5 C Library and one high-level routine that is part of the “lite” API in the high-level wrapper library. The high-level routine uses the low-level routines and presents frequently requested functionality conveniently packaged for application developers’ use.
- -The purpose of this section is to describe the low-level C API routines that support file image operations. These routines allow an in-memory image of an HDF5 file to be opened without requiring file system I/O.
-The basic approach to opening an in-memory image of an HDF5 file is to pass the image to the Core file driver, and then tell the Core file driver to open the file. We do this by using the H5Pget/set_file_image calls. These calls allow the user to specify an initial file image.
-A potential problem with the H5Pget/set_file_image calls is the overhead of allocating and copying of large file image buffers. The callback routines enable application programs to avoid this problem. However, the use of these callbacks is complex and potentially hazardous: the particulars are discussed in the semantics and examples chapters below (see section 3.1 and section 4.1 respectively). Fortunately, use of the file image callbacks should seldom be necessary: the H5LTopen_file_image call should address most use cases.
-The property list facility in HDF5 is employed in file image operations. This facility was designed for passing data, not consumable resources, into API calls. The peculiar ways in which the file image allocation callbacks may be used allows us to avoid extending the property list structure to handle consumable resources cleanly and to avoid constructing a new facility for the purpose.
-The sub-sections below describe the low-level C APIs that are used with file image operations.
- -The H5Pset_file_image routine allows an application to provide an image for a file driver to use as the initial contents of the file. This call was designed initially for use with the Core VFD, but it can be used with any VFD that supports using an initial file image when opening a file. See the “Virtual File Driver Feature Flags” section for more information. Calling this routine makes a copy of the provided file image buffer. See the “H5Pset_file_image_callbacks” section for more information.
-The signature of H5Pset_file_image is defined as follows:
-herr_t H5Pset_file_image(hid_t fapl_id, void *buf_ptr, size_t buf_len)
-
-
The parameters of H5Pset_file_image are defined as follows:
-fapl_id contains the ID of the target file access property list. -buf_ptr supplies a pointer to the initial file image, or NULL if no initial file image is desired. -buf_len contains the size of the supplied buffer, or 0 if no initial image is desired. -If either the buf_len parameter is zero, or the buf_ptr parameter is NULL, no file image will be set in the FAPL, and any existing file image buffer in the FAPL will be released. If a buffer is released, the FAPL’s file image buf_len will be set to 0 and buf_ptr will be set to NULL.
-Given the tight interaction between the file image callbacks and the file image, the file image callbacks in a property list cannot be changed while a file image is defined.
-With properly constructed file image callbacks, it is possible to avoid actually copying the file image. The particulars of this are discussed in greater detail in the “C API Call Semantics” chapter and in the “Examples” chapter.
- -The H5Pget_file_image routine allows an application to retrieve a copy of the file image designated for a VFD to use as the initial contents of a file. This routine uses the file image callbacks (if defined) when allocating and loading the buffer to return to the application, or it uses malloc and memcpy if the callbacks are undefined. When malloc and memcpy are used, it will be the caller’s responsibility to discard the returned buffer via a call to free.
-The signature of H5Pget_file_image is defined as follows:
-herr_t H5Pget_file_image(hid_t fapl_id, void **buf_ptr_ptr, size_t *buf_len_ptr)
-
-
The parameters of H5Pget_file_image are defined as follows:
-fapl_id contains the ID of the target file access property list. -
buf_ptr_ptr
contains a NULL or a pointer to a void*. If buf_ptr_ptr is not NULL, on successful return, *buf_ptr_ptr will contain a pointer to a copy of the initial image provided in the last call to H5Pset_file_image for the supplied fapl_id. If no initial image has been set, *buf_ptr_ptr will be NULL.
buf_len_ptr
contains a NULL or a pointer to size_t. If buf_len_ptr is not NULL, on successful return, *buf_len_ptr will contain the value of the buf_len parameter for the initial image in the supplied fapl_id. If no initial image is set, the value of *buf_len_ptr will be 0.
As with H5Pset_file_image, appropriately defined file image callbacks can allow this function to avoid buffer allocation and memory copy operations.
- -The H5Pset_file_image_callbacks API call exists to allow an application to control the management of file image buffers through user defined callbacks. These callbacks will be used in the management of file image buffers in property lists and in select file drivers. These routines are invoked when a new file image buffer is allocated, when an existing file image buffer is copied or resized, or when a file image buffer is released from use. From the perspective of the HDF5 Library, the operations of the image_malloc, image_memcpy, image_realloc, and image_free callbacks must be identical to those of the corresponding C standard library calls (malloc, memcpy, realloc, and free). While the operations must be identical, the file image callbacks have more parameters. The callbacks and their parameters are described below. The return values of image_malloc and image_realloc are identical to the return values of malloc and realloc. However, the return values of image_memcpy and image_free are different than the return values of memcpy and free: the return values of image_memcpy and image_free can also indicate failure. See the “File Image Callback Semantics” section for more information.
-The signature of H5Pset_file_image_callbacks is defined as follows:
- -typedef enum
-{
- H5_FILE_IMAGE_OP_PROPERTY_LIST_SET,
- H5_FILE_IMAGE_OP_PROPERTY_LIST_COPY,
- H5_FILE_IMAGE_OP_PROPERTY_LIST_GET,
- H5_FILE_IMAGE_OP_PROPERTY_LIST_CLOSE,
- H5_FILE_IMAGE_OP_FILE_OPEN,
- H5_FILE_IMAGE_OP_FILE_RESIZE,
- H5_FILE_IMAGE_OP_FILE_CLOSE
-} H5_file_image_op_t;
-
-typedef struct
-{
- void *(*image_malloc)(size_t size, H5_file_image_op_t file_image_op,
- void *udata);
- void *(*image_memcpy)(void *dest, const void *src, size_t size,
- H5_file_image_op_t file_image_op, void *udata);
- void *(*image_realloc)(void *ptr, size_t size,
- H5_file_image_op_t file_image_op, void *udata);
- herr_t (*image_free)(void *ptr, H5_file_image_op_t file_image_op,
- void *udata);
- void *(*udata_copy)(void *udata);
- herr_t (*udata_free)(void *udata);
- void *udata;
-} H5_file_image_callbacks_t;
-
-herr_t H5Pset_file_image_callbacks(hid_t fapl_id,
- H5_file_image_callbacks_t *callbacks_ptr)
fapl_id
contains the ID of the target file access property list.callbacks_ptr
contains a pointer to an instance of the H5_file_image_callbacks_t structure.The fields of the H5_file_image_callbacks_t structure are defined as follows: - -
image_malloc
contains a pointer to a function with (from the perspective of HDF5) functionality identical to the standard C library malloc() call. The parameters of the image_malloc callback are defined as follows:size
contains the size in bytes of the image buffer to allocate.file_image_op
contains one of the values of H5_file_image_op_t. These values indicate the operation being performed on the file image when this callback is invoked. Possible values for file_image_op are discussed in Table 2.udata
holds the value passed in for the udata parameter to H5Pset_file_image_callbacks.image_malloc
to NULL indicates that the HDF5 Library should invoke the standard C library malloc() routine when allocating file image buffers.- -
image_memcpy
contains a pointer to a function with (from the perspective of HDF5) functionality identical to the standard C library memcpy() call except that it returns NULL on failure. Recall that the memcpy C Library routine is defined to return the dest parameter in all cases. The parameters of the image_memcpy callback are defined as follows:dest
contains the address of the destination buffer.src
contains the address of the source buffer.size
contains the number of bytes to copy.file_image_op
contains one of the values of H5_file_image_op_t. These values indicate the operation being performed on the file image when this callback is invoked. Possible values for file_image_op are discussed in Table 2.udata
holds the value passed in for the udata parameter to H5Pset_file_image_callbacks.- -
image_realloc
contains a pointer to a function with (from the perspective of HDF5) functionality identical to the standard C library realloc() call. The parameters of the image_realloc callback are defined as follows:
ptr
contains the pointer to the buffer being reallocated.size
contains the desired size in bytes of the buffer after realloc.file_image_op
contains one of the values of H5_file_image_op_t. These values indicate the operation being performed on the file image when this callback is invoked. Possible values for file_image_op are discussed in Table 2.udata
holds the value passed in for the udata parameter to H5Pset_file_image_callbacks.- -
image_free
contains a pointer to a function with (from the perspective of HDF5) functionality identical to the standard C library free() call except that it will return 0 (SUCCEED) on success and -1 (FAIL) on failure. The parameters of the image_free callback are defined as follows:ptr
contains the pointer to the buffer being released.file_image_op
contains one of the values of H5_file_image_op_t. These values indicate the operation being performed on the file image when this callback is invoked. Possible values for file_image_op are discussed in Table 2 .udata
holds the value passed in for the udata parameter to H5Pset_file_image_callbacks.image_free
to NULL indicates that the HDF5 Library should invoke the standard C library free() routine when releasing file image buffers.- -
udata_copy
contains a pointer to a function that (from the perspective of HDF5) allocates a buffer of suitable size, copies the contents of the supplied udata into the new buffer, and returns the address of the new buffer. The function returns NULL on failure. This function is necessary if a non-NULL udata parameter is supplied, so that property lists containing the image callbacks can be copied. If the udata parameter (below) is NULL, then this parameter should be NULL as well. The parameter of the udata_copy callback is defined as follows:udata
contains the pointer to the user data block being copied.- -
udata_free
contains a pointer to a function that (from the perspective of HDF5) frees a user data block. This function is necessary if a non-NULL udata parameter is supplied so that property lists containing image callbacks can be discarded without a memory leak. If the udata parameter (below) is NULL, this parameter should be NULL as well. The parameter of the udata_free callback is defined as follows:udata
contains the pointer to the user data block to be freed.udata_free
returns 0 (SUCCEED) on success and -1 (FAIL) on failure.
-
- udata
contains a pointer value, potentially to user-defined data, that will be passed to the image_malloc, image_memcpy, image_realloc, and image_free callbacks.-
Value | -Comments | -
---|---|
H5_FILE_IMAGE_OP_PROPERTY_LIST_SET | -This value is passed to the image_malloc and image_memcpy callbacks when an image buffer is being copied while being set in a FAPL | -
H5_FILE_IMAGE_OP_PROPERTY_LIST_COPY | -This value is passed to the image_malloc and image_memcpy callbacks when an image buffer is being copied when a FAPL is copied | -
H5_FILE_IMAGE_OP_PROPERTY_LIST_GET | -This value is passed to the image_malloc and image_memcpy callbacks when an image buffer is being copied while being retrieved from a FAPL | -
H5_FILE_IMAGE_OP_PROPERTY_LIST_CLOSE | -This value is passed to the image_free callback when an image buffer is being released during a FAPL close operation. | -
H5_FILE_IMAGE_OP_FILE_OPEN | -This value is passed to the image_malloc and image_memcpy callbacks when an image buffer is copied during a file open operation. While the image being opened will typically be copied from a FAPL, this need not always be the case. An example of an exception is when the Core file driver takes its initial image from a file. | -
H5_FILE_IMAGE_OP_FILE_RESIZE | -This value is passed to the image_realloc callback when a file driver needs to resize an image buffer. | -
H5_FILE_IMAGE_OP_FILE_CLOSE | -This value is passed to the image_free callback when an image buffer is being released during a file close operation. | -
In closing our discussion of H5Pset_file_image_callbacks(), we note the interaction between this call and the H5Pget/set_file_image() calls above: since the malloc, memcpy, and free callbacks defined in the instance of H5_file_image_callbacks_t are used by H5Pget/set_file_image(), H5Pset_file_image_callbacks() will fail if a file image is already set in the target property list.
-For more information on writing the file image to disk, set the backing_store
parameter. See the H5Pset_fapl_core entry in the HDF5 Reference Manual.
The H5Pget_file_image_callbacks routine is designed to obtain the current file image callbacks from a file access property list.
-The signature of H5Pget_file_image_callbacks() is defined as follows:
- -herr_t H5Pget_file_image_callbacks(hid_t fapl_id,
- H5_file_image_callbacks_t *callbacks_ptr)
fapl_id
contains the ID of the target file access property list.callbacks_ptr
contains a pointer to an instance of the H5_file_image_callbacks_t structure. All fields should be initialized to NULL. See the “H5Pset_file_image_callbacks” section for more information on the H5_file_image_callbacks_t structure.H5Pget/set_file_image_callbacks()
and H5Pget/set_file_image()
function calls requires a pair of virtual file driver feature flags. The flags are H5FD_FEAT_ALLOW_FILE_IMAGE and H5FD_FEAT_CAN_USE_FILE_IMAGE_CALLBACKS. Both of these are defined in H5FDpublic.h.
-
-The first flag, H5FD_FEAT_ALLOW_FILE_IMAGE, allows a file driver to indicate whether or not it supports file images. A VFD that sets this flag when its ‘query’ callback is invoked indicates that the file image set in the FAPL will be used as the initial contents of a file. Support for setting an initial file image is designed primarily for use with the Core VFD. However, any VFD can indicate support for this feature by setting the flag and copying the image in an appropriate way for the VFD (possibly by writing the image to a file and then opening the file). However, such a VFD need not employ the file image after file open time. In such cases, the VFD will not make an in-memory copy of the file image and will not employ the file image callbacks.
- -File drivers that maintain a copy of the file in memory (only the Core file driver at present) can be constructed to use the initial image callbacks (if defined). Those that do must set the H5FD_FEAT_CAN_USE_FILE_IMAGE_CALLBACKS flag, the second flag, when their ‘query’ callbacks are invoked.
- -Thus file drivers that set the H5FD_FEAT_ALLOW_FILE_IMAGE flag but not the H5FD_FEAT_CAN_USE_FILE_IMAGE_CALLBACKS flag may read the supplied image from the property list (if present) and use it to initialize the contents of the file. However, they will not discard the image when done, nor will they make any use of any file image callbacks (if defined).
- -If an initial file image appears in a file allocation property list that is used in an H5Fopen() call, and if the underlying file driver does not set the H5FD_FEAT_ALLOW_FILE_IMAGE flag, then the open will fail.
- -If a driver sets both the H5FD_FEAT_ALLOW_FILE_IMAGE flag and the H5FD_FEAT_CAN_USE_FILE_IMAGE_CALLBACKS flag, then that driver will allocate a buffer of the required size, copy the contents of the initial image buffer from the file access property list, and then open the copy as if it had just loaded it from file. If the file image allocation callbacks are defined, the driver shall use them for all memory management tasks. Otherwise it will use the standard malloc, memcpy, realloc, and free C library calls for this purpose.
- -If the VFD sets the H5FD_FEAT_ALLOW_FILE_IMAGE flag, and an initial file image is defined by an application, the VFD should ensure that file creation operations (as opposed to file open operations) bypass use of the file image, and create a new, empty file.
- -Finally, it is logically possible that a file driver would set the H5FD_FEAT_CAN_USE_FILE_IMAGE_CALLBACKS flag, but not the H5FD_FEAT_ALLOW_FILE_IMAGE flag. While it is hard to think of a situation in which this would be desirable, setting the flags this way will not cause any problems: the two capabilities are logically distinct.
- -The purpose of the H5Fget_file_image routine is to provide a simple way to retrieve a copy of the image of an existing, open file. This routine can be used with files opened using the SEC2 (aka POSIX), STDIO, and Core (aka Memory) VFDs.
- -The signature of H5Fget_file_image is defined as follows:
-ssize_t H5Fget_file_image(hid_t file_id, void *buf_ptr, size_t buf_len)
file_id
contains the ID of the target file.buf_ptr
contains a pointer to the buffer into which the image of the HDF5 file is to be copied. If buf_ptr is NULL, no data will be copied, but the return value will still indicate the buffer size required (or a negative value on error).buf_len
contains the size of the supplied buffer.If the return value of H5Fget_file_image is a positive value, then the value will be the length of buffer required to store the file image (in other words, the length of the file). A negative value might be returned if the file is too large to store in the supplied buffer or on failure.
- -The current file size can be obtained via a call to H5Fget_filesize(). Note that this function returns the value of the end of file (EOF) and not the end of address space (EOA). While these values are frequently the same, it is possible for the EOF to be larger than the EOA. Since H5Fget_file_image() will only obtain a copy of the file from the beginning of the superblock to the EOA, it will be best to use H5Fget_file_image() to determine the size of the buffer required to contain the image.
- -Other Design Considerations - -Here are some other notes regarding the design and implementation of H5Fget_file_image.
-The H5Fget_file_image call should be part of the high-level library. However, a file driver agnostic implementation of the routine requires access to data structures that are hidden within the HDF5 Library. We chose to implement the call in the library proper rather than expose those data structures.
-There is no reason why the H5Fget_file_image() API call could not work on files opened with any file driver. However, the Family, Multi, and Split file drivers have issues that make the call problematic. At present, files opened with the Family file driver are marked as being created with that file driver in the superblock, and the HDF5 Library refuses to open files so marked with any other file driver. This negates the purpose of the H5Fget_file_image() call. While this mark can be removed from the image, the necessary code is not trivial.
-Thus we will not support the Family file driver in H5Fget_file_image() unless there is demand for it. Files created with the Multi and Split file drivers are also marked in the superblock. In addition, they typically use a very sparse address space. A sparse address space would require the use of an impractically large buffer for an image, and most of the buffer would be empty. So, we see no point in supporting the Multi and Split file drivers in H5Fget_file_image() under any foreseeable circumstances.
- -The H5LTopen_file_image high-level routine encapsulates the capabilities of routines in the main HDF5 Library with conveniently accessible abstractions.
- -The H5LTopen_file_image routine is designed to provide an easier way to open an initial file image with the Core VFD. Flags to H5LTopen_file_image allow for various file image buffer ownership policies to be requested. See the HDF5 Reference Manual for more information on high-level APIs.
-The signature of H5LTopen_file_image is defined as follows:
- -hid_t H5LTopen_file_image(void *buf_ptr, size_t buf_len, unsigned flags)
-
-
The parameters of H5LTopen_file_image are defined as follows:
-buf_ptr
contains a pointer to the supplied initial image. A NULL value is invalid and will cause H5LTopen_file_image to fail.buf_len
contains the size of the supplied buffer. A value of 0 is invalid and will cause H5LTopen_file_image to fail.flags
contains a set of flags indicating whether the image is to be opened read/write, whether HDF5 is to take control of the buffer, and how long the application promises to maintain the buffer. Possible flags are described in the table below:Value | -Comments | -
---|---|
H5LT_FILE_IMAGE_OPEN_RW | -Indicates that the HDF5 Library should open the image read/write instead of the default read-only. | -
H5LT_FILE_IMAGE_DONT_COPY | -
-
|
-
H5LT_FILE_IMAGE_DONT_RELEASE | -
-
|
-
The following table is intended to summarize the semantics of the H5LT_FILE_IMAGE_DONT_COPY and H5LT_FILE_IMAGE_DONT_RELEASE flags (shown as “Don’t Copy Flag” and “Don’t Release Flag” respectively in the table):
- -Don’t Copy Flag | -Don’t Release Flag | -Make Copy of User Supplied Buffer | -Pass User Supplied Buffer to File Driver | -Release User Supplied Buffer When Done | -Permit realloc of Buffer Used by File Driver | -
---|---|---|---|---|---|
False | -Don’t care | -True | -False | -False | -True | -
True | -False | -False | -True | -True | -True | -
True | -True | -False | -True | -False | -False | -
The return value of H5LTopen_file_image will be a file ID on success or a negative value on failure. The file ID returned should be closed with H5Fclose.
-Note that there is no way currently to specify a “backing store” file name in this definition of H5LTopen_image.
- -The purpose of this chapter is to describe some issues that developers should consider when using file image buffers, property lists, and callback APIs.
- -The H5Fget/set_file_image_callbacks() API calls allow an application to hook the memory management operations used when allocating, duplicating, and discarding file images in the property list, in the Core file driver, and potentially in any in-memory file driver developed in the future.
-From the perspective of the HDF5 Library, the supplied image_malloc(), image_memcpy(), image_realloc(), and image_free() callback routines must function identically to the C standard library malloc(), memcpy(), realloc(), and free() calls. What happens on the application side can be much more nuanced, particularly with the ability to pass user data to the callbacks. However, whatever the application does with these calls, it must maintain the illusion that the calls have had the expected effect. Maintaining this illusion requires some understanding of how the property list structure works, and what HDF5 will do with the initial images passed to it.
-At the beginning of this document, we talked about the need to work within the constraints of the property list mechanism. When we said “from the perspective of the HDF5 Library…” in the paragraph above, we are making reference to this point.
-The property list mechanism was developed as a way to add parameters to functions without changing the parameter list and breaking existing code. However, it was designed to use only “call by value” semantics, not “call by reference”. The decision to use “call by value” semantics requires that the values of supplied variables be copied into the property list. This has the advantage of simplifying the copying and deletion of property lists. However, if the value to be copied is large (say a 2 GB file image), the overhead can be unacceptable.
-The usual solution to this problem is to use “call by reference” where only a pointer to an object is placed in a parameter list rather than a copy of the object itself. However, use of “call by reference” semantics would greatly complicate the property list mechanism: at a minimum, it would be necessary to maintain reference counts to dynamically allocated objects so that the owner of the object would know when it was safe to free the object.
-After much discussion, we decided that the file image operations calls were sufficiently specialized that it made no sense to rework the property list mechanism to support “call by reference.” Instead we provided the file image callback mechanism to allow the user to implement some version of “call by reference” when needed. It should be noted that we expect this mechanism to be used rarely if at all. For small file images, the copying overhead should be negligible, and for large images, most use cases should be addressed by the H5LTopen_file_image call.
-In the (hopefully) rare event that use of the file image callbacks is necessary, the fundamental point to remember is that the callbacks must be constructed and used in such a way as to maintain the library’s illusion that it is using “call by value” semantics.
-Thus the property list mechanism must think that it is allocating a new buffer and copying the supplied buffer into it when the file image property is set. Similarly, it must think that it is allocating a new buffer and copying the contents of the existing buffer into it when it copies a property list that contains a file image. Likewise, it must think it is de-allocating a buffer when it discards a property list that contains a file image.
-Similar illusions must be maintained when a file image buffer is copied into the Core file driver (or any future driver that uses the file image callbacks) when the file driver re-sizes the buffer containing the image and finally when the driver discards the buffer.
- -The owner of a file image in a buffer is the party that has the responsibility to discard the file image buffer when it is no longer needed. In this context, the owner is either the HDF5 Library or the application program.
-We implemented the image_* callback facility to allow efficient management of large file images. These facilities can be used to allow sharing of file image buffers between the application and the HDF5 library, and also transfer of ownership in either direction. In such operations, care must be taken to ensure that ownership is clear and that file image buffers are not discarded before all references to them are discarded by the non-owning party.
-Ownership of a file image buffer will only be passed to the application program if the file image callbacks are designed to do this. In such cases, the application program must refrain from freeing the buffer until the library has deleted all references to it. This in turn will happen after all property lists (if any) that refer to the buffer have been discarded, and the file driver (if any) that used the buffer has closed the file and thinks it has discarded the buffer.
- -As mentioned above, the HDF5 property lists are a mechanism for passing values into HDF5 Library calls. They were created to allow calls to be extended with new parameters without changing the actual API or breaking existing code. They were designed based on the assumption that all new parameters would be “call by value” and not “call by reference.” Having “call by value” parameters means property lists can be copied, reused, and discarded with ease.
-Suppose an application wished to share a file image buffer with the HDF5 Library. This means the library would be allowed to read the file image, but not free it. The file image callbacks might be constructed as follows to share a buffer:
- -For more information on user defined data, see the “H5Pset_file_image_callbacks” section.
- -When a file image is opened by a driver that sets both the H5FD_FEAT_ALLOW_FILE_IMAGE and the H5FD_FEAT_CAN_USE_FILE_IMAGE_CALLBACKS flags, the driver will allocate a buffer large enough for the initial file image and then copy the image from the property list into this buffer. As processing progresses, the driver will reallocate the image as necessary to increase its size and will eventually discard the image at file close. If defined, the driver will use the file image callbacks for these operations; otherwise, the driver will use the standard C library calls. See the "H5Pset_file_image_callbacks” section for more information.
-As described above, the file image callbacks can be constructed so as to avoid the overhead of buffer allocations and copies while allowing the HDF5 Library to maintain its illusions on the subject. There are two possible complications involving the file driver. The complications are the possibility of reallocation calls from the driver and the possibility of the continued existence of property lists containing references to the buffer.
-Suppose an application wishes to share a file image buffer with the HDF5 Library. The application allows the library to read (and possibly write) the image, but not free it. We must first decide whether the image is to be opened read-only or read/write.
-If the image will be opened read-only (or if we know that any writes will not change the size of the image), the image_realloc() call should never be invoked. Thus the image_realloc() routine can be constructed so as to always fail, and the image_malloc(), image_memcpy(), and image_free() routines can be constructed as described in the section above.
-Suppose, however, that the file image will be opened read/write and may grow during the computation. We must now allow for the base address of the buffer to change due to reallocation calls, and we must employ the user data structure to communicate any change in the buffer base address and size to the application. We pass buffer changes to the application so that the application will be able to eventually free the buffer. To this end, we might define a user data structure as shown in the example below:
- typedef struct udata {
- void *init_ptr;
- size_t init_size;
- int init_ref_count;
- void *mod_ptr;
- size_t mod_size;
- int mod_ref_count;
- }
-
-Example 1. Using a user data structure to communicate with an application
-We initialize an instance of the structure so that init_ptr points to the buffer to be shared, init_size contains the initial size of the buffer, and all other fields are initialized to either NULL or 0 as indicated by their type. We then pass a pointer to the instance of the user data structure to the HDF5 Library along with allocation callback functions constructed as follows:
- -One can argue whether creating a file with an initial file image is closer to creating a file or opening a file. The consensus seems to be that it is closer to a file open, and thus we shall require that the initial image only be used for calls to H5Fopen().
-Whatever our convention, from an internal perspective, opening a file with an initial file image is a bit of both creating a file and opening a file. Conceptually, we will create a file on disk, write the supplied image to the file, close the file, open the file as an HDF5 file, and then proceed as usual (of course, the Core VFD will not write to the file system unless it is configured to do so). This process is similar to a file create: we are creating a file that did not exist on disk to begin with and writing data to it. Also, we must verify that no file of the supplied name is open. However, this process is also similar to a file open: we must read the superblock and handle the usual file open tasks.
-Implementing the above sequence of actions has a number of implications on the behavior of the H5Fopen() call when an initial file image is supplied:
-As we indicated earlier, if an initial file image appears in the property list of an H5Fcreate() call, it is ignored.
-While the above section on the semantics of the file image callbacks may seem rather gloomy, we get the payback here. The above says everything that needs to be said about initial file image semantics in general. The sub-section below has a few more observations on the Core file driver.
- -At present, the Core file driver uses the open()
and read()
system calls to load an HDF5 file image from the file system into RAM. Further, if the backing_store
flag is set in the FAPL entry specifying the use of the Core file driver, the Core file driver’s internal image will be used to overwrite the source file on either flush or close. See the H5Pset_fapl_core entry in the HDF5 Reference Manual for more information.
This results in the following observations. In all cases assume that use of the Core file driver has been specified in the FAPL.
-The purpose of this chapter is to provide examples of how to read or build an in-memory HDF5 file image.
- -The H5Pset_file_image() function call allows the Core file driver to be initialized from an application provided buffer. The following pseudo code illustrates its use:
- -
-<allocate and initialize buf_len and buf>
-<allocate fapl_id>
-<set fapl to use Core file driver>
-H5Pset_file_image(fapl_id, buf, buf_len);
-<discard buf any time after this point>
-<open file>
-<discard fapl any time after this point>
-<read and/or write file as desired, close>
-
-
-Example 2. Using H5Pset_file_image to initialize the Core file driver
-
-This solution is easy to code, but the supplied buffer is duplicated twice. The first time is in the call to H5Pset_file_image() when the image is duplicated and the duplicate inserted into the property list. The second time is when the file is opened: the image is copied from the property list into the initial buffer allocated by the Core file driver. This is a non-issue for small images, but this could become a significant performance hit for large images.
-If we want to avoid the extra malloc and memcpycalls, we must decide whether the application should retain ownership of the buffer or pass ownership to the HDF5 Library.
-The following pseudo code illustrates opening the image read -only using the H5LTopen_file_image() routine. In this example, the application retains ownership of the buffer and avoids extra buffer allocations and memcpy calls.
- -
-<allocate and initialize buf_len and buf>
-hid_t file_id;
-unsigned flags = H5LT_FILE_IMAGE_DONT_COPY | H5LT_FILE_IMAGE_DONT_RELEASE;
-file_id = H5LTopen_file_image(buf, buf_len, flags);
-<read file as desired, and then close>
-<discard buf any time after this point>
-
-
-Example 3. Using H5LTopen_file_image to open a read-only file image where the application retains ownership of the buffer
-
-If the application wants to transfer ownership of the buffer to the HDF5 Library, and the standard C library routine free is an acceptable way of discarding it, the above example can be modified as follows:
- -
-<allocate and initialize buf_len and buf>
-hid_t file_id;
-unsigned flags = H5LT_FILE_IMAGE_DONT_COPY;
-file_id = H5LTopen_file_image(buf, buf_len, flags);
-<read file as desired, and then close>
-
-
-Example 4. Using H5LTopen_file_image to open a read-only file image where the application transfers ownership of the buffer
-
-Again, file access is read-only. Read/write access can be obtained via the H5LTopen_file_image() call, but we will explore that in the section below.
- -Before the implementation of file image operations, HDF5 supported construction of an image of an HDF5 file in memory with the Core file driver. The H5Fget_file_image() function call allows an application access to the file image without first writing it to disk. See the following code fragment:
- -
-<Open and construct the desired file with the Core file driver>
-H5Fflush(fid);
-size = H5Fget_file_image(fid, NULL, 0);
-buffer_ptr = malloc(size);
-H5Fget_file_image(fid, buffer_ptr, size);
-
-
-Example 5. Accessing the image of a file in memory
-
-The use of H5Fget_file_image() may be acceptable for small images. For large images, the cost of the malloc() and memcpy() operations may be excessive. To address this issue, the H5Pset_file_image_callbacks() call allows an application to manage dynamic memory allocation for file images and memory-based file drivers (only the Core file driver at present). The following code fragment illustrates its use. Note that most error checking is omitted for simplicity and that H5Pset_file_image is not used to set the initial file image.
- -
- struct udata_t {
- void * image_ptr;
- size_t image_size;
- } udata = {NULL, 0};
-
-void *image_malloc(size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- ((struct udata_t *)udata)->image_size = size;
- return(malloc(size));
-}
-
-void *image_memcpy)(void *dest, const void *src, size_t size,
- H5_file_image_op_t file_image_op, void *udata)
-{
- assert(FALSE); /* Should never be invoked in this scenario. */
- return(NULL); /* always fails */
-}
-
-void image_realloc(void *ptr, size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- ((struct udata_t *)udata)->image_size = size;
- return(realloc(ptr, size));
-}
-
-herr_t image_free(void *ptr, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(file_image_op == H5_FILE_IMAGE_OP_FILE_CLOSE);
- ((struct udata_t *)udata)->image_ptr = ptr;
- return(0); /* if we get here, we must have been successful */
-}
-
-void *udata_copy(void *udata)
-{
- return(udata);
-}
-
-herr_t udata_free(void *udata)
-{
- return(0);
-}
-
-H5_file_image_callbacks_t callbacks = {image_malloc, image_memcpy,
- image_realloc, image_free,
- udata_copy, udata_free,
- (void *)(&udata)};
-
-<allocate fapl_id>
-H5Pset_file_image_callbacks(fapl_id, &callbacks);
-<open core file using fapl_id, write file, close it>
-assert(udata.image_ptr!= NULL);
-/* udata now contains the base address and length of the final version of the core file */
-<use image of file, and then discard it via free()>
-
-
-Example 6. Using H5Pset_file_image_callbacks to improve memory allocation
-
-The above code fragment gives the application full ownership of the buffer used by the Core file driver after the file is closed, and it notifies the application that the HDF5 Library is done with the buffer by setting udata.image_ptr to something other than NULL. If read access to the buffer is sufficient, the H5Fget_vfd_handle() call can be used as an alternate solution to get access to the base address of the Core file driver’s buffer.
-The above solution avoids some unnecessary mallocand memcpycalls and should be quite adequate if an image of an HDF5 file is constructed only occasionally. However, if an HDF5 file image must be constructed regularly, and if we can put a strong and tight upper bound on the size of the necessary buffer, then the following pseudo code demonstrates a method of avoiding memory allocation completely. The downside, however, is that buffer is allocated statically. Again, much error checking is omitted for clarity.
- -
-char buf[BIG_ENOUGH];
-struct udata_t {
-void * image_ptr;
-size_t image_size;
-size_t max_image_size;
-int ref_count;
-} udata = {(void *)(&(buf[0]), 0, BIG_ENOUGH, 0};
-
-void *image_malloc(size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(size <= ((struct udata_t *)udata)->max_image_size);
- assert(((struct udata_t *)udata)->ref_count == 0);
- ((struct udata_t *)udata)->image_size = size;
- (((struct udata_t *)udata)->ref_count)++;
- return((((struct udata_t *)udata)->image_ptr);
-}
-void *image_memcpy)(void *dest, const void *src, size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(FALSE); /* Should never be invoked in this scenario. */
- return(NULL); /* always fails */
-}
-void *image_realloc(void *ptr, size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(ptr == ((struct udata_t *)udata)->image_ptr);
- assert(size <= ((struct udata_t *)udata)->max_image_size);
- assert(((struct udata_t *)udata)->ref_count == 1);
- ((struct udata_t *)udata)->image_size = size;
- return((((struct udata_t *)udata)->image_ptr);
-}
-herr_t image_free(void *ptr, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(file_image_op == H5_FILE_IMAGE_OP_FILE_CLOSE);
- assert(ptr == ((struct udata_t *)udata)->image_ptr);
- assert(((struct udata_t *)udata)->ref_count == 1);
- (((struct udata_t *)udata)->ref_count)--;
- return(0); /* if we get here, we must have been successful */
-}
-void *udata_copy(void *udata)
-{
- return(udata);
-}
-herr_t udata_free(void *udata)
-{
- return(0);
-}
-H5_file_image_callbacks_t callbacks = {image_malloc, image_memcpy,
- image_realloc, image_free,
- udata_copy, udata_free,
- (void *)(&udata)};
-/* end of initialization */
-<allocate fapl_id>
-H5Pset_file_image_callbacks(fapl_id, &callbacks);
-<open core file using fapl_id>
-<discard fapl any time after the open>
-<write the file, flush it, and then close it>
-assert(udata.ref_count == 0);
-/* udata now contains the base address and length of the final version of the core file */
-<use the image of the file>
-<reinitialize udata, and repeat the above from the end of initialization onwards to write a new file image>
-
-
-Example 7. Using H5Pset_file_image_callbacks with a static buffer
-
-If we can further arrange matters so that only the contents of the datasets in the HDF5 file image change, but not the structure of the file itself, we can optimize still further by re-using the image and changing only the contents of the datasets after the initial write to the buffer. The following pseudo code shows how this might be done. Note that the code assumes that buf already contains the image of the HDF5 file whose dataset contents are to be overwritten. Again, much error checking is omitted for clarity. Also, observe that the file image callbacks do not support the H5Pget_file_image() call.
- -
-<buf already defined and loaded with file image>
-<udata already defined and initialized>
-void *image_malloc(size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(size <= ((struct udata_t *)udata)->max_image_size);
- assert(size == ((struct udata_t *)udata)->image_size);
- assert(((struct udata_t *)udata)->ref_count >= 0);
- ((struct udata_t *)udata)->image_size = size;
- (((struct udata_t *)udata)->ref_count)++;
- return((((struct udata_t *)udata)->image_ptr);
-}
-void *image_memcpy)(void *dest, const void *src, size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(dest == ((struct udata_t *)udata)->image_ptr);
- assert(src == ((struct udata_t *)udata)->image_ptr);
- assert(size <= ((struct udata_t *)udata)->max_image_size);
- assert(size == ((struct udata_t *)udata)->image_size);
- assert(((struct udata_t *)udata)->ref_count >= 1);
- return(dest); /* if we get here, we must have been successful */
-}
-void *image_realloc(void *ptr, size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- /* One would think that this function is not needed in this scenario, as
- * only the contents of the HDF5 file is being changed, not its size or
- * structure. However, the Core file driver calls realloc() just before
- * close to clip the buffer to the size indicated by the end of the
- * address space.
- *
- * While this call must be supported in this case, the size of
- * the image should never change. Hence the function can limit itself
- * to performing sanity checks, and returning the base address of the
- * statically allocated buffer.
- */
- assert(ptr == ((struct udata_t *)udata)->image_ptr);
- assert(size <= ((struct udata_t *)udata)->max_image_size);
- assert(((struct udata_t *)udata)->ref_count >= 1);
- assert(((struct udata_t *)udata)->image_size == size);
- return((((struct udata_t *)udata)->image_ptr);
-}
-herr_t image_free(void *ptr, H5_file_image_op_t file_image_op, void *udata)
-{
- assert((file_image_op == H5_FILE_IMAGE_OP_PROPERTY_LIST_CLOSE) ||
- (file_image_op == H5_FILE_IMAGE_OP_FILE_CLOSE));
- assert(((struct udata_t *)udata)->ref_count >= 1);
- (((struct udata_t *)udata)->ref_count)--;
- return(0); /* if we get here, we must have been successful */
-}
-void *udata_copy(void *udata)
-{
- return(udata);
-}
-herr_t udata_free(void *udata)
-{
- return(0);
-}
-H5_file_image_callbacks_t callbacks = {image_malloc, image_memcpy,
- image_realloc, image_free,
- udata_copy, udata_free,
- (void *)(&udata)};
-/* end of initialization */
-<allocate fapl_id>
-H5Pset_file_image_callbacks(fapl_id, &callbacks);
-H5Pset_file_image(fapl_id, udata.image_ptr, udata.image_len);
-<open core file using fapl_id>
-<discard fapl any time after the open>
-<overwrite data in datasets in the file, and then close it>
-assert(udata.ref_count == 0);
-/* udata now contains the base address and length of the final version of the core file */
-<use the image of the file>
-<repeat the above from the end of initialization onwards to write new data to datasets in file image>
-
-
-Example 8. Using H5Pset_file_image_callbacks where only the datasets change
-
-Before we go on, we should note that the above pseudo code can be written more compactly, albeit with fewer sanity checks, using the H5LTopen_file_image() call. See the example below:
- -
-<buf already defined and loaded with file image>
-<udata already defined and initialized>
-hid_t file_id;
-unsigned flags = H5LT_FILE_IMAGE_OPEN_RW | H5LT_FILE_IMAGE_DONT_COPY | H5LT_FILE_IMAGE_DONT_RELEASE;
-/* end initialization */
-file_id = H5LTopen_file_image(udata.image_ptr, udata.image_len, flags);
-<overwrite data in datasets in the file, and then close it>
-/* udata now contains the base address and length of the final version of the core file */
-<use the image of the file>
-<repeat the above from the end of initialization onwards to write new data to datasets in file image>
-
-
-Example 9. Using H5LTopen_file_image where only the datasets change -
The above pseudo code allows updates of a file image about as cheaply as possible. We assume the application has enough RAM for the image and that the HDF5 file structure is constant after the first write.
-While the scenario above is plausible, we will finish this section with a more general scenario. In the pseudo code below, we assume sufficient RAM to retain the HDF5 file image between uses, but we do not assume that the HDF5 file structure remains constant or that we can place a hard pper bound on the image size.
-Since we must use malloc, realloc, and free in this example, and since realloc can change the base address of a buffer, we must maintain two of ptr, size, and ref_count triples in the udata structure. The first triple is for the property list (which will never change the buffer), and the second triple is for the file driver. As shall be seen, this complicates the file image callbacks considerably. Note also that while we do not use H5Pget_file_image() in this example, we do include support for it in the file image callbacks. As usual, much error checking is omitted in favor of clarity.
- -
-struct udata_t {
- void * fapl_image_ptr;
- size_t fapl_image_size;
- int fapl_ref_count;
- void * vfd_image_ptr;
- size_t vfd_image_size;
- nt vfd_ref_count;
- } udata = {NULL, 0, 0, NULL, 0, 0};
-boolean initial_file_open = TRUE;
-
-void *image_malloc(size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- void * return_value = NULL;
- switch ( file_image_op ) {
- case H5_FILE_IMAGE_OP_PROPERTY_LIST_SET:
- case H5_FILE_IMAGE_OP_PROPERTY_LIST_COPY:
- assert(((struct udata_t *)udata)->fapl_image_ptr != NULL);
- assert(((struct udata_t *)udata)->fapl_image_size == size);
- assert(((struct udata_t *)udata)->fapl_ref_count >= 0);
- return_value = ((struct udata_t *)udata)->fapl_image_ptr;
- (((struct udata_t *)udata)->fapl_ref_count)++;
- break;
- case H5_FILE_IMAGE_OP_PROPERTY_LIST_GET:
- assert(((struct udata_t *)udata)->fapl_image_ptr != NULL);
- assert(((struct udata_t *)udata)->vfd_image_size == size);
- assert(((struct udata_t *)udata)->fapl_ref_count >= 1);
- return_value = ((struct udata_t *)udata)->fapl_image_ptr;
- /* don’t increment ref count */
- break;
- case H5_FILE_IMAGE_OP_FILE_OPEN:
- assert(((struct udata_t *)udata)->vfd_image_ptr == NULL);
- assert(((struct udata_t *)udata)->vfd_image_size == 0);
- assert(((struct udata_t *)udata)->vfd_ref_count == 0);
- if (((struct udata_t *)udata)->fapl_image_ptr == NULL ) {
- ((struct udata_t *)udata)->vfd_image_ptr = malloc(size);
- ((struct udata_t *)udata)->vfd_image_size = size;
- } else {
- assert(((struct udata_t *)udata)->fapl_image_size == size);
- assert(((struct udata_t *)udata)->fapl_ref_count >= 1);
- ((struct udata_t *)udata)->vfd_image_ptr = ((struct udata_t *)udata)->fapl_image_ptr;
- ((struct udata_t *)udata)->vfd_image_size = size;
- }
- return_value = ((struct udata_t *)udata)->vfd_image_ptr;
- (((struct udata_t *)udata)->vfd_ref_count)++;
- break;
- default:
- assert(FALSE);
- }
- return(return_value);
-}
-
-void *image_memcpy)(void *dest, const void *src, size_t size,
- H5_file_image_op_t file_image_op, void *udata)
-{
- switch(file_image_op) {
- case H5_FILE_IMAGE_OP_PROPERTY_LIST_SET:
- case H5_FILE_IMAGE_OP_PROPERTY_LIST_COPY:
- case H5_FILE_IMAGE_OP_PROPERTY_LIST_GET:
- assert(dest == ((struct udata_t *)udata)->fapl_image_ptr);
- assert(src == ((struct udata_t *)udata)->fapl_image_ptr);
- assert(size == ((struct udata_t *)udata)->fapl_image_size);
- assert(((struct udata_t *)udata)->fapl_ref_count >= 1);
- break;
- case H5_FILE_IMAGE_OP_FILE_OPEN:
- assert(dest == ((struct udata_t *)udata)->vfd_image_ptr);
- assert(src == ((struct udata_t *)udata)->fapl_image_ptr);
- assert(size == ((struct udata_t *)udata)->fapl_image_size);
- assert(size == ((struct udata_t *)udata)->vfd_image_size);
- assert(((struct udata_t *)udata)->fapl_ref_count >= 1);
- assert(((struct udata_t *)udata)->vfd_ref_count == 1);
- break;
- default:
- assert(FALSE);
- break;
- }
- return(dest); /* if we get here, we must have been successful */
-}
-
-void *image_realloc(void *ptr, size_t size, H5_file_image_op_t file_image_op, void *udata)
-{
- assert(ptr == ((struct udata_t *)udata)->vfd_image_ptr); |
- assert(((struct udata_t *)udata)->vfd_ref_count == 1);
- ((struct udata_t *)udata)->vfd_image_ptr = realloc(ptr, size);
- ((struct udata_t *)udata)->vfd_image_size = size;
- return((((struct udata_t *)udata)->vfd_image_ptr);
-}
-
-herr_t image_free(void *ptr, H5_file_image_op_t file_image_op, void *udata)
-{
- switch(file_image_op) {
- case H5_FILE_IMAGE_OP_PROPERTY_LIST_CLOSE:
- assert(ptr == ((struct udata_t *)udata)->fapl_image_ptr);
- assert(((struct udata_t *)udata)->fapl_ref_count >= 1);
- (((struct udata_t *)udata)->fapl_ref_count)--;
- break;
- case H5_FILE_IMAGE_OP_FILE_CLOSE:
- assert(ptr == ((struct udata_t *)udata)->vfd_image_ptr);
- assert(((struct udata_t *)udata)->vfd_ref_count == 1);
- (((struct udata_t *)udata)->vfd_ref_count)--;
- break;
- default:
- assert(FALSE);
- break;
- }
- return(0); /* if we get here, we must have been successful */
-}
-
-void *udata_copy(void *udata)
-{
- return(udata);
-}
-
-herr_t udata_free(void *udata)
-{
- return(0);
-}
-H5_file_image_callbacks_t callbacks = {image_malloc, image_memcpy,
- image_realloc, image_free,
- udata_copy, udata_free,
- (void *)(&udata)};
-/* end of initialization */
-<allocate fapl_id>
-H5Pset_file_image_callbacks(fapl_id, &callbacks);
-if ( initial_file_open ) {
- initial_file_open = FALSE;
-} else {
- assert(udata.vfd_image_ptr != NULL);
- assert(udata.vfd_image_size > 0);
- assert(udata.vfd_ref_count == 0);
- assert(udata.fapl_ref_count == 0);
- udata.fapl_image_ptr = udata.vfd_image_ptr;
- udata.fapl_image_size = udata.vfd_image_size;
- udata.vfd_image_ptr = NULL;
- udata.vfd_image_size = 0;
- H5Pset_file_image(fapl_id, udata.fapl_image_ptr, udata.fapl_image_size);
-}
-
-<open core file using fapl_id>
-<discard fapl any time after the open>
-<write/update the file, and then close it>
-assert(udata.fapl_ref_count == 0);
-assert(udata.vfd_ref_count == 0);
-/* udata.vfd_image_ptr and udata.vfd_image_size now contain the base address and length of the final version of the core file */
-<use the image of the file>
-<repeat the above from the end of initialization to modify the file image as needed>
-<free the image when done>
-
-
-Example 10. Using H5LTopen_file_image where only the datasets change and where the file structure and image size might not be constant
-
-The above pseudo code shows how a buffer can be passed back and forth between the application and the HDF5 Library. The code also shows the application having control of the actual allocation, reallocation, and freeing of the buffer.
- -Using the file image operations described in this document, we can bundle up data in an image of an HDF5 file on one process, transmit the image to a second process, and then open and read the image on the second process without any mandatory file system I/O.
-We have already demonstrated the construction and reading of such buffers above, but it may be useful to offer an example of the full operation. We do so in the example below using as simple a set of calls as possible. The set of calls in the example has extra buffer allocations. To reduce extra buffer allocations, see the sections above.
-In the following example, we construct an HDF5 file image on process A and then transmit the image to process B where we then open the image and extract the desired data. Note that no file system I/O is performed: all the processing is done in memory with the Core file driver.
- -*** Process A *** | -*** Process B *** | -
---|---|
<Open and construct the desired file with the Core file driver> | -hid_t file_id; | -
H5Fflush(fid); | -- |
size = H5Fget_file_image(fid, NULL, 0); | -- |
buffer_ptr = malloc(size); | -- |
H5Fget_file_image(fid, buffer_ptr, size); | -- |
<transmit size> | -<receive size> |
-
<transmit *buffer_ptr> | -buffer_ptr = malloc(size) | -
free(buffer_ptr); | -<receive image in *buffer_ptr> | -
<close core file> | -file_id = H5LTopen_file_image(buf, - buf_len, - H5LT_FILE_IMAGE_DONT_COPY); | -
- | <read data from file, then close. note that the Core file driver will discard the buffer on close> | -
After the above examples, an example of the use of a template file might seem anti-climactic. A template file might be used to enforce consistency on file structure between files or in parallel HDF5 to avoid long sequences of collective operations to create the desired groups, datatypes, and possibly datasets. The following pseudo code outlines a potential use:
- -
-<allocate and initialize buf and buflen, with buf containing the desired initial image (which in turn contains the desired group, datatype, and dataset definitions), and buf_len containing the size of buf>
-<allocate fapl_id>
-<set fapl to use desired file driver that supports initial images>
-H5Pset_file_image(fapl_id, buf, buf_len);
-<discard buf any time after this point>
-<open file>
-<discard fapl any time after this point>
-<read and/or write file as desired, close>
-
-
-Example 12. Using a template file
-
-Observe that the above pseudo code includes an unnecessary buffer allocation and copy in the call to H5Pset_file_image(). As we have already discussed ways of avoiding this, we will not address that issue here.
-What is interesting in this case is to consider why the application would find this use case attractive.
-In the serial case, at first glance there seems little reason to use the initial image facility at all. It is easy enough to use standard C calls to duplicate a template file, rename it as desired, and then open it as an HDF5 file.
-However, this assumes that the template file will always be available and in the expected place. This is a questionable assumption for an application that will be widely distributed. Thus, we can at least make an argument for either keeping an image of the template file in the executable or for including code for writing the desired standard definitions to new HDF5 files.
-Assuming the image is relatively small, we can further make an argument for the image in place of the code, as, quite simply, the image should be easier to maintain and modify with an HDF5 file editor.
-However, there remains the question of why one should pass the image to the HDF5 Library instead of writing it directly with standard C calls and then using HDF5 to open it. Other than convenience and a slight reduction in code size, we are hard pressed to offer a reason.
-In contrast, the argument is stronger in the parallel case since group, datatype, and dataset creations are all expensive collective operations. The argument is also weaker: simply copying an existing template file and opening it should lose many of its disadvantages in the HPC context although we would imagine that it is always useful to reduce the number of files in a deployment.
-In closing, we would like to consider one last point. In the parallel case, we would expect template files to be quite large. Parallel HDF5 requires eager space allocation for chunked datasets. For similar reasons, we would expect template files in this context to contain long sequences of zeros with a scattering of metadata here and there. Such files would compress well, and the compressed images would be cheap to distribute across the available processes if necessary. Once distributed, each process could uncompress the image and write to file those sections containing actual data that lay within the section of the file assigned to the process. This approach might be significantly faster than a simple copy as it would allow sparse writes, and thus it might provide a compelling use case for template files. However, this approach would require extending our current API to allow compressed images. We would also have to add the H5Pget/set_image_decompression_callback() API calls. We see no problem in doing this. However, it is beyond the scope of the current effort, and thus we will not pursue the matter further unless there is interest in our doing so.
- -Potential Java function call signatures for the file image operation APIs are described in this section. These have not yet been implemented, and there are no immediate plans for implementation.
-Note that the H5LTopen_file_image() call is omitted. We have not supported high-level library calls in Java.
- -H5Pset_file_image
-int H5Pset_file_image(int fapl_id, const byte[] buf_ptr);
herr_t H5Pget_file_image(hid_t fapl_id, byte[] buf_ptr_ptr);
public static H5_file_image_op_t
-{
- H5_FILE_IMAGE_OP_PROPERTY_LIST_SET,
- H5_FILE_IMAGE_OP_PROPERTY_LIST_COPY,
- H5_FILE_IMAGE_OP_PROPERTY_LIST_GET,
- H5_FILE_IMAGE_OP_PROPERTY_LIST_CLOSE,
- H5_FILE_IMAGE_OP_FILE_OPEN,
- H5_FILE_IMAGE_OP_FILE_RESIZE,
- H5_FILE_IMAGE_OP_FILE_CLOSE
-}
-
-H5_file_image_malloc_cb
-public interface H5_file_image_malloc_cb extends Callbacks {
- buf[] callback(H5_file_image_op_t file_image_op, CBuserdata udata);
-}
-
-H5_file_image_memcpy_cb
-public interface H5_file_image_memcpy_cb extends Callbacks {
-buf[] callback(buf[] dest, const buf[] src, H5_file_image_op_t file_image_op, CBuserdata
-udata);
-}
-
-H5_file_image_realloc_cb
-public interface H5_file_image_realloc_cb extends Callbacks {
- buf[] callback(buf[] ptr, H5_file_image_op_t file_image_op, CBuserdata udata);
-}
-
-H5_file_image_free_cb
-public interface H5_file_image_free_cb extends Callbacks {
- void callback(buf[] ptr, H5_file_image_op_t file_image_op, CBuserdata udata);
-}
-
-H5_file_udata_copy_cb
-public interface H5_file_udata_copy_cb extends Callbacks {
- buf[] callback(CBuserdata udata);
-}
-
-H5_file_udata_free_cb
-public interface H5_file_udata_free_cb extends Callbacks {
- void callback(CBuserdata udata);
-}
-
-H5_file_image_callbacks_t
-public abstract class H5_file_image_callbacks_t
-{
- H5_file_image_malloc_cb image_malloc;
- H5_file_image_memcpy_cb image_memcpy;
- H5_file_image_realloc_cb image_realloc;
- H5_file_image_free_cb image_free;
- H5_file_udata_copy_cb udata_copy;
- H5_file_udata_free_cb udata_free;
- CBuserdata udata;
- public H5_file_image_callbacks_t(
- H5_file_image_malloc_cb image_malloc,
- H5_file_image_memcpy_cb image_memcpy,
- H5_file_image_realloc_cb image_realloc,
- H5_file_image_free_cb image_free,
- H5_file_udata_copy_cb udata_copy,
- H5_file_udata_free_cb udata_free,
- CBuserdata udata) {
- this.image_malloc = image_malloc;
- this.image_memcpy = image_memcpy;
- this.image_realloc = image_realloc;
- this.image_free = image_free;
- this.udata_copy = udata_copy;
- this.udata_free = udata_free;
- this.udata = udata;
- }
-}
-
-H5Pset_file_image_callbacks
-int H5Pset_file_image_callbacks(int fapl_id, H5_file_image_callbacks_t callbacks_ptr);
-
-H5Pget_file_image_callbacks
-int H5Pget_file_image_callbacks(int fapl_id, H5_file_image_callbacks_t[] callbacks_ptr);
-
-H5Fget_file_image
-long H5Fget_file_image(int file_id, byte[] buf_ptr);
-
-Potential Fortran function call signatures for the file image operation APIs are described in this section. These have not yet been implemented, and there are no immediate plans for implementation.
- -The Fortran low-level APIs make use of Fortran 2003’s ISO_C_BINDING module in order to achieve portable and standard conforming interoperability with the C APIs. The C pointer (C_PTR) and function pointer (C_FUN_PTR) types are returned from the intrinsic procedures C_LOC(X) and C_FUNLOC(X), respectively, defined in the ISO_C_BINDING module. The argument X is the data or function to which the C pointers point to and must have the TARGET attribute in the calling program. Note that the variable name lengths of the Fortran equivalent of the predefined C constants were shortened to less than 31 characters in order to be Fortran standard compliant.
- -The signature of H5Pset_file_image_f is defined as follows:
-SUBROUTINE H5Pset_file_image_f(fapl_id, buf_ptr, buf_len, hdferr)
-
-
The parameters of H5Pset_file_image are defined as follows:
- -INTEGER(hid_t), INTENT(IN):: fapl_id |
-Will contain the ID of the target file access property list. | -
TYPE(C_PTR), INTENT(IN):: buf_ptr |
-Will supply the C pointer to the initial file image or C_NULL_PTR if no initial file image is desired. | -
INTEGER(size_t), INTENT(IN):: buf_len |
-Will contain the size of the supplied buffer or 0 if no initial image is desired. | -
INTEGER, INTENT(OUT) :: hdferr |
-Will return the error status: 0 for success and -1 for failure. | -
The signature of H5Pget_file_image_f is defined as follows:
-SUBROUTINE H5Pget_file_image_f(fapl_id, buf_ptr, buf_len, hdferr)
-
-
The parameters of H5Pget_file_image_f are defined as follows:
-INTEGER(hid_t), INTENT(IN) :: fapl_id |
-Will contain the ID of the target file access property list | -
TYPE(C_PTR), INTENT(INOUT), VALUE :: buf_ptr |
-Will hold either a C_NULL_PTR or a scalar of type c_ptr. If buf_ptr is not C_NULL_PTR, on successful return, buf_ptr shall contain a C pointer to a copy of the initial image provided in the last call to H5Pset_file_image_f for the supplied fapl_id, or buf_ptr shall contain a C_NULL_PTR if there is no initial image set. The Fortran pointer can be obtained using the intrinsic C_F_POINTER. | -
INTEGER(size_t), INTENT(OUT) :: buf_len |
-Will contain the value of the buffer parameter for the initial image in the supplied fapl_id. The value will be 0 if no initial image is set. | -
INTEGER, INTENT(OUT) :: hdferr |
-Will return the error status: 0 for success and -1 for failure. | -
The signature of H5Pset_file_image_callbacks_f is defined as follows: - -
-INTEGER :: H5_IMAGE_OP_PROPERTY_LIST_SET_F=0,
- H5_IMAGE_OP_PROPERTY_LIST_COPY_F=1,
- H5_IMAGE_OP_PROPERTY_LIST_GET_F=2,
- H5_IMAGE_OP_PROPERTY_LIST_CLOSE_F=3,
- H5_IMAGE_OP_FILE_OPEN_F=4,
- H5_IMAGE_OP_FILE_RESIZE_F=5,
- H5_IMAGE_OP_FILE_CLOSE_F=6
-TYPE, BIND(C) :: H5_file_image_callbacks_t
- TYPE(C_FUN_PTR), VALUE :: image_malloc
- TYPE(C_FUN_PTR), VALUE :: image_memcpy
- TYPE(C_FUN_PTR), VALUE :: image_realloc
- TYPE(C_FUN_PTR), VALUE :: image_free
- TYPE(C_FUN_PTR), VALUE :: udata
- TYPE(C_FUN_PTR), VALUE :: udata_copy
- TYPE(C_FUN_PTR), VALUE :: udata_free
- TYPE(C_PTR), VALUE :: udata
-END TYPE H5_file_image_callbacks_t
-
-
-The semantics of the above values will be the same as those defined in the C enum. See Section 2.1.3 for more information.
-Fortran Callback APIs
-The Fortran callback APIs are shown below.
-FUNCTION op_func(size, file_image_op, udata,) RESULT(image_malloc) - -
INTEGER(size_t) :: size | -Will contain the size of the image buffer to allocate in bytes. | -
INTEGER :: file_image_op | -Will be set to one of the values of H5_IMAGE_OP_* indicating the operation being performed on the file image when this callback is invoked. | -
TYPE(C_PTR), VALUE :: udata | -Will be set to the value passed in for the udata parameter to H5Pset_file_image_callbacks_f. | -
TYPE(C_FUN_PTR), VALUE :: image_malloc | -Shall contain a pointer to a function with functionality identical to the standard C library memcpy() call. | -
FUNCTION op_func(dest, src, size, & file_image_op, udata) RESULT(image_memcpy) - -
TYPE(C_PTR), VALUE :: dest | -Will contain the address of the buffer into which to copy. | -
TYPE(C_PTR), VALUE :: src | -Will contain the address of the buffer from which to copy | -
INTEGER(size_t) :: size | -Will contain the number of bytes to copy. | -
INTEGER :: file_image_op | -Will be set to one of the values of H5_IMAGE_OP_* indicating the operation being performed on the file image when this callback is invoked. | -
TYPE(C_PTR), VALUE :: udata | -Will be set to the value passed in for the udata parameter to H5Pset_file_image_callbacks_f. | -
TYPE(C_FUN_PTR), VALUE :: image_memcpy | -Shall contain a pointer to a function with functionality identical to the standard C library memcpy() call. | -
FUNCTION op_func(ptr, size, & file_image_op, udata) RESULT(image_realloc) -
TYPE(C_PTR), VALUE :: ptr | -Will contain the pointer to the buffer being reallocated | -
INTEGER(size_t) :: size | -Will contain the desired size of the buffer after realloc in bytes. | -
INTEGER :: file_image_op | -Will be set to one of the values of H5_IMAGE_OP_* indicating the operation being performed on the file image when this callback is invoked. | -
TYPE(C_PTR), VALUE :: udata | -Will be set to the value passed in for the udata parameter to H5Pset_file_image_callbacks_f. | -
TYPE(C_FUN_PTR), VALUE :: image_realloc | -Shall contain a pointer to a unction functionality identical to the standard C library realloc() call. | -
FUNCTION op_func(ptr, file_image_op, udata) RESULT(image_free) -
TYPE(C_PTR), VALUE :: ptr | -Will contain the pointer to the buffer being released. | -
INTEGER :: file_image_op | -Will be set to one of the values of H5_IMAGE_OP_* indicating the operation being performed on the file image when this callback is invoked. | -
TYPE(C_PTR), VALUE :: udata | -Will be set to the value passed in for the udata parameter to H5Pset_file_image_callbacks_f. | -
TYPE(C_PTR), VALUE :: image_free | -Shall contain a pointer to a function with functionality identical to the standard C library free() call | -
FUNCTION op_func(udata) RESULT(udata_copy) -
TYPE(C_PTR), VALUE :: udata | -Will be set to the value passed in for the udata parameter to H5Pset_file_image_callbacks_f. | -
TYPE(C_FUN_PTR), VALUE :: udata_copy | -Shall contain a pointer to a function that will allocate a buffer of suitable size, copy the contents of the supplied udata into the new buffer, and return the address of the new buffer. The function will return C_NULL_PTR on failure. | -
FUNCTION op_func(udata) RESULT(udata_free) -
TYPE(C_PTR), VALUE :: udata | -Shall contain a pointer value, potentially to user-defined data, that will be passed to the image_malloc, image_memcpy, image_realloc, and image_free callbacks. | -
The signature of H5Pset_file_image_callbacks_f is defined as follows:
-SUBROUTINE H5Pset_file_image_callbacks_f(fapl_id, &callbacks_ptr, hdferr) -The parameters are defined as follows:
-INTEGER(hid_t), INTENT(IN) :: fapl_id |
-Will contain the ID of the target file access property list. | -
TYPE(H5_file_image_callbacks_t), INTENT(IN) :: callbacks_ptr |
-Will contain the callback derived type. callbacks_ptr shall contain a pointer to the Fortran function via the intrinsic functions C_LOC(X) and C_FUNLOC(X). | -
INTEGER, INTENT(OUT) :: hdferr |
-Will return the error status: 0 for success and -1 for failure. | -
The H5Pget_file_image_callbacks_f routine is designed to obtain the current file image callbacks from a file access property list.
-The signature is defined as follows
-SUBROUTINE H5Pget_file_image_callbacks_f(fapl_id, callbacks_ptr, hdferr) -The parameters are defined as follows:
-INTEGER(hid_t), INTENT(IN) :: fapl_id | -Will contain the ID of the target file access property list. | -
TYPE(H5_file_image_callbacks_t), INTENT(OUT) :: callbacks_ptr | -Will contain the callback derived type. Each member of the derived type shall have the same meaning as its C counterpart. See section 2.1.4 for more information. | -
INTEGER, INTENT(OUT) :: hdferr | -Will return the error status: 0 for success and -1 for failure. | -
Implementation of the H5Pget/set_file_image_callbacks_f() and H5Pget/set_file_image_f() APIs requires a pair of new virtual file driver feature flags:
-H5FD_FEAT_LET_IMAGE_F -H5FD_FEAT_LET_IMAGE_CALLBACK_F -See the “Virtual File Driver Feature Flags” section for more information.
- -The signature of H5Fget_file_image_f shall be defined as follows:
-SUBROUTINE H5Fget_file_image_f(file_id, buf_ptr, buf_len, hdferr, buf_size) -The parameters of H5Fget_file_image_f are defined as follows:
-INTEGER(hid_t), INTENT(IN) :: file_id | -Will contain the ID of the target file. | -
TYPE(C_PTR), INTENT(IN) :: buf_ptr | -Will contain a C pointer to the buffer into which the image of the HDF5 file is to be copied. If buf_ptr is C_NULL_PTR, no data will be copied. | -
INTEGER(size_t), INTENT(IN) :: buf_len | -Will contain the size in bytes of the supplied buffer. | -
INTEGER(ssizet_t), INTENT(OUT), OPTIONAL :: buf_size | -Will indicate the buffer size required to store the file image (in other words, the length of the file). If only the buf_size is needed, then buf_ptr should be also be set to C_NULL_PTR | -
INTEGER, INTENT(OUT) :: hdferr | -Returns the error status: 0 for success and -1 for failure. | -
SUBROUTINE H5LTopen_file_image_f(buf_ptr, buf_len, flags, file_id, hdferr) -The parameters of H5LTopen_file_image_f are defined as follows:
-TYPE(C_PTR), INTENT(IN), VALUE :: buf_ptr | -Will contain a pointer to the supplied initial image. A C_NULL_PTR value is invalid and will cause H5LTopen_file_image_f to fail. | -
INTEGER(size_t), INTENT(IN) :: buf_len | -Will contain the size of the supplied buffer. A value of 0 is invalid and will cause H5LTopen_file_image_f to fail. | -
INTEGER, INTENT(IN) :: flags | -Will contain a set of flags indicating whether the image is to be opened read/write, whether HDF5 is to take control of the buffer, and how long the application promises to maintain the buffer. Possible flags are as follows: H5LT_IMAGE_OPEN_RW_F, H5LT_IMAGE_DONT_COPY_F, and H5LT_IMAGE_DONT_RELEASE_F. The C equivalent flags are defined in the “H5LTopen_file_image” section. | -
INTEGER(hid_t), INTENT(IN) :: file_id | -Will be a file ID on success. | -
INTEGER, INTENT(OUT) :: hdferr | -Returns the error status: 0 for success and -1 for failure. | -
raaCOykMsKzk|x&e)|^#FOj@^{(4%Y!mCSN%?#!? z9yc{%76$=v)k4FX!J@MpqG$Q5MuS2`FdHREk*BITZyzvE*N$(XO+dqO{#BmG-7-z` zYa>hd@#2q5D$A`FtfWq#2~PXgg=%mdS5%1T@2asi6cpg&20r=9Bkeh@cS!RLsdEdEgYwIQT_n;aIB`~ zBG5D$yrat@Xarld?bA_*qp@L#I(Ri2TWUr^hQ1hdnc1e)v8CxpojpTK=ycCNqfpI) z?T5#Bn)ZRI->5q|saaYmJN=nSdAfm5!-KE1bq34@OzygvORTKyf5>xQH!_O@Pqj^0 zH9{${$Mej`>)-1A_i9ay3k@-4HrM>~${(0|Kg-Hut-@~FcbAJe&R* }KI)^?okk3Xi%*|)Eh7yoh;pmxO*=0x$ z)tk+JV0F0kohj%BX*dvT!x(J $ss9KfsvsmWRTN`l ^%b`_w|^xE(t;d%Vwc zf^TMi`$R1cFT);4YK`tP;1SG5WjNnitu9^}&X0Flx9Zq&`t|wTsDp)t1@P`!WA+Ye zvhgyf{Y{gHl{Vp)NdQ5}amnRfapi#DhM0VVu?!#+sNs7x;*9FqncoB1!+Dy4QBj2M zkvRoi>(ug^V@-b2ybnL##f<1_Wjib}>TG_1r4Q@Ug!&Q?!qrtC@^`X1DPpdm*>v{x zwF?k}J~%|De&0n!M&=xV!dxK7K>fES-KXJKq|!Hdn#4&ve;|r)O(f|ahSAPBmYzsZ z@2Zkos>zY=bgBLH-tn4!3;wI%{6tNnOzJVFAai*E)gf0o@p}UTgs|0~Tjc*IBp8iO zo}9?yK~Z&P$wy)bjeNG}75@z9%Yi;o#uM*8 Q2OSj z9S{*{cf8Rpy7yz~7n)92nmqmv+y@DQ7#MiV%I@s0m)~pz$S`u-xK6A~L=lT~o8ekd zdr~?ZeU7SsXZ;(t*9%RgS}eEQRibA&yKhJ9djT{FrwtdN+;B7v&xKTnCNTg8&FXl$ zf&$F-UnM5io@dsl@B0KI!c5QGTX5qMAWM`x%rpn38zi;?POVu=d#+dh1i#3j>eA JB71P5U$Q;X%by)im#I#RANSkSq=~)4(UVi@{IUy2zr+RZVWc&)^eet`B;p z`%x|$`xh^K-mpzLsI;ZI($?e(W6xMixh R#{VTBLt=^QAhuyN!8tK;5iJhX}w@%o!rz0)V&7DaO_$b=g9 zP?^JwtmE3J(FhPoBkyJy(n)w|e6$)-0wHRsAs477S&Z~)HHchc9DIDFJ4^jkFr`RC zOwiK? }n|}66xbo4_Hz`mPv4 L^?D#nYjddwDk-Xet>BXt-S%HaD3TG6TZ_NYeoJ=eP?$XM`}0 z-@I9{HP~wLwJTpvO@|k!c}Am$b;B{9HMPJ$+97OOn7T>%v*NHaTgr~RlPvfe1qFkJ z`nT>y({;iXeT6HczHLPgXc+XXU%OuT!LlELV0AF~rQDhJjz?q=xZqe2-q^Sgy`^AI zhMVgXUKe}iIM+V#PMNA|=9Og Ux7Y{`vWrps(kLtWrip4 zXaIvb9j#S8C^41<3K(xH?)s luR zPi6-xtE&+>AmpsHLos%|g=3Y=HoHSeWajr|-JOm49pbr;7 il;YM>0-m8_lENj`-EKo`v{$>U@ zW)h8I-!MJ;w?Ar`Ypo4%1w7tDOu@_T)NL=x>H_Lvy(PpgF zUoMomro^? r% zmCseDefErquXE)*YRWrmf|bhoAeV)D{yx2D5D@=MB=jmk-u3I;ct7ZtxZ$>fzT*j$ zre?Lgn_9|SMAy&}7ex51jEv8>PXS@gL6cEQdi{O8`Rfa^@ZYEV{+#>*>O6nb@lDu? z*2Og;P+aApZcq80`MC3fl@`SyordUj@UgIO+!y;&;e!k&K7nfzj&y0UQ5~ZHfLPoO zH3WFvX+=Ui>@si#1-g%*cWif7G~k8jSY^E|dlhsx3*ZNh8%RoqQd;#SRK-k4? W>7ZZx50o4g2KDybLiqN#Xuspe* |`&LnadOrVs+Tj<8TTgqQ zfN+Z15t#Je>BTqU?f}a_p$)VP1a3ejU-vs4O80;R?CUyWj7za*nO=K^)(Y?HjEjB| z#&4f$gU2`#Boouo0w~4u=SNOaQPHxlm_+enz@5mU!m$-s&l2o?+VbH@61G+H@)AR^ z8Rmly1oDEMzRTi?xvhTfjAzf+DWY}#GyFv$Du^pXDm@+gA|C1>S=bgdv2}^)jy!NI zg>1%R8=mhCmYLm0bp5}}Jwvsw)?gn9sltaxfL5_=ZP9>$!H($bLUt2xYF$^_3k rxjs@u z3*E1G4wlk){(LC8>H0@o>R>)T$W#fmy@eJMX+#=Ax^Qpmc?|NDfXbYUd%eeM->^#w zV@G~S2pL`fR)5Ut;PUeGTL9ukL G+=SF_NbYn zs;SwEektc8JC2Cve{15%LbUz)F%w)S1l0l(Y^U{n3kVB5q>fjd52(f6c;Jf6_tq!Z zEp9Z10%06ZS(7CL0N-9|TNM@ym_&eZYc`UfKwcADsqo5nd=^NAtS0Kk{$jFy5d5wG zWqK6M9#X`(EiKhhNfpB&K%9v*9Ir(H;8K=BmEA1E*?uV4SyzwtHmIN!DNCA8Ye5BS z7E;|R@jFpAe7^R{6b>K0tIX%f4Hi2|ndi<^_n!IA__k`txz;a%U@K|H=~X*qR{tsZ zfCjS}oNWMV$dKQfdcF}JoQ;2Nw>QS1sz=ojxOF6O+u}K%1buD?s@&g O|~Ot z@H;LO?+?Je(rQ~Z$>z2(#T)P?<7;;}neBKb1y}-3k>N;HgWWA?Hz8s|-TuSZng#28 z9?=_P9=Hxx+9IeDpd)-ErBu2)A+^BWAlb}{hSoEYXpgY91lKk>xI7Zczp-xwOak6W zc$QZmt=Ab1P~crej-#-!hXMr)Al?IT^~9(cg~ZPGHX6RT6^0HP#!+jU ?-@Pbm##7jXk+63o@yXb#F>+ux2oTDph&alEm@YfNv=j+$ zOc?YEYauIuJyzw!u+x*HNErQE2v&p*b`Mevx*Ah}0MDaf$x4sgN1~njZbYILLA>ir zQp-v=*SAv-e!@%eWL!8uKi`fxEZ$GRaTcBC^@kIp0d0WLW)?|6g1j@a^}{0FaUnr> zd;08T+R*Q0cirv2MYjdq4?HYJ)g+-zkXQ=~d8$_OAL!0D9<3v=OXcm&V8{s6yevt% zrw4 2b<_wEkN~^X_$Xz$N($Y)b?e~YxV`1~LYfMT{>AccK4&tH)K75Wh_zVU zbn3OevomQ-6_HhafCBV+n+K%+9`ImrCN4^uZtcLyIjxQfO6~s&%z5ejp+M;d%E91~ z`1zHA`wZZ@W0RA+efB0Vml S@gMZd;YA1Bmh#DhR(gW zz0Xee41Et40BNun55D)eq*X~4e%UX5mIBuv<9l*wGG1x01c6i#@~ftO#a|n_(SW@w zvD0%4MTU!$sS*2FKmpjGKAcL7 C_+6>1`MZX5qPqG;qGBRuhZLo$k5FrY70e5Y>u zgs<2%yN^QG@XK6`wAyH~(OVHmrbj-z6kyOee-O5tO6h > zgMenLKm7B-Wpymhz^*138e#~70-#hu&gZ>+Bqf62N*WI%W@k;MPWR2)qpvH&-V{`g z2SD|(%B~ma!97?kz|J~^R1u_)u##L$gq1*sVDncX$snlz_hRUKA@Hy~-6Rc)ETjhp zKUzK@zAn~U@454}uWu2=#b2-$vjBABAOISc9&m8kXa?y%!00Wyk9^wlG_$|<_R8bQ z09IKWuUeW3Wmvq^bU;MLuJ!~V!Nv3EYF(CPON<+^+Z8wBwD0}rs|j^EnNG;!Ngb`; zngVEe^X5$edLCMo;bIWXfG?9Db#+^xP=p6eBlHXfRI62p*TsmPl{NSh(I1Fxzqx5n zz k@`mb`bysbiKVboD3r4B%_ATL~shlqs$OPfGD`r{8vp%-iPyEY$;w8vUpBT z<(*w!uq+yX!7m_0B$BW8t8MPP6*1s2xgfqH`wNJ21WcWslfCI7Aph_&u9efYh@l9c ztD^b>=}2qTduA?dnb4>B>|gWG4ZW7%A^0A)b4_PhUp1t6Ds0DvktGeVbN0J&nDloj zTHs?~XO=q7-Gh0aocG7@3~>?Q`XTW31J*Mr5k8MnBtmanfB#ltq274BtsG4!irw^* z*A(&I7l8GJa4TpEO&y)ZM!&u{un$0Rfn1t5fHj!#W7Ph(24j_NRFAx6Ky8}TnJ(`O zL}L}UPcW|(xUPbg7^0~5u5-UDurnx&qfAy~z*N5_l;pk7(LbJtAHx9Cj4C4+4n$H8 z0Aban+qIXpw`Uh#KUxHWk5ZO00=@c??*Z%UN4}z s1Fm025Hr*%*7r-tlnF`8K zZFsaS-qXI6HfXPYXk=ADaGX3mPrzs_n`9X{j((r!6#<=PSHE?y3j1!y>n$kHXizJ` zJAVCM81%UWaapDx6|1Q$h^T;^EJ*%-rSKX- %O$Kzdm`K)3fyOc;@B)vtK^cVjvoKpVjM4glPo z92}1U1%ZzbqysoVSeKd*yQAbc?g2uwKiYKKmUP?t*+3R9a;M_r$;b}?N=~+KRMgj} z0X5GY1~~GW%KQzlkUwA-?f^9qw4aiIc-{Wd^z`g-{!kR{?TC~VI)vzh^Z@>_;z}@9 zp+MHoFp))at6^p~nWxSDhswRIb}eO*vvIq+m&hv=RwG<6Z=fTh=p}y^&B3cFWNa#r zn}hIq469?|xeGr2j03EEIoc8~5RPUZil=pHXKTanEEygNHZ^sPb{U=^Ftc=@e>Bs+ zN4#xqZ4!GE&g+6W>rH3A#(pQRiTsb$K+Y|C_zA%;P94LA`td;Mk_64#1>h!txFT96 z3>No}W0)&4LqdR}<-vhOOmiUV$eMPU`}Oh4YXufCW#U*SRJ|IIRFh2LB}31J1Vfhr zc|~i1AQZ@(ccwNbJjHuUAm9f^tAVTu8%}zE2h6|D?(S@GzyV<}6^lRBj$N3a{{{>S z>Zvo9Q7%uXC~0NJA<7K~Y-eYu2{3Kg(w~X0fq9F9bh d-}j{#gDR^8m}*$?O$ z;Ys?9-lAX`aRR<|-l3%1&;lq~Jr1pO06k7Pg^=V9#XIAE6PadJPEZMjYG5Dts {0_Pwf!1^ZGM#Qj W`BJNoQ3_V_=7ONV2Tm~dQ{LhsG;{w+&1GN _cT7Zwo{)Eghf zTHD}u+kmO?(EA-bnox(p4 6i1fV>BIh^=8)nI}Av%fO{Fya8p57NMgI4#qWw zQ75aJ)xe7_Q%pi@Q6Qc&eaDM7Y0%_Za1ck{Wc=R`<5|Cyu2~b%D6%a2(`EteMBmQ4 zX%u)2cr+ZYEDXkXIN(l42si>XgJ^KEiQk|6_pPvvTgUC2ctZ$jW`%1Af@Rkop+z|X zvoM05z TdLDU+kvl;&j*leepq0&gw>*kZMu{lIWitu^`$f7yyjeUXej@XdUe=Zb@t*$Bt z18Hq`NWV~D0u@@y?`V8l?~@e;=!!WxIj!yT&oL6mn=Rh^Goc9BrW0|vVe%TH>xhPo zXm>yi;f$p~(lc{qDb&N}bgflQ_d!$q0FejIgB3iAdyj^Mpv%@NvkpPV=x;hZ*$6rh z5@OktPA%X97yi?WN$BA&EFd8 `ShRDh+wExhd>8ls|Rfr+4i_sDB;zyOpP#P zMpZSPyadz`2h>T>c~ND(@yL#tUIMSodLQ%}4K1x^pdlS_6Ep&kaS)+~sMdCz@S1Sc zN^xB@Y=^CHawMAC+O43IyYERt_g{9OaJ-Btq@a8uDHMcGfrwDOgIp!xo6Rw^m;r!R z$ekiI37%iw!aowf{R>32LcbCX>}I|j*uR4jgsARisr0m&8gGCzZtu^ NHs?Kz8w8wKe=p300-J!~i(-w@kY$v7VXiC;OYjDfcwUN>PZFql zNO~EpO-0zfWwyzXfkCqJ2n~RK^#qN1%R>VB1R=}8i%rw&z-j~t13~e006AO0v=#3X z5iR=c^aN3N;C2w)3f;T~pastVErge0%<_Zh2nl%zyedP?7_r#^cEEbwgx$V1K7E=9 zS|*s)aKjmZJCS<*`+Fb|h*m7nJ~-=qgDO6V;SCoVyar&2sN;OKznRt>_Ker-chrh& z|8RlT2fJ(r5^RXM=(6^6WNno*jzt^-6)el)cP?!OXQmwDcGfmc=fC^qfrkUzCd`By z0s-*i1&xG VuYVF1qIhGo_k3WZuDBBd70O)73hzE4xO&t%`Z>_pA-2E0Yb!A1L=_l z8U`S&tF;bdq`)G8q9;XII|ywZ88U(2Tqlcmenq)90vtxNF%y!<(f*~j$skE?!X_Pp z1yRvZUlD<7wvAn&{s%C k-nV^(-4Ewog(}I; z$q52P&!$aGbn!eQwzj{#R$)Ia#nbJ?twQ?xGP5;=DkzXBTWc+A`?XML@SAR{dQe%3 zAMdr`&$YfzGY0L^plY#>?|7r}C>rQ!t;zN7au2ZSL|p%nBa%CS=O-ft7$AmUt^@rz zdMo8429pPDy%5M&PI;~H!n+-Rrhb$d?1>_E4h@gAA#5}O5b&yctK#PnfQ_7``D>$g z!0DY46x 5<% s~9XrFkmIoie za;%gQt{WtGc5p5LC3hSj>;OVDo2Yg~lwde@XUGYFIs)tk%vl6MT3T` `)veyFwE58t(ERJyuZZIY?SKVRfnZs9{TcBo7`!*n zfhs=>GX7)GTzT|LUjzim*1D}J!JU}{qdLzat7NH&B|GE(^<=mS82&NPj2zG*cnm6e zmPy%yQFKz?V?^T!`3JSj86hJ;W$$KKibb3Uh-WsK(}XOFL_VV~kjIs+tO^XWAxIB< zOU~C JU;i$Rk5ul7OO- z_%~)IWc|H|$l_U2mqmKf0@&SIFoQrkNyvZjzIJUS^;7+J7dJTt#W@Wac5uh1K@*oS z7z8p_!@oe@$l%r~%qT65g=M}4lRsr-Po7CsSD|s7GdzcMOFm0ws1*i=Xgt{?bzz%v zw@48PpXI8hD#3dx&)^aYV-shht%vgjA#rydW& $3~i-~jOkF@Qh>Ec4hh zg&W`4GaDE7zZ(nU%?>EbH!b?;kU f*jthL4F-FI1$j$5%8-Bj^|MZqg>Y;@+Tl5gbfaAA*B`uNy=`cZXic!uyy2R zWc)#W&&tbFAfyw~fCtf%Y5^?U6qpFYGC+n0&cgvk!b(6u0J(&$`g+5O|4$UJ&auI= z#c2maf(}d1o=xx2KtvzIp8woHM7uA{3()9~Mj!E#hJUn3t8?oXKlu`Xn^*8B(QXP_ z0BI=%M*thn0V)x7U7XU?Boomtkr~p>-$XZFX$f*9{I35IWX3!ocY|OHxX40-s%TL8 z;?vV(axLFH{rGbZVP&9azu7zc }K$vZL?UMSq xy_N?P@H8p z(#S!6c7;NMLrfyq4d=zs@~dT+rs-)D3j>tYw0qs8^fw8Q(}N|*qj>`b1%FSQerIgW zO-MBxn$%{xF?A{3FAcI?HTua(SJmFfpSy5r6M>#mK8G3x2Fjq$72XEN{>>X|7%E-x za^7dyULFAS9YyHC5zvox8n>;|G&b%7YZ$Hy!Z}@sc*(sXy-%~jjRY 3uSJq(_-TY4{j!l|6d6DQ-}X8L7m$D~Y;15~QP z9L3eiy4QfIy3T92;9bSr2RF>#18B| )}JLwq$s^<>gq!XMioU{X09Ri-vl!y~I)&Izq vMh%6(6~q<~Rvx(faI^GC1AR}us!c#RI&WiymxwZ|4y5QfV~ zilO$@$30v&Q-ev902i$?Bs4TA?zW1=D^espvC|+dUB78$8-d1Wgr|{HUrz@iI~^Ti z0Dqu-i-_u30b~FM2glt52j2-|v#@X?44ugReGgh%S`nW^p`xN9$Sb`C_~;9)4+tQ< z?$^5JRso7F= A7mdpCY zTd?2C0U$yP>6On@lvh;1m!gM+AbJ!Q8JS&N91TNstKaW5J|yH4>@;dn$tfd)f#&>j zs{?|!bb7S$zP>)?&mXI# O9WU{5ekC9LnyrRg@ zU(&* UA}Z6{{bBUnuV1T^$G}a9sXPE4#FpM5KG-o zb6fdN1a|}(X%a97)Bv@^n+#anBpjc^zwkoAHaG&adq?nY9zr*_&)OH9;!;w24tAE1 z;)Ydcf&4dA)+h82Q{H&6 Iyf{QcsW=$Gn$v2pxVDHhtyd9yC2l`gk9CJH{Kbh zc*r%ivQl_ QCiu&gKC#%&F2UGPgrk1mKCmr&JBMvqv%tEr)Bnr7JbBx=d{ z^oiniH*1|elZ876IRP9yxh9k5(*a`+a%{rCr{|3CH=CCIauEZl0rM$WJzdSlhG%4C z1l)BcN>rNI?rlK&^dTll)xlxq7Z$z>e24RIeNx%SM-q1S9XREzva&dkZL%$s3K_>> z{{YNm1(|rN;rW9{kJRPl FLO4@G-#p*BGpR%$%_!_B6irUDmj zT%voJX+=*S;ikQr@p}{fM* | zl>&(?^E>lJB0@kqcXxM(hldeyKmg`{2=VnU==A)O5|G~{KYlvME1v2qHd=GO1s(VX zRt`x{IRyc(OZ|PW;8`)Ibd_GA>U^@teU-oSX75r)V)|ThIJmfGKaCojL5G9%3_?0s zaj)OfCifx&>04o2TDQ7X2w2_%K>&mqbG9Y$`QJe}8GH`NOCS=WUW3 *REaG*BuLN7QA)n22?+@{6V;mEZQlGOmc $C^D>U0c3axi2xkn;NXZYh<$q9zk9*7FGUo2 z7GOr)0$ES=)8NF5=YN08Xlh=CaLq5!Fj)l!uNsY-f5>f_c?_@&G>`|nx*fOrC73M? z-z1U3dPnvFm`AftHNfMr{=q*U(o#`%%(g{>7OhKt>nR~XE%o6XwdCG=O};1KRX$xF zd XH=1JEpPJ}xn**>kco2bnp7+2t3Ks&*q4Zc%` zzxeXBf-&gxN<%yN>fJ!Qx x zerpv^5X{t5L|>e#aZC-7AlcE~JVyPv!1A9zf6iM2v<9X6vQV%r<5dV}OmD 3+fcS8ckX%u&gP2HbM@J;cDcv&q>QV H}2@Q-18qP4WJ5b^MB1Z)yfP{3956r^0_KnGug!L0<% zGcY*FwD#o5ld!HXC5Wd4Pj66xh(wBPnMR>IcOn7vfyY6^7=-%rSOfVFC6G}@%J-y* z?&$cbQbaHiaqYlVdNcAj?&b)>QNcEY3=MH}2MvzYqigWLp!A+zcGnLh5tDWu0#Tc^ z=_&)sZ?D`cDp`^7ZWMMgdcEv=OVW=1>i$V1hj>t=C@ XvgB)Q8ym&}fh5NkA~kd~37n9lq2N1R{HjQP&z^?aV^d0(FA z7kYx{ydP5|qh*_A)>Tj(Fql0Df%;3QJyb*xDz*~AZq$y;U=?rz vNrr_fwSHjx?M;;D{m;=F6 ziJh;2r6derd;1%NJX$CW8oet>??U*Jq5q-NZ )itj@T3N z8OMJsvcFSv-qUb6vLa6!7%mO8|05 R z$sbtpGJHI_ccMwe@a6aM`X=q3HqY zi d_poU}j2W4)vngY8p`k=8m%~GUzv3(RE7GuUJ@ ZgRfFLwxnIhB8qOA%?~}g(aFnLs`9XG;HI`QO{iOHF zjY3WSif4}lR_w_)bGhgCepAK?TusHB&O4S{JlJxg#~EM!Le!9Ock`u}g#Z-`;ChIk z7CQ_GeV(6BQ8ZM0k$*&H6==A(C_rePKDL<_qqA16mp8R}^6ihfYbHoX#M_5nnxSE= z_K8a$Edg^0zTh$}c@ZOxD;@OdpF?;@f`Wp4!TVx+g)6!}WvPQ}NILQM5#}Ce2vzLk zaZB+M-|G>I?BC$)c_0=c3K-X#{$%US4RVxINm5SPwpF<4PInYt8k#>(;~Mh~>(;JY zTDnD{1*&u?Wiu-N`tyo;Z *!Hj zPeW7FWaxMl8Y@mL9&(-E&plWWu!?OE(tSlyp+lq5X`Q@9hIwx5QcUdO>fK^WSP^cz ztD_-DC-r+LXy1Tqi#uktIi*@I_hw`JsW#dj!y_ykU;zLG%xZ}zddu`~WUl_bU0s^( zbiEUtf)>5EulkHtSXfwNb2CXP#C@|p(zv3?f+Y#Mk(EUf5y+d{H8nR8IK+kwrA`LW zr3zyr#Au0b)a~%$J-Iv2p*9;(shm4^j37`~GN=4eqHI6bEhk(O!ZGdob#~DNxxsTR z((K37<^rTL#I34u`aJ_<4`Bq^v0Er~fCqr^vUz#mBsA;_45R>iU1B`RD=bWEO8(tE zSZIX(1a)vzp3prrNgWv(8Gl_qnA<7?s9G?*i<`)5OZn445S2sN>{B1S NP zLZSL8SnTxgw6y14U2og2n-XGmHPzx7_8jQy14hDGF$=Wx7o!tmI&1+t#J1shZSBH^ zocMTIwB^cm=sIqTZ_yxe9>9}*g@#&CQ1FzSgIs~ps#iEnbdNJp76QTJPcr@>x#49Y zIQ{;8KirbW#zw}cNzX~94dygb=M{Z1@!?$_!3IbsR!|Vf?(Xhe9z=-h<%stYqC5%M zY->)fbgtNSi$F<9DcdggCUQOOoHVrCNKRKx7O~_BHS#+kHP6pA;3k5hI}$*T&KLg= zAQ-1=uzXvCx!C9hsznLh+tS6icC*L~Zy44%ctq`*qE2%lnRP$U@8%|s{9Ja%SglMO zec3=>-zN_bkBtx@C^CMfUhJkgAicc%_xFR5r!t1^{rG+H?W1d$mGmb+!mHyA&iQY+ z)>qkEuT>T?E7|$Mz&7&s& UVvbt$nETbDdmA+pKgI*eI#8I9Rpdw1dx>N za!+#CqxhJpr#*GNzx2U>2Bzw#mHMS#1DpRBHMz6iViCq_yM`! za53iVBWZ<+nw`Sov0Z 5itZkJsKKTTjbe_C}2FVA8g_6JR~Wo zF6A#-Z=mx2^;CIxYEI78uD5A)jr<&+rS~5AG@1eQ;7 lN)PpX0W>>B5{Hey06Ac_+oBuJ!g7rvO(wEm}>pi$&3k_ zPe=L@H`z&+&Dc^>Xy)8$Ta40pj=yJEYnl6=-<0HTynVX{w=F>(<^t})*RqA59WhR- zrqD#=f-=INjMc2z3c9F*
!iL}Ug5D>%=k|KgCZW$--?!WO(65rpe+V6K67NI3U(ohCWA5mRk1ygVW(NG_v zi1Vqbw{yr!@{wu6NyA5*QF=o`Kgbw~Nz>646n_Y4#0jtuiJgt>wA#5k&+AU!#~f9y zNKd)H>f%bI>}RU$R_m5!Rh_juA8}4{?mJ?V1PYMGN|2+6iTwMB(}B3X<%9U|fByTs zRtYNA|8qwh;vdPx|93z7FhhWo62<>K4Sdv%5@)LXpT|#_|Gz(eRn})O0}cj}{F(1z zBnSQYxZG#Yv)QuE0deRPy)6UIf7Znx#L$h|2Cd&yS$`5mQ^Y(Cw4;hW@oJqHyHCUM zt^Ch)uaePWx&z%)&!4%+3J=EG1}2MjIln~Yg)$<#xap9(^C||uL?pc60AoZOsi=z1 z-}1L*o pHJczRX?Td@!58^IX*#j6DOi_~fVSyv3U~q6O!%vs}?^VY| zlg>KO2~K0S5P~zTJ&YoX2{&0-qG*Ds1=%FAC)Ne=G>eey_sYHVg?MwUvhl DNz#fi27wvpTP(qt)pLu@3nKl{53w2eH-84|@HjctgO!KDFC+i61U26F zzNA7Ro=LL!Vd6l6P_U}^ZU3~b>GDGhIMaRp0d_8|8zyY0L6EC7j(?a|efka49v>1u zJnlvQtpLUVTv+Z_R#sI0otRP9>cgPpZ!}C&N|2C~&U5^LD(T|fA;S@f p^0`lRI6>l@>`1xSD3LG$@>wxZ(Nn* z5}T$VYR>T!Oo;NlGu>!fjgBD*3$eHB-urLGx85XrE$ZY^p4S@(cAKS@uwP;oI&Dd^ z>Fj91Btb%1m0BMbn(fJeKVaTsdAgMYLZpiX)QM8&X~9*;SLYi4yucSLn(K)A>XFAu z$&EG<-f?d%SF^Inx$Jsx+~{g&7&AijZqbZrWS7|hj4q1SS`~vJbaD_1PXdA>l9kpy zKZ Y=qS(Kf|2z?jXxHx+QzdFm*6^~ zg{(w|f*@+3d0JyYj?$;pGup+>AyXFP`ftG@;yD &V#8_RYFIo@Zfa4=yV!jbBuaWs!~l_d})1 z#c`v%3%>8Kz(_{c6@b;AwBN7Eq%sFlhkw^eKAb6L@w~Gxn>#AUj}WQzoU&dzL|S8& zT{?B~h)?z%{TDbMrQMH8cO#?b!nVOPzrciqdyo6sW=>8CGL*tC+NS;CSpU|iQmd)X za*iet^)T -h72sKi=7j iQSX>$vqo8?l92^Y@=^~ksB(}hbKh1O6~M@Eo6EaWw2ISi`&z&|L7 N}9`As}2@-_wou;Dk; z{U@oaukHcK|J>?$oOaIR_?8PWKBIpN3|O4!SxX32mK*6DE+og*5E )%e<(8wHq*OUC&|1O6X9zmw6 z=hTk#*vU`2Gv{AoPs&)+k;40xJfM@&-~O4&BGv&Mm2T-uI`uS_cZYg~Mz_4r6~-B` zce4ht2Zx6wh{DJ`7Xl2;8&Gn*w<=FhPc8@fTx*y3=Q$mJNKVV(U4xh@;u!u&u0|bl zDRBnzm9WUg7gbmTSTrpF<^$vYB6(u{?EGBALtH0um>t)UTy@Se4pg) z+^K3iGyMJ?b 2ruCAw8Zw!x1Ggy2Xez_mLR_V7 zOxx;31J@&=;}1)6e*XTD9gpL74#$;EH9oHq5IYW>(cEVoa$JYx6CB%!QibS2J$EJM zn4;X^_)FQmHg8#z@l27pi;<3!VP{Frbozhi`|xo$Z2&k@dJaoy!<{fl2$BRQdOL?2 zKB=L-v)yKy=VnS++xPW$#q+$qxhlc}mrnCnN_kalIE8eQOlSt>mYlDL54A{So`jhy zGumOn* OgGd*h|Uhe7T^=@VXk7NoS9e zDii_-2O3CF#>cj6P*jIIDY+W8qW!s4oXO24$?`noWaxM>)b%)|;F&Z!nl06K+YI`> z&wFkV!+R%je|jhA&DM)5s%XjDdhTj Qd(@}-MqD}^>DqVwWvV^Zla>f_zq>~KB@&g^eKhO2u@wko2^`}a;rZ3 zl`fu2Ud}Uvvb6OnTf61YuRi@q-*v`F^-6yrS>%o@!Ab+g(p@--QoB%|ay*_7lEC2k z*|7R6LZAtpY(J7H()qEwkv%X4X1;0}vEj0k1iL_A_+$TusetRrsW-qBYEO#nCguo3 zZ$?0MLJmagQiBj%>OHh|L$=SCgK;B$H151boPxj&j2efSUD?TBY0ShA1^2*`LYNUS zOu-!aa4*(eQ~WU9zrT2?J+svl*^+%WM$FPf!7BItW zx$5hD(HOP#$QDim_Rw)mKNuHDQBnN~5E`utrODr$&{k>9WdwLBd#JklW4=P?a|mdG ztRZ|^IPk#HxnhGUYzBUima~4z-8#~|P#;1KyDmv01%LU2KP_NGM3QV7WXSFR;0Pv$ zfQnJWC#Muagf =(1uKx?*RT0-+H&YM^}Jjv)p? zE m1O)aH?4VGYPrOS;xK#;DO7jAHF=y5~?y6L~?8|K~- zR;^oXFARRybFB%QWK<1DPh!c;h|J%z_A+Eo788*E^U|}nb044Eay|Qt?hm!Lt6U82 zH%i!FbjR4{wS~*)#O515F1l~;x34l_>BLfz81hU??{}@X)a^ut048j}Dw{u7_ZbuZ zupeiTY0)}~BU<^x6%eKL?@qpDxGY&%uQbOl)|)x5I?rET^rARE?k2~No0iU5Y9xsQ z(eH~JP&!;sIeQD?70;6 ZBoO#h6 z tNwgM i)Xyk1|5^_9;Xedx?HPgJMcE$Sq-H-rC&WeDrb1P2bqaNhP%hVG*ccO@Ff# zhKA=FCnklt@_-Cs7hL%E0l`~FR{VI@AlN{HxJyM~t~J6R3JF05K_oEL8k6*jQpAS; zk*Nn&B*hD^>wuE_hMd;DS~=Qy-acyET }m-x8$B>UhOpc$2jc*?-Ou}^pMjVI2;;reQt6h&FwgM z?1uexG2xmu9*l%{cznE_zWxIDFTn7gyQ9n$%ZJFt%vv@`VTyuCjHQA^>f#T>Q*gxM zo{Yh8(KBm@u`2zmVnw_VC5Urys3|gRs!N3WHc;mlaGF2IQwXU3<|RZJoKR#)QsB>< z#!2MiHP_@q7 JEIRRvz=@J#u7bkCU5^w^ z5Lxq$5kE|@!lGCsI)p4~7eyCdBvU;O^N#s5q(c7?*6@yZV3rIyOZX4159SI)a-^u? zCI&~S@S(Hp!BwaT3fq3v$QQ?KPUTW Z%}#~=uEiIw35RG%QZjdJVHye&5; N!2^WO=OZg{-Ac8JN@;}#osVT~aJcxs0X515J| zL*X~j!nl60n VFr#t2KRmRO#23ZN@g`Y5W%v7h1it X+Mv=gGk)Y$~)9(xA)Cn9mg9iE}Jh|MuP`4)unpNfKZuIp2jQvLJfOg$QU zZmh%1vh~KPiM17G$pdVsq84ksI3rLvAnM~u)v9iCsh=55u7t9*pUIh+CReW=pQ2Jv z*o@@Mg7~f;@x$} x`+h&LEIv?1%5#tBr-kTs9*7UKBeP0Adcp~)9`?IZ;Z0u9tuDfQpBW` zUkl;qX~f2&*7PG1z@O1$wukNO1B2 U%FcDMaWLCPt8m_7_Bfc>(1Ha%ieMjQogn7;Mte-=I7=PK Qt(V2_Gl&_uVbj6u-7FUh~NDJz^4i-zc87-|GUk#lr)z z=fB;rAI9x`gz*OxMt50Tm)es}nQ?%F1b{fIo$JlX?RbEEG(x$2?vH9=Z7owEs@L0f z_WqPGipgrL^Y3kf^9>+w?v&(53xNa~lF2|`TO5?hqlM2dR$ERw$vyPw!*Y}$sHIbO zpq?DvHE)Wiw){$FSiD)4qu(^e!aMsB2QkiymllFus5!WVFR`cKP=5#10JSr~7%bjR zEBv!iJ|x+KSt8zXk6?oHnP6N1IhaL3RZC`~ }CH7YkxN;uiB9n= BdPX$I<%p6qPo{<9)%qgd+*76*EXyi&R zVrG$Aq*j#X9p;T7PHkUI*bkw@xWm@0p_su!*_8U`Q2i)znygM`xWo4h_ZyG&q{KId zQgeTRGzERwR%rMaBg7RhCTpGPN3o`>_-x(@1Mzq0#4^EA!h$1|3q$&tVe;QgxIWLL z=YNj=c08MeK0$yj1f#Kj-)toE)OrvgE1U1_q5o=(t^J_DZGqU|V@#ZEZ0=;vz7$vn zUmGRYCE-nr24AGwb*rO7KmC4?x&qx*1J0F0$|G+RQs_Nb*`K*lxc(NNmSrqwzQKmS zlqHOep*(762``0B>WfH pF>BEsAQi;l4Z>VXM5Gu*ew zXT*~Lz~N%nea=~a&(?ue;Xzj4Jy&0>#>!y@>lW#fOk_)l8t5Wi+QttKsHrlY(C^p( zbJiMf4@BWRjxhmgdmDfhN zUsxYpK@~}0w?9F8+0I8Q4G2o^JD6`$c!Ge6*gU&jD z-L+^V%gz9(3io|Ly6lL__ly)o%5P-JvobgWo&a yr26%MCLxN zGpXQ7tLTaMr9wk2o!#nh#pXu;APXa$0eJJ-`8zK&E}hTwG7grrW+k?fZPfHMt{uN6 z35C9!8yKtkp^e}8bO9m&DLpu}^qC!n=K> 2=f2;g>cJ#VIBpe<0SN3}tWtJLP~XDzX|Xt$y*DJCpH3F;QfP8cih3 z4>Ti1Xjz7Qz@c0GyC+1bDzeLHIrx*m-(L*`GAm}uueHRVp(b!N2 G$tHwd}kx35IamBO5;GjVg2b+gjRU266wQIWc{5hW^aU%PXSTSMjaB`LuEQ;moyD zT3$Xe-L^Hg#c|(ZH&MMP+9Fp$q8 $*Gha^E-{^r% z?qM_J`g#35>-}Z&SlMj$`nNF@JV&N#&)JdkJh2(`w|}5);Y5K*9ZT4Dibp$uuxca) z*({en^va34eooDSL&N9!{v6Ma?g~pz<*=K^uuLyhg8A1*R*FBOBkzEgtxv#{y5}nj(%5kF) zDhNLQZVVD{IQe=OmLv0L^dMty35D_o>THQDlTn>0ECS<)5JncR{3^_fcnOH`MJWGm z)GeihQ)v2~2|OQxTvy2YyW?(1%{xZ#5wL-Uo#tP{m)^sWZN#<}$9+2mzb@uWx_BuF zZ^{RG>upz*;IZ^|=%N93;wtV9gkLpl!)a5OWbt0PAOS}w7zwEhdpv<5^ES~@^=Yn& z;ykxL|A=G%Z%|Qm`D9Gv4xtxpAX<**g}hYl3qhG6sQ5} +Zp+G)b (y5J(2&j}Egkc_D=q{Bw#N;Z2vEzg0! JdMaQXLV8>TE?){wFL{6Ha~SXhZxol&8h2+=g07k zbbla{L4^UpdXsNJQHd$K&?w80yiP)qx6GQV9*YIwp1wLNs)b6W=rEAurtw0#0+>N3 zH)pb~6GNb#Yj}Fdaa0){ab-+KdQEyfO|64-`Mh!;=!5gY#&|5>`P+C9?2o)}FtXc! zwQT)!D_>sDvhCjSX9}>$O*9n6jsGaf3JlH?V@rl7e zKhki={W|?ij_!x`CVwa}SIoLAQT D>)S(|e zEiZ?ft2o08UU(cgIB%OxfA{>XxL|DexoK@6#@7qsQ7BBis^`nu8|)`NCnmL`knGo! zs)C`)IpXrJc$Am$x6knpC+ZzEuzcik$U=dDOGXFbHGZG3*qs*t>M*M)E+~FINV5}~ zM_sCS>n+E9${H^Eu!-U;Q3)$6gzGjOo`Tf0?!zric+z&>vB8lrnp1`gh7X^MxZqiP zmnX_oqRw$$s>pEP!$Gw3PK|`q!1irEw=I%QANZGolleXfE7|EW7@5>*d!DL?Z5NEs z!oIsf77F< n#FNFBxi)Fi3tTKepXGN}f3Z3IV $T6`VShSbld!n*{rMWa+RYZ^EPcLTQ>y&9q24(gO9|(_ zCv&$#Hrk8AKUPEu-ud3YaO~uQfhpR}82Qnq`dFi^ENMcqZHZQ=m@R#r*<=J0z;w9? z-8LWrG_iZB!HV )m4XDmn=W{CqmpqVS4-16 zm`8ud`)UPdT&~!t9jLL*vxG(cbqUy;*j?8$WN9-#28&OjZ}_Le;>r&Bogym#k-eZp zWCk1OREwU6)oT>U6rZb4)~cy4{1D_WHc;=vwVC-)Vi_u)y2&U^Nd5Vu$6P?;$NR$q z+ezuz;EA$=QzF7-a|WHlIS3Q&M?`p0GBLOZlklf#32tlEHZ-q00Y$J&ib_aX8AM_K zJ>RALnAUf?03SjYnE^|_dtvN*+KQWd!%AvDgM7wN!WQkf+L-A)lpFYC_F0np;L1_k zoTgu)<6p49XHXvKms~8=72@09h+A;6t9>S>!gSh*SZ|5-Z9T}}vA_spdaLp9ljgs7 z-lJyoalKZ*yn(8eA)+w!aoevhRo)5AM=35CU9J=nDo(5^!+lW16P^dJ)j>`MlseuC z&ya*0&Q6KW3mLM-Eb>PfcJCMwaR#t;>P-x=+3@@jsg$sWbY)9k*EQRdJm1`?gKl9r zl8f(FN+N*;d&f%4zvB5^R1!f*j+R-HjCyzKj`&~iU(}(nhNNgtU2Y+s^F8CQz5I>u z8rpbbO8S`nGAr)>4trj- 3d!L{ )s*hI7`udnbot9 ll#d#)BMDn~n5VdTD(*kMTKfs?(yBSCb+9^(znD~^q-)e2_J}Kk!pC_H@ z4z=yc72)-ti19kMXg_5YS Vo$c*vXbvQF@$ocN380%%Bc^ywC|>UMHS(xrM>`tCNp%jslD%~4e!wK#`RL9 zo~00vJc=+tzDj~J^z%jvb}3(rDO@>)-3GUOzOqQI+Ii_(bb+1DbI0Im?psQcB!lt> zy`4KkelJ@i5@P*@tev41H2UxgwtqJ3b=!6{w?YC<7S%&UT(%?U1+QH51F2ecMc*}n z$c95-T(8B)hv}Fz^3-}zkzA&fh9l=gmoTa!UivC-cKZn_n1BLNn7M_I-C8pWWF(@$ z-D JPqIdXO^5m7!+!x zJ@0QA3!GxOfg7E^cPuwmUN-pGqj+}4GuQIpFGW95J@+$FOP(vhKFp1Oz2?8?pC~oU z6vkvjjh*c3f@pfXXhZzVo8{%ji5Vd|m9rWw#P509@ag!gjR>fFzQYB{eRU@osFA zZ*v&SYVnr^mfqXQ{ QXGL32HEtN+ zo_cGg>-QJCNG2633R3O;GDh2Fk)qE4i3ILpC6TDW?1>c{zlN%_<$3>?=0-DecplK{ z8yy $dn&HwIJ_BH8J?K`&F^*_7(gAyX>t)JtQlJT2 zGG}R?HUO-_x$pb)<4rNYY}@MI2yOKafM@SDZ%60PSsJ8K7FiS-k^o@Nd_lSP!s+E- z+osTMw|(i|5S%m{1*GHxfwG_~%0=8Fp*(?OI}vJ^Kc}m(Pz1aPEiUoDKgvSlQo?h! zWPPr&r?=9iS2LQ)J{E=)ku0`xF=oHTCsiUg-xPE9@F `y;-d&;qIr8uG;^VMk?U zuQ$A{rrt@r!9uEf9f|qF@DUOY5)OnS{NN{5K^?XL%0VC^0Ac21@P~C}^*-}m^RGP3 zxz@!nixg<&XlZ*ctA_Hqz4)rJNPl>te^^!%-o<2i7%KK7@n8k_hxBxeWw=5GL{X(M zLzBX2;mNAxhsgVG!Gk@L{9_eEX;YTe5V05@X4M}z_&Yz0u~V~I=cJ5ZM@g>-FZxOQ z{!T#hds;bPU}PLPsb7oA?Mo?pa5R%6edHIdFeg??kSP3Xxdqw(hTjoa@JvIUld69- z%k$x2Q&v2q@9h (h( zkmKomr+lw`-^7?#Z2g)$?zzvt 1fW_xzTLny&a$5ca#jj587qW zu|2^iN#f~46O$~JHAI69T%mrHDE#MCq?VhOK9`v>O4pQ^*Hr9()JNjJ)|u}N9hmsM zKc_FYsqc0dmXmCg<)M|x?QKo^2FrV2u9K87p1^X@=*w=czCVzBSG!(ce?#j5E!MRR z%%pErU|`pOKphC`LuwKmx^>@R!~ONWb=quc904o-%Aa 5_&V;}&ujI&p>U(%qPM4Po2FTjW>_s6G%ap^(P~mJ^2W*?$(jCEkcwL>2k^rg z^OF64J+`XOH-658oPg%{x966Fya Sbh< zjR)unyH_P;t=H;KY*7^-vs?D0o2Nm7JAW3Ke}pa}wpg_sWy=eX-y1!nCMEX8+>#1? z>S>J0^-~b&hvWg*SS-QQuNqHm#V-g?>W7AQ5#?(K=F9U2B(Ekm+xihSXuP$Fhy8~5 z4UU8t!L3C$qMyp+@s#Hd)O$Npvz|AdtN%h_C=6w&VyJ?ORwkj;2m?>*j;AV8B2ZC6 zuEzRVWb`v(Feg$7x`=F7VRdy3AI9I9*~M8HC?rvhjY$1n2Z1a&G!vPUYO_(tnD4i$ z@7)Q*&$+a?^`_@xRW}ZMsmXq91iug=T4bUb10_msz#HyS1rr=l6v3RunZx!WvRl!| z9g}H-F@>F;k#7_smuuJWgfCzo9Dsm!P@-QB9Ex30pYQLspnCJuhph;9ZB(fgWQNLe zf;S*jKfmm>Wee65@VzgdZ$mWsTw~k1ZHW3joR$wgY#eg25{OhNisVxB5I()0F9uJp z)tf4g%M@9sri4q`xzl`I_aG9Q?y@nWfx6*a^Lh!J-7;3395KCIVK2`3j(*xTfY4)X zUrAtS${fZ!l41=v#a!WL8hVI;V!HlHA~uMkB?6-nEa^O%O6M`t&6-9iLShdV0{K)3 z(ModFA_g#qcVlE+-JfqnHWBpudT+z)t~3Wd6L|!GCGY@eKNQzr21Vor977sJu=5?b zZ@FX?$eaVPk8Wuxx~O7W;*CFwm*2l0bG7%W)y>V#_pLCht1jsKViPt+%-C_$VPoe1 zaNaDORO_BeGG{X*tBzp=1@H$m2l=x0i!4on0{!ndGyBVaqKzL24Em!<^j?<{dd3qO z?KO0?vY%Y=sbsOvqA7shvJTMb3nKGcmsfXpr7@5g@S7Slq-CPVq9a2gh4D?u_(M67 zglbBtdj6{j4c8KpbB-Jp4Q{ID1LMc88Tb3k;=0;hhvE4Se=N8J-k3 ~tc{F;g@*Uy7-BlsfnL*dHk`zJ|N92bT(-4p)GEPeI z>N;Z!MUu#P2;(4#7=QjZ$V%AM;TdXd{}*zaU7w31m@4sAWK?rW4_&5>;7Oek)pr+q z07Ie^0yRec7NO4CH*3=QsEhq&RxqG(bwDL|_bJfN1Z(OSPZFGedZ?uL6U#f|B>x~# zCw-0KZ~SrZeSnUfk|8f7{Kr(9d r1k1TH;b)`~4-4@7onieDQQ@r8%1me(G_BYOM`ijrD)1cWH+lPm}(zLGEPj2g>N zfWnDqM4+HN0W!l+3azn1XPhVtHSA}`T12oQ=F{waixc(NPx3_+ zS+J%@sYDLnzh?TbC!VkZ;CK9C)sE8)607WB9dn?;kWmcp0KmCu0gxrP_2;$0M0MTN zvwe5nv?qNq9MGg$zef`qLXZQx3e=eJ+n5dURxZW&pYJsL*nd^v*XA)wK?@<^WBeKi zS7J?#{6U3x*2nYPR-W!2#|-5a#182YA}aT_M;Q6hvcgvrZ4}e&=Ss+mGX{Z)S9$d3 zZ!T|9ymD8J4s%|JA#0EZo`HtmC_la%m+*b1k@8M0*3=l~7v4wW4`29TzXn@%I+>6n zMY^d{sZyVSB~7%+*}je?SGWHB;(esGbFel(z;SEWF}YVubKdSIwiSvTgVy9{0p~+m z6h}`M8t<)J75h Z=)gX9ms}{i1#aN1iDchZyZJ AeB84PuM-c0gYfO zyWPHR%KE#0Gmg{qFNFyoD3XO~Zv-~3P>Qp9dgZMll%+{aZR0Iw%%&k6aDNR14<#3E z4DiWePMuREh1W^_9m >;XJ6dPItapj$WQSr6x@mVu`uwGkATne!RPXNj?U9GVOx$DIWXogI?RA+ z4&-JBPpy)F*+hbrnYZ27ld(VJtnfb|1}!jxCDmlrC;OdORq!#1vnL(rRP~Ij+wZnY zi<2Bz><2HmiPj}rw7c!`EgNuKwB^JALd`$R!^z(I1mjc2=dz#a>d^C%3ZNuO<9l6n zdp(Zo$z}fGhz9t$hu607!mfGCr|9b()oGX-Lz~a(bmhuj*9s-HctgUfTF{%$a;l zXw;IWkV&HWuC3KSMIdxg8hP%(EHw2)$qqL(?HLzMpZv?1o`>>VUhh!ytsvmb67v_W z4PBhLN#pKVbXg?T@zC-FA;=T1-0Eok+!UrY2Lq1go}EW<5wlP~cmZ*R{S z-1K|n%Q1!>D7_=`lSDmNiA)Dv@Yt=1GNGD^b*)6xoG!3ECHc&_g9;3ncJFs~FPoSA z8;B?@CYoW?ewMkULGb|CL-KRn=}FxJfZcc$Ho~Z~B#vjdH6)jI!%`SQ2NiPuMbl0C zkk6;fF{{C;;vI>w9|XwbZnSk~h~p)#tCEc)GMt_cPx!U={keUl)G|ef85J>uN?#ox z7bo9CXRL+56lsu&rIBm{p(5hyRXnhr?MNg5A2{v{=EWAOhyXquSYX4|Qi Y_H!+8gAHVU4z63MpzKT9Fh55mr%Fa^w ;91{dhK9&Hd83c&ug}F5zTGuu^vz-1#_K@XEN%ua2T3aDJK(988`Zu@ zr+O?p>wd?r*35FI2#jABB; ;n zk;g)4^cfo_3s?yuk9G8N>@37DyhLOhTDvhVN_=7I;WJ*gO0}pilHFo!e0`j9 F|8L z^}1<>&iM}dlJF^S`3s|@b&)zGZL~j48E5)**#sA^@i9_SDHk =oD_nY|yEY!RhF$3g*g?N0`mEi3Mfuzrt zGW-xG8Tz<8drbEwoyDCQ+Ye;ejNfGAj5IazBG@T5XBN~9hq8J&rIQ%>QH;9GZSOHG zXPn70-)((u2JgaNUx&W{WEX%BkVg4+&5dmRal*BRF5-*tBktPs+Wi*CM4&Br6=L!s zvgq=*I!*_b v #G&tq$CK8!Wp#{eEs )-MCfFq3!KNSFTM>U^15H{K?njFlHe};LIE^k9+JFMM?$m@TYCIH zsStewQD4G1unj9>wFA1F+cWIt^Qh7BzS^$X8&6^YO#k7#XqN8aQ2RzhVA%Ev-bv|G zqx!{+nn|~|DHAt04qmwFTBcJM&mNyJ?-ifTn5Nv-$1Iy=L+gsHr5%pwpkpqKljMtR z%pOlAnR9*xuWK!S3}_@0p=s=nnjloIAdoKVSMO`MSSm6fk5Rk8W%N9Xz>@YSl=t95 zHR+DaI+Ib6QWf;HqVO4ce%^-lC^6ws`}^RtN3d@+c^e^f+><)Vx7zSL&}iKS56X m6Dm zw>h|ZPbE}U0qIEeHtTbbJ$@jn1zd1m36ReHASO;9FDxGg6czAW_V|U3pVP
dr9Md3zlrxTATe_DRtf(T_It5@I@IgXL zJ>rT9PFU^BA&ql&XhH=&4uCkt$c1cK$l4AUz`MipjIC8wsi>=5=Vk%%ix9vZk+5;6 zh<~XXH?`7^;e)N$=%&8Adf1Ms-`q9W8%qr+^{?_P5|~+jCB*}{5G2?+l#d-zPpH(m zpLD#?<}8DJ-}gE(qUx!*rfPInAeaf_r)`@6=TBuZE9!|83RNL5_M|ss#?hXaoih*@ z6Z>Q3;*|_5aguMdOhE7NS6f@_gyW+=I1(b@jX7JeBDD#NgDM z!=T!9_d4yKaYW09V#OJ{NBo6D7YHJh-l1p`;UPls9YE5_3?@LW1;A6>94{BW01Xn8 z+xt5ftoD3ABcgR%Sk-wjyz+a 4Cc`YV=U4RVOXM#f`fzqY{%qryz;iudn5J|M;pn^0MTIvfa^R884k&t z99PCoF=S^zy{N$OtVpsmE;TK|u_a9V4lNVpVuOr`z-lzwDBFF25hzU=oW UAYR zB{3+UFyp(%G1Tw=0Sm+QOFO0l+1={?M29-|WloU|m9S%`NJUcV>pPLB9pM*A)wTD| z$ *jQ$2+s$cJ)osQ)QE{2=dMczCLB{6Vb(VVjlbovaT 2gtni8NnMI z$Wp0hI&G^}za&N+s3EfU)#HPJT*(AydwV<4fj$MZ_A0fUZEAf8sKlg-JMbo#tcdCm zmEdNBI41%+f2ZrcTz&e->aNf*SW3fs-@|zIAqwArH3%UXQ17_HLgu|u@BzyGi@Z Bor_X%meC#8;;-D=c9 zrgF8{z~tXM6oCAtUAH=6WaNV_CrCz_w$7lIN116L{qd64aol^|cZs)D5`D(9fu@%+ z+6XQ{Ffl;6dOkHRtqgNeI=^L_k#NpIf8o%n?6pA+z$>!$MPe`E5j_QYC+ST~D8^08 zJzQF c@T zISr{KUp`a;OG^_Aaq7}W1FOR4I{GSg#Z&zY8bY&@9eCxXHOmI`S>E5dMDiqsOwo@0 ztw=^I_YryyvzwE-EX+Ts5YE{z5NDZcHL|c(viw08$4JCxsCVX#;_d)Nx|hIpWf<_l zP`V@Kj|MjZJ=DF9CTz8OVww*f@@Yjk9=yO nE@pr3!h zHSzKBzm{zZ66RRU9RBRgBv1NnG8;yT1#qxW;|ii;PKqdsYEu`|^oI37r<2>?%2=`D z0wx5M&E-Z(lZyt`Cj+$%P3yM^fY40dMT $ $;Xg*d6 zO4Nz!2GD@o01R|+xy#k7_oD2)$N& q#G?+)OWZnw~R|WY`kP}3fG;T1c z8aX P ztuA+zn4qg;_WcV7!)&%JKipSF4~6(kaK0egpY$_0LMJNO-y=iRkr?$qi{G0})eRmN z`FL$IL?B}Lo XWTmDONpSe)UtT*KJ{=5L}aF z(7j7+3UrVnBO|R (cxdzfKy8szT<~sjkbzA1tEnWlWui?uqAPlw zi;IfDI5;GeED3*9ecT@rF`$(HnuanTiAuGM*UsPEunMB8#1ihM?rw0LRhVg{T74-b zkKVSy%|ztA&nm%BRSGAU3|6rbCZ>$S-uSCuh9cN5t>YpiM}xz~8p{|ZT67x!2(f-? z!Pu!ZI5G@t^m2T3Z+*Jd@ULIL6QTD)YLD-D+Mu_)V5G+({WxD|Wf(O}AaIxnh#q++ z4{tsGK@AmpomrncANsuxrS NMC{P61~ zEKM>fMaPkSk_CUT67llty)Y!N4^a$Di7PQVZ!8^Fr=h`G+5PRRxiEJrh9%8kx` (owZBu6{ zX5)T=cRUD>Cgw`NmQN?be@|;VzCG^$Q;U!l?$izv8V$!$a70JhFDBNC9l0>S2)O5U zo2C^3gthbcxASX2ap!O|Eq?ua?(0kS>#t$$3H%8n+1%BQlBMay!BY}{uGU)uyi=`u zOMke42Ay>w4{akedf&x_e6xj*!cN~YcH&D0BZhMwrnnNNhuA0Yc!ZhGD(|f^Qdwm_ zKsRbXEqcEH<^AeQ%fnX0N2pHpciO?n-p(1V;3O@drUh;{ZbsdS)bvJ7DXAGJHCrHg z{I~VSw?}go+AVQ-cE9bZ8kc~#qLX )mA*K^)SQ#%EIvbLa3^F z3;u7lp0)s2?f-HCP(W}2HA*TB(shanKYzg;P4s&xH3j^ER@#f_ie@S2)5Fodn5*_l zbZtM@$#9(D{3q*z5xn3=1C)hyZP+H%hadPZAhqXr+t0o2ZpUhVm^&uApI_B#G)B #PlL>vo7yoh7*=3}{85SI+WXv-DEsZEm=mDwH@{qs z`cjd8#*Q$VFF-`2QMDf>k_gw#PBLR&W*76#Y``A%Dw>`1&Lh@?ybuwdaZh4ayh{wA ziIhibRUrFRpUo_1k&)z0Rq?6Dm}8YvR(XqK4=5D&srVm10{9NVtnlFk&QRMX>9$qE zvX5DdVY`{34S7CDQu*~62A`$bv;mU0*O79zAtU%ZAYY5y|E7pB`j@LgV~?y*LL3z) z(ZTqe^r9iKUf84O(bUXrHqvuXnw0~|oi8Q;iE;nBWzstufuY&2TVJ4fCbdx84rw+s zf(_V)%_TL}Q<_1FQRb?aJ=_WYcYwydN2~V;|L*_=JkhwN-!RoT!K7s5Vy(3F^U-Z< zW^)?G!8bysSD4taC&@LVS`D97yZ)CK T6QgOak;G)*0POd+g!G7+qT=Xu`Msx{qFPq z|LzBSum@M?b)9b Sr7yXp1MvMfb*d3I3uBL z+XD;Y#5m($%bG9ytRS3#OgNTW=JV=WV)pAofW5LEMBF)9rgLrXN50wk%h~`ir3U(? z*$G=DI{q?es3jH!-*MAiYbi~361?Z$20Vh-! oX08e7hBd5+#_ydFONyEV{;9kPGj6x4_&8BTZ_iR=W)f{P%bIV`r xAPoTAmhJ{*Y>ATtD)+^6 HFsY`rH zXelnJG5_BWIfyr60R4v7pZ$6j0NR}iIH)UtQd7$$A*+E{l-SDE2UoUt)ED&dOyt=1 zb+JNg2rwEWF#JGwXDr?AxCEq?sS|MG(g`Y`HHfr=z?F@XR6SZ-KkD?~BL-?6MlREt z#t{5`hREC#o^C`Sd=$O!=U#ON^#5jeTYBxNWeEqKsYA~_Pn10cYVhW(#46NY6N((2 zs>^>(FPKH?_DB#FYRJmemvk~JB_S)*!r?~GSV0t`d-AA$-9*K&udC*__^zwYk8onr zuidD;tD-m(udGdDkW&um`1R=esBn4-nFW?)elL%0&FHrqKPt+aq>2{oePs-R)5-wW zz}0$wUe>=pLY24SQ-w?107sTg91Lg&FVo3j6&4jae~ a{Hk@w!4Mi$Z9D?`35*7Q~tuJQYP2rPmZ*^@`Rldw7}^v@chRFCW_0jP32QhU`Wz z%S#Km${# +O7z@PG_m zM*o!evxD$uN|y-`_bzlACFV~gdfeB5lhISrv_3kf^N*u-jpbDbYvZv)oRXfNzhbE- zPG4}bq|27AQ#DoEV+gkP8>6rDOk=hr!CdlI9o(<)@XL3Kpp&9~R~`Q3zpW=MI(MV$ zwLmx5c~!uAUaH=K1e$NMTYQt{p3I8qi}9K<2ZinRDy0}ADux<^AnRk+xh_?oIdW|N z_tU}^u*(%Q4 +WCZu|KAdR z`7XS-TeDbN3k!iJa7f;HnQ?lSeo(#(S_#_@Sq~(dC#BSpJ72Jmj|SaTkn0!g1W7e5 z{&qfCzv|Ax3f9?>LmOT;&^lt(``HyhZGbk<6=!*Alv^p(=!;_<)-#we{QuI1$O2nfn>1 z9(1qU{%yl#kFFA7;rD4X5UZ%=RqvmKK`@fc6C(%$6{#-U@`^l2cEQ6qKA1r#fEtoy z2!h0A(-*X#-%!e^ShpV0!udl+O7=ZxzMxfS&Qruu_uP064)NA`J;A-!!9b%f0(OlW zILTs425xSxSR#Z`22Ed>67}jAT{1srru`h*yL??S5Em-NN{DNi8+0n1a`>Tv7@AU+ zp%d)=ZP&~v`4>{%HJ|qh|5|tWUitp;n?k2|4nU+Po|*PI$8o?FW#H}r>JKXp0u&7K z@v!<~4lUmQ!cF2=M2Zha+54`IKg_F?kEWWhW -J1qU!)I#XGD}cD%7oPCTdx4nTR|Pz8t)? z<5PV2LGrbVS31s6t>e8#_UreLHCicm^wC=o{cjsHwV}Pb!K86lI+GNIg{Uwd989sHEZ4U${rMIONe0Uy>?_@#@Uyg3{(I7 zM~4sQ-JishqnGMhe)V7CWO+%O6LPN3a;&gnng_Yn27#!l%UXsH!s-*TPQN@|lV zv=>=+WTS994r9{87(tRY{nY=Nfaid(!x;XN9)5{}-XJ|`j@S=%U&t-xXsB0aUIyzj zm{mH~kZ*o&o3FB^Qm4k|En@if^>((za$s_b1g23|QU%`rXjt9miOi)G4-y9q9ZbV= zeHULe-q%a1bp{r`x0N{`)Utea!+)wPUNO&bW>xsh Xh3ZNuvp-Aj3|fzBOeK<05K_PX<4wbg)c3@Gc(l^-)rsAV(@V9YBp zdY>0x8pqqnm_AOREWI7SOn2)5`+U>l+XK<75*D}W7FuwsZXF$Djl}E(0lO`^8@%ys znij9C1J1z(ia5g#$+>6N)6GUSwxVFHSJ#yL`lSizAb^?(RM|!hA2+fp!k>R}+6b2H z$stJS*C$M>aicsu3BOgZGz1mSBOuZ3@lUy;xn@#@--M37({#I(RV-c6aZ@1`J#1d4 zD{e%kD&ua| zq#c4y<^4}#pj_A0RK3bbV740Eh<-T_Eda+eKA6MCa zZMsP1A(97iJo59H%~1P!z6wEUpX7lqo2aY|1`K_{msUrHHS45Ts@YP Edv~~`E4(ZA4^Mw!tbIgg=e;t7416`O$24Ib{;r;WsEAgAA0_a z5EtXw<-z8?Uflj-{jXYHU^ipRMI?B;30<#gRlf)$rbJk-_T{?CL`VK4>0*`!l=R1G z>N>78X}mW3pw`=u;G6`ntZd VlF*pF=6ztMWng{x`ttHrhl=!*>c~F4 }cL5{NxTEbvzMHG{!mRi0&4=x0rm~-}Y*q;CE|kLCxAYD6UMep9 z<+~(nKU+p){FE@qmns}+-F6s5nkspxqdnD|D>P7}g5J*D_<6LOIGmQ5{l)oyV4pTu z2`?((uH51(+<+^*G38|u36tgPKA%C#n4F@sS<|~8_P2?#B>(N+FW)_9xdxoR_OgML z2DF}EWgQ@|WCBU-s`G{?aD^iA0o9D8q-5qra$=G3x{FGK+5%Gu22=r#rUi)8N!s>^ z0HZ&Mez}e2-%mEc5RW@Be$mv+w!TarpbU!|uN~j*Fb-wc-GNkAppk+>+a=6al87m0 z*>egA;(~*``#r47VT7Wbu@_TDu|gB7MwC;}pdFvwKx@wzyh`v*BQYoJyRmG}C%wnL z60^5{)KM4%45xP^@*cbXZ;Tf_qp{5xsjy>{b9sKgKiD6IHzG`)*Up-R6OF=>yro|< z))&o|pBrOuRYMrRmlG`y`v|QxaObF#vS3zIbMbzUP(V?Prri5hO!(0^`tr}|3xRL= zG!_H7g8rQPXgc52*gKuv8EP4wdMUrfWP0- o{f>S^|7<1rCGi?lVj5a7OHU4Q$ zbzD?SBj}I()?|fOc$0Jyf5 NPaWLw|6~Q<24az(}k-oM)bCnhUw${hX)hEK}sCC4XBVN74^D*yXSR zGGT8?wFODg6QFRV$Z3?lM_mFXwh|Jj|Ilsn8B0;5tO`n!DD1q8vnIeS;D6$R=M`<1 zS3)$1ehm94d@Ez!wWBlWwqvN012g}Kr;J5e@IIA>&zFCQ91U~+%mU30#f~6VP**1B z_+-;%&@>Ti7DaHS8#nJ(_)2_O=~jVw%1$&`(01yp_Qg`xX6M5al@YSi;qvkgD#ej8 z!!G~4`4wEvb?ziI$xsQLuPj3V#$r4u_P!Gu!+eu-E$54RdwuTJ^$k(sr_nPr03&wx zM`W;>@rLYh?*+1KNs8`u61sNwZ0@wyFYwlZ&)OxBF~v-I>~jL_Ru?1JYIhz@Z=IlI zZ-R1fyqjHK^>rnC*>>A6CjpD@pAV&TLp#G9SA}39Cwx4)VS`uw5jHMQg;vO0X46Fi z_i}h*x*Th9DX9}b9Vi>Y9d!Ob^4@aV$t_Mz)3T|W@RbaIRW8Neq9P^iun|I=BjHBQ zn0-fBC5qIHXF~0X9QFxC_>^nwNuCnqD5_MJGZPS68W5VxMUz3sNo)VpU$F1`B!3j< zeDOn;A%rJ)ud$C{{|yaexKg05hJ|B5`I~bBHZ*cr7_xW^?~yRIcx)Xvyg9-M7_t)C zkUE)M94ypYmtk(Gxok+bOd?jZqN8}!LNULv2gjeNeubD-NXOjAnaH!Sl+v{1=o-t8 zVmC3J?~57|Aet_k7oT|S(zjZy;s|Kz;O71JMB}ZKsh|*utO|MLvk`rLO6(*^0{K`z z$OIgS6{cBc5=<+?)oYQL#6p!UlHW7^ZC3g=8Wb8J=n3;fo*YTOKzQ0#2;qHldag1$ z<5abgwMtm6Yo-k1nP7AedA*1xXhI|b4VU&3jSN-*27t_a*~&jqF<;_OUB;+@hCyCl zo}8Q<4&1V?39nUB?uz9_lyf_@&lqbLsKsqb@GDG;a5cR+_akE&3#h~7(DE_-ddNYB z+%=NnygmZV5{7{O5nx<$ii+_@ges%Qp1 xR^ST z#_KTm;(@vme5&)nt+w`r;0YU#h|$beOG3*-<>Lj{i%XuvqDRd}QCVwzK|*&At&f^{ zB1-l-lmF=Br`&EEj^tEvN(_X=`3a{FNai}IRXV)uiUr;5iwP>zOW=M9@rj*L#&-M7 z)m=eRAdWB`8N(_rtJzKIBZgono=W=x^Y>l35TbSn1U|wlEFapuAcVZP2j^hw;;!f# z=L1avWvVSEqZphO*|$pwXruPg4mmx`U%6 +)a-F(_a |QVqOITt zok7XG9w0L`c8&ba(Uo}N1pq`Y`7hPFkiY>iPp3-cNQir5TC>i6Pc};A+2z$NvPk5~ z%wt1JRtw!zimgg=v=xKa{wqEMDe$&If>;NwHMP2+22Iqd%Lw>}v*Jm%cjHZ0`Dpg0 z@MxNQrWwS%rbO%wi1okp#or6oKx&-Y{vN5?{>|G(nC{bd+L2Wlkcf`b8e`*EIF`VX z_;s1vNO6#ACz5IdBELK 2=deZNv8Ih@S7=UbR*$@=aiQ9gcO)ti+sF5-4>)1Ph) z>!Fj0@(o2 abcFLD6M6}J^ zI{x5cDf_eAlE$F5F`J6v<;oCTQ&;9KFK6WM!#2-Ky~XH#KLR5^X}BS_$p%4-EpOwX z{9V#ovK39^&Alg8LVgH_g7_w49KoWZFanAxEQ_9HtyXL*-zo|w*Sj%tbg_!}^5P~f zEiABX1l?58sTQAk-a4_w7c%G(Y`c0XZ@y!q9s12wKcK(DO@I=Cu?=WW!T}fnK>e_# z{Iy2^PJuV!cdbK`LQ&sq#W}jE+39gtE@+JGGk7WZ+KUyO+L`^x8b}$v@NdCps-0cs zQV^`SdW+s`4tal41?pzIzbyxGcz!$u?(23D-LwC)!=@*+OT;CX^3 |mBAi$4;8AOQ zx}N5X2uu-hU9R|bv;3D$_KxIkL|wRN=XaetUliBF4q6|t{ i?Hr^#1 zb)F|~!>8NL$Np}4k%1BR*g5vjr^2iA7_j#`>q{dzVuI4$@XYeZ@spsULQp=4m{XKu zI5eZGokxQyCcu^0Y?j=Wk!t__qX^3dH2{MvpwB|bf$?Nf5XCI3#2Wu1o&2oD+bUYq z0ucxW3 aj=dT>9;e=RX^u@dk`A2<-oTLCGx^JP%WE_wz>+2m#q_gqC$n zAeX!)VJc}i*cxM}pG!V{2lj;!XQ@l+vPE+}cIoC?I1! zTjO^Egof*fj%APrHhi?KTSL3 nfORa|8a{Y4ep(S=@qGDq7pazHaA=8>9St4a>%=@z`7WK9@@|K|3nXrbIgjh7 zPh^;E6+L!66DlF@?RP(_HaA3GvY5=_<*u+0G#^hvyRO3Kh5juRf}{fvlxdPy=Y!C? z`-~LAh{mFCs8?3jh68Jr=h{Q0z KL z<$|FT=eO=p*T-b&x-!_ ?uovLNRaeyTxtIMjK_jzftHSikb2islRB^NpS12@`%YC1Do$-S{Ya-#sX1xtiV! zTXI;R6fXyjV#+fSFd#G3m4Hv?3`NXy6_Xjt)p_#ojr}%a6~zoxoE?lL4gV5ZNP#2W zqW!cl>u!#d#qEVWuid?}RNpNOlXiD&35y!9opoM WapIs^CT!d1TWtrW~ndY8VszdLeiKut5 ta}ao?Xr88Nr99eq8`qD1kPp!SpuMA03dU zxEs4^$FF!OfAdRpHfuLxPN#41)GO(iI%HnnV;}J1n||W0EruQHdcJn#T^ph0-YlTY z_>fpQFLkqhK13!-hbaU6h#%G4FPT{lCMI|RkJp}~JKksGgB0d$V6@kj&6hf3p{jS6 za%3CTkUy?ucE}Doo=}aAt7dqfD!BtFf`woB3U}E5J}fBCXx6!1L$A7xRb-3d>k;-N zs;YjaEqwT`hAwwU^5#MtQwr0|2h*65vdx0o<#~&)dh7gKj5Dv@l+cEW;g;8`!<3Mv z<_+Jiutq8r;4h@D@3OjV)vy|RcyNk5_e9@49SBbmJVe4-uJ&KVqNSkTWXwr35R!fs zO)Uh;6<{F^U!s(yfqwjVT+kEFn3U#bNyUtG3_ti$4|c{V;sPI!8!>Q?ZfMjO)l!nw zHr5ygcvgSNxOLgVx%n*M7#9yxz{qXAq702+$^F_CEtz^+aW6E~h8VxuVvRw>_H z^HTF6c;VKz1xP>*cKA0RU=Bbc>%x3NV1i}oe0)<9{TDIqAwU$N{g{SM37RZfbIgEX zAy0(wpM+c_a_93?*3JdFh{l@H7Q6RY3H;h%#fux% m9=j@ehPV zSmN`AJU)IGeW7npS<$`@d`sRYVX$xi Z}L3=y?ePe%a!m7tvom5l;T^!7J-0#_Z zxPw2VNQA*O@B@1NYCEqx&F~4@+0TVVQ!q)*3ezw~;zYT-m~C+5T&g2RKKd!ZC{J_| z`rG^J2*6=*+C;Q&droea^gQ||qKGZj!*zUn;EmrSGjUj7!b!*DsPvmGhtgNYu%3?{= z&cz5B`iJU CC$IjjXo8+Y~iuM$J8KqWf= zgSPQ=hex+tc_-YzDLiqrxHi$7msIzTO&rA)qGCWa#K13~*&+Ets`E;TfZv<=8Q`qn zQAe-*nZ_8SLB&m9@Vm}Dlar)WQ^3Zu_;fL0gP4LEKzYQkHm7E(cGn9_ 4