mirror of
https://github.com/HDFGroup/hdf5.git
synced 2025-01-06 14:56:51 +08:00
11dfa25910
* Updated source file copyright headers to remove "Copyright by the Board of Trustees of the University of Illinois", which is kept in the top-level COPYING file.
2038 lines
70 KiB
C
2038 lines
70 KiB
C
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
* Copyright by The HDF Group. *
|
|
* All rights reserved. *
|
|
* *
|
|
* This file is part of HDF5. The full HDF5 copyright notice, including *
|
|
* terms governing use, modification, and redistribution, is contained in *
|
|
* the COPYING file, which can be found at the root of the source code *
|
|
* distribution tree, or in https://www.hdfgroup.org/licenses. *
|
|
* If you do not have access to either file, you may request a copy from *
|
|
* help@hdfgroup.org. *
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
/*
|
|
* HDF5 Subfiling VFD tests
|
|
*
|
|
* NOTE: these tests currently assume that the default I/O concentrator
|
|
* selection strategy for the Subfiling VFD is to use 1 I/O
|
|
* concentrator per node. If that changes in the future, some of
|
|
* these tests will need updating.
|
|
*/
|
|
|
|
#include <mpi.h>
|
|
|
|
#include "testpar.h"
|
|
#include "H5srcdir.h"
|
|
#include "H5MMprivate.h"
|
|
|
|
#ifdef H5_HAVE_SUBFILING_VFD
|
|
|
|
#include "H5FDsubfiling.h"
|
|
#include "H5FDioc.h"
|
|
|
|
#define SUBFILING_TEST_DIR H5FD_SUBFILING_NAME
|
|
|
|
/* The smallest Subfiling stripe size used for testing */
|
|
#define SUBFILING_MIN_STRIPE_SIZE 128
|
|
|
|
#ifndef PATH_MAX
|
|
#define PATH_MAX 4096
|
|
#endif
|
|
|
|
#define ARRAY_SIZE(a) sizeof(a) / sizeof(a[0])
|
|
|
|
#define CHECK_PASSED() \
|
|
do { \
|
|
int err_result = (nerrors > curr_nerrors); \
|
|
\
|
|
mpi_code_g = MPI_Allreduce(MPI_IN_PLACE, &err_result, 1, MPI_INT, MPI_MAX, comm_g); \
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Allreduce succeeded"); \
|
|
\
|
|
if (MAINPROCESS) { \
|
|
if (err_result == 0) \
|
|
PASSED(); \
|
|
else \
|
|
H5_FAILED(); \
|
|
} \
|
|
} while (0)
|
|
|
|
static MPI_Comm comm_g = MPI_COMM_WORLD;
|
|
static MPI_Info info_g = MPI_INFO_NULL;
|
|
static int mpi_rank;
|
|
static int mpi_size;
|
|
static int mpi_code_g;
|
|
static int num_nodes_g;
|
|
static int num_iocs_g;
|
|
|
|
static MPI_Comm node_local_comm = MPI_COMM_WORLD;
|
|
static int node_local_rank;
|
|
static int node_local_size;
|
|
|
|
static MPI_Comm ioc_comm = MPI_COMM_WORLD;
|
|
static int ioc_comm_rank;
|
|
static int ioc_comm_size;
|
|
|
|
static long long stripe_size_g = -1;
|
|
static long ioc_per_node_g = -1;
|
|
static int ioc_thread_pool_size_g = -1;
|
|
|
|
int nerrors = 0;
|
|
int curr_nerrors = 0;
|
|
|
|
/* Function pointer typedef for test functions */
|
|
typedef void (*test_func)(void);
|
|
|
|
/* Utility functions */
|
|
static hid_t create_subfiling_ioc_fapl(MPI_Comm comm, MPI_Info info, hbool_t custom_config,
|
|
H5FD_subfiling_params_t *custom_cfg, int32_t thread_pool_size);
|
|
|
|
/* Test functions */
|
|
static void test_create_and_close(void);
|
|
static void test_config_file(void);
|
|
static void test_stripe_sizes(void);
|
|
static void test_read_different_stripe_size(void);
|
|
static void test_subfiling_precreate_rank_0(void);
|
|
static void test_subfiling_write_many_read_one(void);
|
|
static void test_subfiling_write_many_read_few(void);
|
|
static void test_subfiling_h5fuse(void);
|
|
|
|
static test_func tests[] = {
|
|
test_create_and_close,
|
|
test_config_file,
|
|
test_stripe_sizes,
|
|
test_read_different_stripe_size,
|
|
test_subfiling_precreate_rank_0,
|
|
test_subfiling_write_many_read_one,
|
|
test_subfiling_write_many_read_few,
|
|
test_subfiling_h5fuse,
|
|
};
|
|
|
|
/* ---------------------------------------------------------------------------
|
|
* Function: create_subfiling_ioc_fapl
|
|
*
|
|
* Purpose: Create and populate a subfiling FAPL ID.
|
|
*
|
|
* Return: Success: HID of the top-level (subfiling) FAPL, a non-negative
|
|
* value.
|
|
* Failure: H5I_INVALID_HID, a negative value.
|
|
* ---------------------------------------------------------------------------
|
|
*/
|
|
static hid_t
|
|
create_subfiling_ioc_fapl(MPI_Comm comm, MPI_Info info, hbool_t custom_config,
|
|
H5FD_subfiling_params_t *custom_cfg, int32_t thread_pool_size)
|
|
{
|
|
H5FD_subfiling_config_t subfiling_conf;
|
|
H5FD_ioc_config_t ioc_conf;
|
|
hid_t ret_value = H5I_INVALID_HID;
|
|
|
|
HDassert(!custom_config || custom_cfg);
|
|
|
|
if ((ret_value = H5Pcreate(H5P_FILE_ACCESS)) < 0)
|
|
TEST_ERROR;
|
|
|
|
if (H5Pset_mpi_params(ret_value, comm, info) < 0)
|
|
TEST_ERROR;
|
|
|
|
if (!custom_config) {
|
|
if (H5Pset_fapl_subfiling(ret_value, NULL) < 0)
|
|
TEST_ERROR;
|
|
}
|
|
else {
|
|
/* Get defaults for Subfiling configuration */
|
|
if (H5Pget_fapl_subfiling(ret_value, &subfiling_conf) < 0)
|
|
TEST_ERROR;
|
|
|
|
/* Set custom configuration */
|
|
subfiling_conf.shared_cfg = *custom_cfg;
|
|
|
|
if (subfiling_conf.require_ioc) {
|
|
/* Get IOC VFD defaults */
|
|
if (H5Pget_fapl_ioc(ret_value, &ioc_conf) < 0)
|
|
TEST_ERROR;
|
|
|
|
/* Set custom configuration */
|
|
ioc_conf.thread_pool_size = thread_pool_size;
|
|
|
|
if (H5Pset_fapl_ioc(subfiling_conf.ioc_fapl_id, &ioc_conf) < 0)
|
|
TEST_ERROR;
|
|
}
|
|
else {
|
|
if (H5Pset_fapl_sec2(subfiling_conf.ioc_fapl_id) < 0)
|
|
TEST_ERROR;
|
|
}
|
|
|
|
if (H5Pset_fapl_subfiling(ret_value, &subfiling_conf) < 0)
|
|
TEST_ERROR;
|
|
}
|
|
|
|
return ret_value;
|
|
|
|
error:
|
|
if ((H5I_INVALID_HID != ret_value) && (H5Pclose(ret_value) < 0)) {
|
|
H5_FAILED();
|
|
AT();
|
|
}
|
|
|
|
return H5I_INVALID_HID;
|
|
}
|
|
|
|
/*
|
|
* A simple test that creates and closes a file with the
|
|
* subfiling VFD
|
|
*/
|
|
#define SUBF_FILENAME "test_subfiling_basic_create.h5"
|
|
static void
|
|
test_create_and_close(void)
|
|
{
|
|
hid_t file_id = H5I_INVALID_HID;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("file creation and immediate close");
|
|
|
|
/* Get a default Subfiling FAPL */
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, FALSE, NULL, 0);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
CHECK_PASSED();
|
|
}
|
|
#undef SUBF_FILENAME
|
|
|
|
/*
|
|
* Test to check that Subfiling configuration file matches
|
|
* what is expected for a given configuration
|
|
*/
|
|
#define SUBF_FILENAME "test_subfiling_config_file.h5"
|
|
static void
|
|
test_config_file(void)
|
|
{
|
|
H5FD_subfiling_params_t cfg;
|
|
int64_t stripe_size;
|
|
int64_t read_stripe_size;
|
|
FILE *config_file;
|
|
char *config_filename = NULL;
|
|
char *config_buf = NULL;
|
|
long config_file_len;
|
|
hid_t file_id = H5I_INVALID_HID;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
int read_stripe_count;
|
|
int read_aggr_count;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("subfiling configuration file format");
|
|
|
|
/*
|
|
* Choose a random Subfiling stripe size between
|
|
* the smallest allowed value and 32MiB
|
|
*/
|
|
if (mpi_rank == 0) {
|
|
stripe_size = (rand() % (H5FD_SUBFILING_DEFAULT_STRIPE_SIZE - SUBFILING_MIN_STRIPE_SIZE + 1)) +
|
|
SUBFILING_MIN_STRIPE_SIZE;
|
|
}
|
|
|
|
if (mpi_size > 1) {
|
|
mpi_code_g = MPI_Bcast(&stripe_size, 1, MPI_INT64_T, 0, comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Bcast succeeded");
|
|
}
|
|
|
|
cfg.ioc_selection = SELECT_IOC_ONE_PER_NODE;
|
|
cfg.stripe_size = (stripe_size_g > 0) ? stripe_size_g : stripe_size;
|
|
cfg.stripe_count = num_iocs_g > 1 ? (num_iocs_g / 2) : 1;
|
|
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, TRUE, &cfg, H5FD_IOC_DEFAULT_THREAD_POOL_SIZE);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
if (MAINPROCESS) {
|
|
h5_stat_t file_info;
|
|
char *resolved_path;
|
|
char *subfile_dir;
|
|
char *subfile_name;
|
|
char *tmp_buf;
|
|
char *substr;
|
|
char scan_format[256];
|
|
int num_digits;
|
|
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
|
|
config_filename = HDmalloc(PATH_MAX);
|
|
VRFY(config_filename, "HDmalloc succeeded");
|
|
|
|
HDsnprintf(config_filename, PATH_MAX, H5FD_SUBFILING_CONFIG_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino);
|
|
|
|
config_file = HDfopen(config_filename, "r");
|
|
VRFY(config_file, "HDfopen succeeded");
|
|
|
|
HDfree(config_filename);
|
|
|
|
VRFY((HDfseek(config_file, 0, SEEK_END) >= 0), "HDfseek succeeded");
|
|
|
|
config_file_len = HDftell(config_file);
|
|
VRFY((config_file_len > 0), "HDftell succeeded");
|
|
|
|
VRFY((HDfseek(config_file, 0, SEEK_SET) >= 0), "HDfseek succeeded");
|
|
|
|
config_buf = HDmalloc((size_t)config_file_len + 1);
|
|
VRFY(config_buf, "HDmalloc succeeded");
|
|
|
|
VRFY((HDfread(config_buf, (size_t)config_file_len, 1, config_file) == 1), "HDfread succeeded");
|
|
config_buf[config_file_len] = '\0';
|
|
|
|
/* Check the stripe_size field in the configuration file */
|
|
substr = HDstrstr(config_buf, "stripe_size");
|
|
VRFY(substr, "HDstrstr succeeded");
|
|
|
|
VRFY((HDsscanf(substr, "stripe_size=%" PRId64, &read_stripe_size) == 1), "HDsscanf succeeded");
|
|
VRFY((read_stripe_size == cfg.stripe_size), "Stripe size comparison succeeded");
|
|
|
|
/* Check the aggregator_count field in the configuration file */
|
|
substr = HDstrstr(config_buf, "aggregator_count");
|
|
VRFY(substr, "HDstrstr succeeded");
|
|
|
|
VRFY((HDsscanf(substr, "aggregator_count=%d", &read_aggr_count) == 1), "HDsscanf succeeded");
|
|
if (cfg.stripe_count < num_iocs_g)
|
|
VRFY((read_aggr_count == cfg.stripe_count), "Aggregator count comparison succeeded");
|
|
else
|
|
VRFY((read_aggr_count == num_iocs_g), "Aggregator count comparison succeeded");
|
|
|
|
/* Check the subfile_count field in the configuration file */
|
|
substr = HDstrstr(config_buf, "subfile_count");
|
|
VRFY(substr, "HDstrstr succeeded");
|
|
|
|
VRFY((HDsscanf(substr, "subfile_count=%d", &read_stripe_count) == 1), "HDsscanf succeeded");
|
|
VRFY((read_stripe_count == cfg.stripe_count), "Stripe count comparison succeeded");
|
|
|
|
/* Check the hdf5_file and subfile_dir fields in the configuration file */
|
|
resolved_path = HDrealpath(SUBF_FILENAME, NULL);
|
|
VRFY(resolved_path, "HDrealpath succeeded");
|
|
|
|
VRFY((H5_dirname(resolved_path, &subfile_dir) >= 0), "H5_dirname succeeded");
|
|
|
|
tmp_buf = HDmalloc(PATH_MAX);
|
|
VRFY(tmp_buf, "HDmalloc succeeded");
|
|
|
|
substr = HDstrstr(config_buf, "hdf5_file");
|
|
VRFY(substr, "HDstrstr succeeded");
|
|
|
|
HDsnprintf(scan_format, sizeof(scan_format), "hdf5_file=%%%zus", (size_t)(PATH_MAX - 1));
|
|
VRFY((HDsscanf(substr, scan_format, tmp_buf) == 1), "HDsscanf succeeded");
|
|
|
|
VRFY((HDstrcmp(tmp_buf, resolved_path) == 0), "HDstrcmp succeeded");
|
|
|
|
substr = HDstrstr(config_buf, "subfile_dir");
|
|
VRFY(substr, "HDstrstr succeeded");
|
|
|
|
HDsnprintf(scan_format, sizeof(scan_format), "subfile_dir=%%%zus", (size_t)(PATH_MAX - 1));
|
|
VRFY((HDsscanf(substr, scan_format, tmp_buf) == 1), "HDsscanf succeeded");
|
|
|
|
VRFY((HDstrcmp(tmp_buf, subfile_dir) == 0), "HDstrcmp succeeded");
|
|
|
|
HDfree(tmp_buf);
|
|
H5MM_free(subfile_dir);
|
|
HDfree(resolved_path);
|
|
|
|
subfile_name = HDmalloc(PATH_MAX);
|
|
VRFY(subfile_name, "HDmalloc succeeded");
|
|
|
|
/* Verify the name of each subfile is in the configuration file */
|
|
num_digits = (int)(HDlog10(cfg.stripe_count) + 1);
|
|
for (size_t i = 0; i < (size_t)cfg.stripe_count; i++) {
|
|
HDsnprintf(subfile_name, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, (int)i + 1, cfg.stripe_count);
|
|
|
|
substr = HDstrstr(config_buf, subfile_name);
|
|
VRFY(substr, "HDstrstr succeeded");
|
|
}
|
|
|
|
/* Verify that there aren't too many subfiles */
|
|
HDsnprintf(subfile_name, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, (int)cfg.stripe_count + 1, cfg.stripe_count);
|
|
substr = HDstrstr(config_buf, subfile_name);
|
|
VRFY(substr == NULL, "HDstrstr correctly failed");
|
|
|
|
HDfree(subfile_name);
|
|
HDfree(config_buf);
|
|
|
|
VRFY((HDfclose(config_file) >= 0), "HDfclose on configuration file succeeded");
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
CHECK_PASSED();
|
|
}
|
|
#undef SUBF_FILENAME
|
|
|
|
/*
|
|
* Test a few different Subfiling stripe sizes with a fixed
|
|
* stripe count
|
|
*/
|
|
/* TODO: Test collective I/O as well when support is implemented */
|
|
#define SUBF_FILENAME "test_subfiling_stripe_sizes.h5"
|
|
#define SUBF_NITER 10
|
|
static void
|
|
test_stripe_sizes(void)
|
|
{
|
|
H5FD_t *file_ptr = NULL;
|
|
void *write_buf = NULL;
|
|
char *tmp_filename = NULL;
|
|
hid_t dxpl_id = H5I_INVALID_HID;
|
|
int num_subfiles;
|
|
int num_digits;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("random subfiling stripe sizes");
|
|
|
|
tmp_filename = HDmalloc(PATH_MAX);
|
|
VRFY(tmp_filename, "HDmalloc succeeded");
|
|
|
|
dxpl_id = H5Pcreate(H5P_DATASET_XFER);
|
|
VRFY((dxpl_id >= 0), "DCPL creation succeeded");
|
|
|
|
for (size_t i = 0; i < SUBF_NITER; i++) {
|
|
H5FD_subfiling_params_t cfg;
|
|
h5_stat_size_t file_size;
|
|
const void *c_write_buf;
|
|
h5_stat_t file_info;
|
|
int64_t file_size64;
|
|
int64_t stripe_size;
|
|
haddr_t file_end_addr;
|
|
haddr_t write_addr;
|
|
size_t nbytes;
|
|
herr_t write_status;
|
|
hid_t file_id;
|
|
|
|
/*
|
|
* Choose a random Subfiling stripe size between
|
|
* the smallest allowed value and the default value
|
|
*/
|
|
if (mpi_rank == 0) {
|
|
stripe_size = (rand() % (H5FD_SUBFILING_DEFAULT_STRIPE_SIZE - SUBFILING_MIN_STRIPE_SIZE + 1)) +
|
|
SUBFILING_MIN_STRIPE_SIZE;
|
|
}
|
|
|
|
if (mpi_size > 1) {
|
|
mpi_code_g = MPI_Bcast(&stripe_size, 1, MPI_INT64_T, 0, comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Bcast succeeded");
|
|
}
|
|
|
|
cfg.ioc_selection = SELECT_IOC_ONE_PER_NODE;
|
|
cfg.stripe_size = (stripe_size_g > 0) ? stripe_size_g : stripe_size;
|
|
cfg.stripe_count = 1;
|
|
|
|
/* First, try I/O with a single rank */
|
|
if (MAINPROCESS) {
|
|
FILE *subfile_ptr;
|
|
|
|
num_subfiles = 1;
|
|
num_digits = (int)(HDlog10(num_subfiles) + 1);
|
|
|
|
nbytes = (size_t)(cfg.stripe_size * num_subfiles);
|
|
|
|
write_buf = HDmalloc(nbytes);
|
|
VRFY(write_buf, "HDmalloc succeeded");
|
|
|
|
HDmemset(write_buf, 255, nbytes);
|
|
|
|
c_write_buf = write_buf;
|
|
|
|
fapl_id = create_subfiling_ioc_fapl(MPI_COMM_SELF, MPI_INFO_NULL, TRUE, &cfg,
|
|
H5FD_IOC_DEFAULT_THREAD_POOL_SIZE);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
/* Create and close file with H5Fcreate to setup superblock */
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "H5Fclose succeeded");
|
|
|
|
/* Re-open file through H5FDopen for direct writes */
|
|
file_ptr = H5FDopen(SUBF_FILENAME, H5F_ACC_RDWR, fapl_id, HADDR_UNDEF);
|
|
VRFY(file_ptr, "H5FDopen succeeded");
|
|
|
|
/*
|
|
* Get the current file size to see where we can safely
|
|
* write to in the file without overwriting the superblock
|
|
*/
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
file_size = (h5_stat_size_t)file_info.st_size;
|
|
|
|
H5_CHECK_OVERFLOW(file_size, h5_stat_size_t, haddr_t);
|
|
file_end_addr = (haddr_t)file_size;
|
|
|
|
/* Set independent I/O on DXPL */
|
|
VRFY((H5Pset_dxpl_mpio(dxpl_id, H5FD_MPIO_INDEPENDENT) >= 0), "H5Pset_dxpl_mpio succeeded");
|
|
|
|
/* Set EOA for following write call */
|
|
VRFY((H5FDset_eoa(file_ptr, H5FD_MEM_DEFAULT, file_end_addr + nbytes) >= 0),
|
|
"H5FDset_eoa succeeded");
|
|
|
|
/*
|
|
* Write "number of IOCs" X "stripe size" bytes to the file
|
|
* and ensure that we have "number of IOCs" subfiles, each
|
|
* with a size of at least "stripe size" bytes. The first
|
|
* (few) subfile(s) may be a bit larger due to file metadata.
|
|
*/
|
|
write_addr = file_end_addr;
|
|
write_status = H5FDwrite(file_ptr, H5FD_MEM_DRAW, dxpl_id, write_addr, nbytes, c_write_buf);
|
|
VRFY((write_status >= 0), "H5FDwrite succeeded");
|
|
|
|
file_end_addr += nbytes;
|
|
|
|
VRFY((H5FDtruncate(file_ptr, dxpl_id, 0) >= 0), "H5FDtruncate succeeded");
|
|
|
|
for (int j = 0; j < num_subfiles; j++) {
|
|
h5_stat_size_t subfile_size;
|
|
h5_stat_t subfile_info;
|
|
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, j + 1, num_subfiles);
|
|
|
|
/* Ensure file exists */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr, "HDfopen on subfile succeeded");
|
|
VRFY((HDfclose(subfile_ptr) >= 0), "HDfclose on subfile succeeded");
|
|
|
|
/* Check file size */
|
|
VRFY((HDstat(tmp_filename, &subfile_info) >= 0), "HDstat succeeded");
|
|
subfile_size = (h5_stat_size_t)subfile_info.st_size;
|
|
|
|
VRFY((subfile_size >= cfg.stripe_size), "File size verification succeeded");
|
|
}
|
|
|
|
/* Verify that there aren't too many subfiles */
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, num_subfiles + 1, num_subfiles);
|
|
|
|
/* Ensure file doesn't exist */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr == NULL, "HDfopen on subfile correctly failed");
|
|
|
|
/* Set EOA for following write call */
|
|
VRFY((H5FDset_eoa(file_ptr, H5FD_MEM_DEFAULT, file_end_addr + nbytes) >= 0),
|
|
"H5FDset_eoa succeeded");
|
|
|
|
/*
|
|
* Write another round of "number of IOCs" X "stripe size"
|
|
* bytes to the file using vector I/O and ensure we have
|
|
* "number of IOCs" subfiles, each with a size of at least
|
|
* 2 * "stripe size" bytes. The first (few) subfile(s) may
|
|
* be a bit larger due to file metadata.
|
|
*/
|
|
H5FD_mem_t write_type = H5FD_MEM_DRAW;
|
|
write_addr = file_end_addr;
|
|
write_status =
|
|
H5FDwrite_vector(file_ptr, dxpl_id, 1, &write_type, &write_addr, &nbytes, &c_write_buf);
|
|
VRFY((write_status >= 0), "H5FDwrite_vector succeeded");
|
|
|
|
VRFY((H5FDtruncate(file_ptr, dxpl_id, 0) >= 0), "H5FDtruncate succeeded");
|
|
|
|
for (int j = 0; j < num_subfiles; j++) {
|
|
h5_stat_size_t subfile_size;
|
|
h5_stat_t subfile_info;
|
|
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, j + 1, num_subfiles);
|
|
|
|
/* Ensure file exists */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr, "HDfopen on subfile succeeded");
|
|
VRFY((HDfclose(subfile_ptr) >= 0), "HDfclose on subfile succeeded");
|
|
|
|
/* Check file size */
|
|
VRFY((HDstat(tmp_filename, &subfile_info) >= 0), "HDstat succeeded");
|
|
subfile_size = (h5_stat_size_t)subfile_info.st_size;
|
|
|
|
VRFY((subfile_size >= 2 * cfg.stripe_size), "File size verification succeeded");
|
|
}
|
|
|
|
/* Verify that there aren't too many subfiles */
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, num_subfiles + 1, num_subfiles);
|
|
|
|
/* Ensure file doesn't exist */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr == NULL, "HDfopen on subfile correctly failed");
|
|
|
|
HDfree(write_buf);
|
|
write_buf = NULL;
|
|
|
|
VRFY((H5FDclose(file_ptr) >= 0), "H5FDclose succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
/* Next, try I/O with all ranks */
|
|
|
|
cfg.stripe_count = num_iocs_g;
|
|
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, TRUE, &cfg, H5FD_IOC_DEFAULT_THREAD_POOL_SIZE);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
/* Create and close file with H5Fcreate to setup superblock */
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "H5Fclose succeeded");
|
|
|
|
/* Re-open file through H5FDopen for direct writes */
|
|
file_ptr = H5FDopen(SUBF_FILENAME, H5F_ACC_RDWR, fapl_id, HADDR_UNDEF);
|
|
VRFY(file_ptr, "H5FDopen succeeded");
|
|
|
|
num_subfiles = num_iocs_g;
|
|
num_digits = (int)(HDlog10(num_subfiles) + 1);
|
|
|
|
nbytes = (size_t)(cfg.stripe_size * num_subfiles);
|
|
|
|
write_buf = HDmalloc(nbytes);
|
|
VRFY(write_buf, "HDmalloc succeeded");
|
|
|
|
HDmemset(write_buf, 255, nbytes);
|
|
|
|
c_write_buf = write_buf;
|
|
|
|
/*
|
|
* Get the current file size to see where we can safely
|
|
* write to in the file without overwriting the superblock
|
|
*/
|
|
if (MAINPROCESS) {
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
file_size = (h5_stat_size_t)file_info.st_size;
|
|
|
|
H5_CHECK_OVERFLOW(file_size, h5_stat_size_t, int64_t);
|
|
file_size64 = (int64_t)file_size;
|
|
}
|
|
|
|
if (mpi_size > 1) {
|
|
mpi_code_g = MPI_Bcast(&file_size64, 1, MPI_INT64_T, 0, comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Bcast succeeded");
|
|
}
|
|
|
|
H5_CHECK_OVERFLOW(file_size64, int64_t, haddr_t);
|
|
file_end_addr = (haddr_t)file_size64;
|
|
|
|
/* Set independent I/O on DXPL */
|
|
VRFY((H5Pset_dxpl_mpio(dxpl_id, H5FD_MPIO_INDEPENDENT) >= 0), "H5Pset_dxpl_mpio succeeded");
|
|
|
|
/* Set EOA for following write call */
|
|
VRFY((H5FDset_eoa(file_ptr, H5FD_MEM_DEFAULT, file_end_addr + ((size_t)mpi_size * nbytes)) >= 0),
|
|
"H5FDset_eoa succeeded");
|
|
|
|
/*
|
|
* Write "number of IOCs" X "stripe size" bytes to the file
|
|
* from each rank and ensure that we have "number of IOCs"
|
|
* subfiles, each with a size of at least "mpi size" * "stripe size"
|
|
* bytes. The first (few) subfile(s) may be a bit larger
|
|
* due to file metadata.
|
|
*/
|
|
write_addr = file_end_addr + ((size_t)mpi_rank * nbytes);
|
|
write_status = H5FDwrite(file_ptr, H5FD_MEM_DRAW, dxpl_id, write_addr, nbytes, c_write_buf);
|
|
VRFY((write_status >= 0), "H5FDwrite succeeded");
|
|
|
|
file_end_addr += ((size_t)mpi_size * nbytes);
|
|
|
|
VRFY((H5FDtruncate(file_ptr, dxpl_id, 0) >= 0), "H5FDtruncate succeeded");
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
if (MAINPROCESS) {
|
|
FILE *subfile_ptr;
|
|
|
|
for (int j = 0; j < num_subfiles; j++) {
|
|
h5_stat_size_t subfile_size;
|
|
h5_stat_t subfile_info;
|
|
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, j + 1, num_subfiles);
|
|
|
|
/* Ensure file exists */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr, "HDfopen on subfile succeeded");
|
|
VRFY((HDfclose(subfile_ptr) >= 0), "HDfclose on subfile succeeded");
|
|
|
|
/* Check file size */
|
|
VRFY((HDstat(tmp_filename, &subfile_info) >= 0), "HDstat succeeded");
|
|
subfile_size = (h5_stat_size_t)subfile_info.st_size;
|
|
|
|
VRFY((subfile_size >= (mpi_size * cfg.stripe_size)), "File size verification succeeded");
|
|
}
|
|
|
|
/* Verify that there aren't too many subfiles */
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, num_subfiles + 1, num_subfiles);
|
|
|
|
/* Ensure file doesn't exist */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr == NULL, "HDfopen on subfile correctly failed");
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
/* Set EOA for following write call */
|
|
VRFY((H5FDset_eoa(file_ptr, H5FD_MEM_DEFAULT, file_end_addr + ((size_t)mpi_size * nbytes)) >= 0),
|
|
"H5FDset_eoa succeeded");
|
|
|
|
/*
|
|
* Write another round of "number of IOCs" X "stripe size"
|
|
* bytes to the file from each rank using vector I/O and
|
|
* ensure we have "number of IOCs" subfiles, each with a
|
|
* size of at least 2 * "mpi size" * "stripe size" bytes.
|
|
* The first (few) subfile(s) may be a bit larger due to
|
|
* file metadata.
|
|
*/
|
|
H5FD_mem_t write_type = H5FD_MEM_DRAW;
|
|
write_addr = file_end_addr + ((size_t)mpi_rank * nbytes);
|
|
write_status =
|
|
H5FDwrite_vector(file_ptr, dxpl_id, 1, &write_type, &write_addr, &nbytes, &c_write_buf);
|
|
VRFY((write_status >= 0), "H5FDwrite_vector succeeded");
|
|
|
|
VRFY((H5FDtruncate(file_ptr, dxpl_id, 0) >= 0), "H5FDtruncate succeeded");
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
if (MAINPROCESS) {
|
|
FILE *subfile_ptr;
|
|
|
|
for (int j = 0; j < num_subfiles; j++) {
|
|
h5_stat_size_t subfile_size;
|
|
h5_stat_t subfile_info;
|
|
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, j + 1, num_subfiles);
|
|
|
|
/* Ensure file exists */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr, "HDfopen on subfile succeeded");
|
|
VRFY((HDfclose(subfile_ptr) >= 0), "HDfclose on subfile succeeded");
|
|
|
|
/* Check file size */
|
|
VRFY((HDstat(tmp_filename, &subfile_info) >= 0), "HDstat succeeded");
|
|
subfile_size = (h5_stat_size_t)subfile_info.st_size;
|
|
|
|
VRFY((subfile_size >= (2 * mpi_size * cfg.stripe_size)), "File size verification succeeded");
|
|
}
|
|
|
|
/* Verify that there aren't too many subfiles */
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, num_subfiles + 1, num_subfiles);
|
|
|
|
/* Ensure file doesn't exist */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr == NULL, "HDfopen on subfile correctly failed");
|
|
}
|
|
|
|
VRFY((H5FDclose(file_ptr) >= 0), "H5FDclose succeeded");
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
HDfree(write_buf);
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
}
|
|
|
|
HDfree(tmp_filename);
|
|
|
|
VRFY((H5Pclose(dxpl_id) >= 0), "DXPL close succeeded");
|
|
|
|
CHECK_PASSED();
|
|
}
|
|
#undef SUBF_FILENAME
|
|
#undef SUBF_NITER
|
|
|
|
/*
|
|
* Test that opening a file with a different stripe
|
|
* size/count than was used when creating the file
|
|
* results in the original stripe size/count being
|
|
* used. As there is currently no API to check the
|
|
* exact values used, we rely on the assumption that
|
|
* using a different stripe size/count would result
|
|
* in data verification failures.
|
|
*/
|
|
#define SUBF_FILENAME "test_subfiling_read_different_stripe_sizes.h5"
|
|
#define SUBF_HDF5_TYPE H5T_NATIVE_INT
|
|
#define SUBF_C_TYPE int
|
|
static void
|
|
test_read_different_stripe_size(void)
|
|
{
|
|
H5FD_subfiling_params_t cfg;
|
|
hsize_t start[1];
|
|
hsize_t count[1];
|
|
hsize_t dset_dims[1];
|
|
size_t target_size;
|
|
hid_t file_id = H5I_INVALID_HID;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
hid_t dset_id = H5I_INVALID_HID;
|
|
hid_t fspace_id = H5I_INVALID_HID;
|
|
char *tmp_filename = NULL;
|
|
void *buf = NULL;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("file re-opening with different stripe size");
|
|
|
|
tmp_filename = HDmalloc(PATH_MAX);
|
|
VRFY(tmp_filename, "HDmalloc succeeded");
|
|
|
|
/* Use a 1MiB stripe size and a subfile for each IOC */
|
|
cfg.ioc_selection = SELECT_IOC_ONE_PER_NODE;
|
|
cfg.stripe_size = (stripe_size_g > 0) ? stripe_size_g : 1048576;
|
|
cfg.stripe_count = num_iocs_g;
|
|
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, TRUE, &cfg, H5FD_IOC_DEFAULT_THREAD_POOL_SIZE);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
target_size = (size_t)cfg.stripe_size;
|
|
|
|
/* Nudge stripe size to be multiple of C type size */
|
|
if ((target_size % sizeof(SUBF_C_TYPE)) != 0)
|
|
target_size += sizeof(SUBF_C_TYPE) - (target_size % sizeof(SUBF_C_TYPE));
|
|
|
|
target_size *= (size_t)mpi_size;
|
|
|
|
VRFY(((target_size % sizeof(SUBF_C_TYPE)) == 0), "target size check succeeded");
|
|
|
|
dset_dims[0] = (hsize_t)(target_size / sizeof(SUBF_C_TYPE));
|
|
|
|
fspace_id = H5Screate_simple(1, dset_dims, NULL);
|
|
VRFY((fspace_id >= 0), "H5Screate_simple succeeded");
|
|
|
|
dset_id = H5Dcreate2(file_id, "DSET", SUBF_HDF5_TYPE, fspace_id, H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset creation succeeded");
|
|
|
|
/* Select hyperslab */
|
|
count[0] = dset_dims[0] / (hsize_t)mpi_size;
|
|
start[0] = (hsize_t)mpi_rank * count[0];
|
|
VRFY((H5Sselect_hyperslab(fspace_id, H5S_SELECT_SET, start, NULL, count, NULL) >= 0),
|
|
"H5Sselect_hyperslab succeeded");
|
|
|
|
buf = HDmalloc(count[0] * sizeof(SUBF_C_TYPE));
|
|
VRFY(buf, "HDmalloc succeeded");
|
|
|
|
for (size_t i = 0; i < count[0]; i++)
|
|
((SUBF_C_TYPE *)buf)[i] = (SUBF_C_TYPE)((size_t)mpi_rank + i);
|
|
|
|
VRFY((H5Dwrite(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, fspace_id, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset write succeeded");
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Sclose(fspace_id) >= 0), "File dataspace close succeeded");
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
/* Ensure all the subfiles are present */
|
|
if (MAINPROCESS) {
|
|
h5_stat_t file_info;
|
|
FILE *subfile_ptr;
|
|
int num_subfiles = cfg.stripe_count;
|
|
int num_digits = (int)(HDlog10(num_subfiles) + 1);
|
|
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
|
|
for (int j = 0; j < num_subfiles; j++) {
|
|
h5_stat_size_t subfile_size;
|
|
h5_stat_t subfile_info;
|
|
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, j + 1, num_subfiles);
|
|
|
|
/* Ensure file exists */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr, "HDfopen on subfile succeeded");
|
|
VRFY((HDfclose(subfile_ptr) >= 0), "HDfclose on subfile succeeded");
|
|
|
|
/* Check file size */
|
|
VRFY((HDstat(tmp_filename, &subfile_info) >= 0), "HDstat succeeded");
|
|
subfile_size = (h5_stat_size_t)subfile_info.st_size;
|
|
|
|
VRFY((subfile_size >= cfg.stripe_size), "File size verification succeeded");
|
|
}
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
/* Add a bit to the stripe size and specify a few more subfiles */
|
|
cfg.stripe_size += (cfg.stripe_size / 2);
|
|
cfg.stripe_count *= 2;
|
|
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, TRUE, &cfg, H5FD_IOC_DEFAULT_THREAD_POOL_SIZE);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fopen(SUBF_FILENAME, H5F_ACC_RDONLY, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fopen succeeded");
|
|
|
|
dset_id = H5Dopen2(file_id, "DSET", H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset open succeeded");
|
|
|
|
fspace_id = H5Dget_space(dset_id);
|
|
VRFY((fspace_id >= 0), "Dataspace retrieval succeeded");
|
|
|
|
/* Select hyperslab */
|
|
count[0] = dset_dims[0] / (hsize_t)mpi_size;
|
|
start[0] = (hsize_t)mpi_rank * count[0];
|
|
VRFY((H5Sselect_hyperslab(fspace_id, H5S_SELECT_SET, start, NULL, count, NULL) >= 0),
|
|
"H5Sselect_hyperslab succeeded");
|
|
|
|
buf = HDcalloc(1, count[0] * sizeof(SUBF_C_TYPE));
|
|
VRFY(buf, "HDcalloc succeeded");
|
|
|
|
VRFY((H5Dread(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, fspace_id, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset read succeeded");
|
|
|
|
for (size_t i = 0; i < count[0]; i++) {
|
|
SUBF_C_TYPE buf_value = ((SUBF_C_TYPE *)buf)[i];
|
|
|
|
VRFY((buf_value == (SUBF_C_TYPE)((size_t)mpi_rank + i)), "data verification succeeded");
|
|
}
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Sclose(fspace_id) >= 0), "File dataspace close succeeded");
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
/* Ensure only the original subfiles are present */
|
|
if (MAINPROCESS) {
|
|
h5_stat_t file_info;
|
|
FILE *subfile_ptr;
|
|
int num_subfiles = cfg.stripe_count;
|
|
int num_digits = (int)(HDlog10(num_subfiles / 2) + 1);
|
|
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
|
|
for (int j = 0; j < num_subfiles; j++) {
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, j + 1, num_subfiles / 2);
|
|
|
|
if (j < (num_subfiles / 2)) {
|
|
/* Ensure file exists */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr, "HDfopen on subfile succeeded");
|
|
VRFY((HDfclose(subfile_ptr) >= 0), "HDfclose on subfile succeeded");
|
|
}
|
|
else {
|
|
/* Ensure file doesn't exist */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr == NULL, "HDfopen on subfile correctly failed");
|
|
}
|
|
}
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
HDfree(tmp_filename);
|
|
|
|
CHECK_PASSED();
|
|
}
|
|
#undef SUBF_FILENAME
|
|
#undef SUBF_HDF5_TYPE
|
|
#undef SUBF_C_TYPE
|
|
|
|
/*
|
|
* Test that everything works correctly when a file is
|
|
* pre-created on rank 0 with a specified target number
|
|
* of subfiles and then read back on all ranks.
|
|
*/
|
|
#define SUBF_FILENAME "test_subfiling_precreate_rank_0.h5"
|
|
#define SUBF_HDF5_TYPE H5T_NATIVE_INT
|
|
#define SUBF_C_TYPE int
|
|
static void
|
|
test_subfiling_precreate_rank_0(void)
|
|
{
|
|
hsize_t start[1];
|
|
hsize_t count[1];
|
|
hsize_t dset_dims[1];
|
|
size_t target_size;
|
|
size_t n_elements_per_rank;
|
|
hid_t file_id = H5I_INVALID_HID;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
hid_t dset_id = H5I_INVALID_HID;
|
|
hid_t fspace_id = H5I_INVALID_HID;
|
|
void *buf = NULL;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("file pre-creation on rank 0");
|
|
|
|
/* Calculate target size for dataset to stripe it across available IOCs */
|
|
target_size = (stripe_size_g > 0) ? (size_t)stripe_size_g : H5FD_SUBFILING_DEFAULT_STRIPE_SIZE;
|
|
|
|
/* Nudge stripe size to be multiple of C type size */
|
|
if ((target_size % sizeof(SUBF_C_TYPE)) != 0)
|
|
target_size += sizeof(SUBF_C_TYPE) - (target_size % sizeof(SUBF_C_TYPE));
|
|
|
|
target_size *= (size_t)mpi_size;
|
|
|
|
VRFY(((target_size % sizeof(SUBF_C_TYPE)) == 0), "target size check succeeded");
|
|
|
|
if (stripe_size_g > 0) {
|
|
VRFY((target_size >= (size_t)stripe_size_g), "target size check succeeded");
|
|
}
|
|
else {
|
|
VRFY((target_size >= H5FD_SUBFILING_DEFAULT_STRIPE_SIZE), "target size check succeeded");
|
|
}
|
|
|
|
dset_dims[0] = (hsize_t)(target_size / sizeof(SUBF_C_TYPE));
|
|
n_elements_per_rank = (dset_dims[0] / (size_t)mpi_size);
|
|
|
|
/* Create and populate file on rank 0 only */
|
|
if (MAINPROCESS) {
|
|
H5FD_subfiling_params_t cfg;
|
|
h5_stat_size_t file_size;
|
|
h5_stat_t file_info;
|
|
FILE *subfile_ptr;
|
|
char *tmp_filename = NULL;
|
|
int num_subfiles;
|
|
int num_digits;
|
|
|
|
/* Create a file consisting of 1 subfile per application I/O concentrator */
|
|
cfg.ioc_selection = SELECT_IOC_ONE_PER_NODE;
|
|
cfg.stripe_size = (stripe_size_g > 0) ? stripe_size_g : H5FD_SUBFILING_DEFAULT_STRIPE_SIZE;
|
|
cfg.stripe_count = num_iocs_g;
|
|
|
|
fapl_id = create_subfiling_ioc_fapl(MPI_COMM_SELF, MPI_INFO_NULL, TRUE, &cfg,
|
|
H5FD_IOC_DEFAULT_THREAD_POOL_SIZE);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
|
|
fspace_id = H5Screate_simple(1, dset_dims, NULL);
|
|
VRFY((fspace_id >= 0), "H5Screate_simple succeeded");
|
|
|
|
dset_id =
|
|
H5Dcreate2(file_id, "DSET", SUBF_HDF5_TYPE, fspace_id, H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset creation succeeded");
|
|
|
|
buf = HDmalloc(dset_dims[0] * sizeof(SUBF_C_TYPE));
|
|
VRFY(buf, "HDmalloc succeeded");
|
|
|
|
for (size_t i = 0; i < dset_dims[0]; i++)
|
|
((SUBF_C_TYPE *)buf)[i] = (SUBF_C_TYPE)((i / n_elements_per_rank) + (i % n_elements_per_rank));
|
|
|
|
VRFY((H5Dwrite(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, fspace_id, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset write succeeded");
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Sclose(fspace_id) >= 0), "File dataspace close succeeded");
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "H5Fclose succeeded");
|
|
|
|
/*
|
|
* Ensure that all the subfiles are present
|
|
*/
|
|
|
|
num_subfiles = cfg.stripe_count;
|
|
num_digits = (int)(HDlog10(num_subfiles) + 1);
|
|
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
|
|
tmp_filename = HDmalloc(PATH_MAX);
|
|
VRFY(tmp_filename, "HDmalloc succeeded");
|
|
|
|
for (int i = 0; i < num_subfiles; i++) {
|
|
h5_stat_t subfile_info;
|
|
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, i + 1, num_subfiles);
|
|
|
|
/* Ensure file exists */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr, "HDfopen on subfile succeeded");
|
|
VRFY((HDfclose(subfile_ptr) >= 0), "HDfclose on subfile succeeded");
|
|
|
|
/* Check file size */
|
|
VRFY((HDstat(tmp_filename, &subfile_info) >= 0), "HDstat succeeded");
|
|
file_size = (h5_stat_size_t)subfile_info.st_size;
|
|
|
|
VRFY((file_size >= cfg.stripe_size), "File size verification succeeded");
|
|
}
|
|
|
|
/* Verify that there aren't too many subfiles */
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino, num_digits, num_subfiles + 1, num_subfiles);
|
|
|
|
/* Ensure file doesn't exist */
|
|
subfile_ptr = HDfopen(tmp_filename, "r");
|
|
VRFY(subfile_ptr == NULL, "HDfopen on subfile correctly failed");
|
|
|
|
HDfree(tmp_filename);
|
|
tmp_filename = NULL;
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
/* Open the file on all ranks */
|
|
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, FALSE, NULL, 0);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fopen(SUBF_FILENAME, H5F_ACC_RDONLY, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fopen succeeded");
|
|
|
|
dset_id = H5Dopen2(file_id, "DSET", H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset open succeeded");
|
|
|
|
fspace_id = H5Dget_space(dset_id);
|
|
VRFY((fspace_id >= 0), "Dataset dataspace retrieval succeeded");
|
|
|
|
/* Select hyperslab */
|
|
count[0] = n_elements_per_rank;
|
|
start[0] = (hsize_t)mpi_rank * count[0];
|
|
VRFY((H5Sselect_hyperslab(fspace_id, H5S_SELECT_SET, start, NULL, count, NULL) >= 0),
|
|
"H5Sselect_hyperslab succeeded");
|
|
|
|
buf = HDcalloc(1, count[0] * sizeof(SUBF_C_TYPE));
|
|
VRFY(buf, "HDcalloc succeeded");
|
|
|
|
VRFY((H5Dread(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, fspace_id, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset read succeeded");
|
|
|
|
for (size_t i = 0; i < n_elements_per_rank; i++) {
|
|
SUBF_C_TYPE buf_value = ((SUBF_C_TYPE *)buf)[i];
|
|
|
|
VRFY((buf_value == (SUBF_C_TYPE)((size_t)mpi_rank + i)), "data verification succeeded");
|
|
}
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Sclose(fspace_id) >= 0), "File dataspace close succeeded");
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "H5Fclose succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
CHECK_PASSED();
|
|
}
|
|
#undef SUBF_FILENAME
|
|
#undef SUBF_HDF5_TYPE
|
|
#undef SUBF_C_TYPE
|
|
|
|
/*
|
|
* Test to check that an HDF5 file created with the
|
|
* Subfiling VFD can be read back with a single MPI
|
|
* rank
|
|
*/
|
|
#define SUBF_FILENAME "test_subfiling_write_many_read_one.h5"
|
|
#define SUBF_HDF5_TYPE H5T_NATIVE_INT
|
|
#define SUBF_C_TYPE int
|
|
static void
|
|
test_subfiling_write_many_read_one(void)
|
|
{
|
|
hsize_t start[1];
|
|
hsize_t count[1];
|
|
hsize_t dset_dims[1];
|
|
size_t target_size;
|
|
hid_t file_id = H5I_INVALID_HID;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
hid_t dset_id = H5I_INVALID_HID;
|
|
hid_t fspace_id = H5I_INVALID_HID;
|
|
void *buf = NULL;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("reading back file with single MPI rank");
|
|
|
|
/* Get a default Subfiling FAPL */
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, FALSE, NULL, 0);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
/* Create file on all ranks */
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
/* Calculate target size for dataset to stripe it across available IOCs */
|
|
target_size = (stripe_size_g > 0) ? (size_t)stripe_size_g : H5FD_SUBFILING_DEFAULT_STRIPE_SIZE;
|
|
|
|
/* Nudge stripe size to be multiple of C type size */
|
|
if ((target_size % sizeof(SUBF_C_TYPE)) != 0)
|
|
target_size += sizeof(SUBF_C_TYPE) - (target_size % sizeof(SUBF_C_TYPE));
|
|
|
|
target_size *= (size_t)mpi_size;
|
|
|
|
VRFY(((target_size % sizeof(SUBF_C_TYPE)) == 0), "target size check succeeded");
|
|
|
|
if (stripe_size_g > 0) {
|
|
VRFY((target_size >= (size_t)stripe_size_g), "target size check succeeded");
|
|
}
|
|
else {
|
|
VRFY((target_size >= H5FD_SUBFILING_DEFAULT_STRIPE_SIZE), "target size check succeeded");
|
|
}
|
|
|
|
dset_dims[0] = (hsize_t)(target_size / sizeof(SUBF_C_TYPE));
|
|
|
|
fspace_id = H5Screate_simple(1, dset_dims, NULL);
|
|
VRFY((fspace_id >= 0), "H5Screate_simple succeeded");
|
|
|
|
dset_id = H5Dcreate2(file_id, "DSET", SUBF_HDF5_TYPE, fspace_id, H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset creation succeeded");
|
|
|
|
/* Select hyperslab */
|
|
count[0] = dset_dims[0] / (hsize_t)mpi_size;
|
|
start[0] = (hsize_t)mpi_rank * count[0];
|
|
VRFY((H5Sselect_hyperslab(fspace_id, H5S_SELECT_SET, start, NULL, count, NULL) >= 0),
|
|
"H5Sselect_hyperslab succeeded");
|
|
|
|
buf = HDmalloc(count[0] * sizeof(SUBF_C_TYPE));
|
|
VRFY(buf, "HDmalloc succeeded");
|
|
|
|
for (size_t i = 0; i < count[0]; i++)
|
|
((SUBF_C_TYPE *)buf)[i] = (SUBF_C_TYPE)((size_t)mpi_rank + i);
|
|
|
|
VRFY((H5Dwrite(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, fspace_id, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset write succeeded");
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
if (MAINPROCESS) {
|
|
fapl_id = create_subfiling_ioc_fapl(MPI_COMM_SELF, MPI_INFO_NULL, FALSE, NULL, 0);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fopen(SUBF_FILENAME, H5F_ACC_RDONLY, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fopen succeeded");
|
|
|
|
dset_id = H5Dopen2(file_id, "DSET", H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset open succeeded");
|
|
|
|
buf = HDcalloc(1, target_size);
|
|
VRFY(buf, "HDcalloc succeeded");
|
|
|
|
VRFY((H5Dread(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, H5S_ALL, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset read succeeded");
|
|
|
|
for (size_t i = 0; i < (size_t)mpi_size; i++) {
|
|
for (size_t j = 0; j < count[0]; j++) {
|
|
SUBF_C_TYPE buf_value = ((SUBF_C_TYPE *)buf)[(i * count[0]) + j];
|
|
|
|
VRFY((buf_value == (SUBF_C_TYPE)(j + i)), "data verification succeeded");
|
|
}
|
|
}
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
VRFY((H5Sclose(fspace_id) >= 0), "File dataspace close succeeded");
|
|
|
|
CHECK_PASSED();
|
|
}
|
|
#undef SUBF_FILENAME
|
|
#undef SUBF_HDF5_TYPE
|
|
#undef SUBF_C_TYPE
|
|
|
|
/*
|
|
* Test to check that an HDF5 file created with the
|
|
* Subfiling VFD can be read back with less MPI ranks
|
|
* than the file was written with
|
|
*/
|
|
#define SUBF_FILENAME "test_subfiling_write_many_read_few.h5"
|
|
#define SUBF_HDF5_TYPE H5T_NATIVE_INT
|
|
#define SUBF_C_TYPE int
|
|
static void
|
|
test_subfiling_write_many_read_few(void)
|
|
{
|
|
MPI_Comm sub_comm = MPI_COMM_NULL;
|
|
hsize_t start[1];
|
|
hsize_t count[1];
|
|
hsize_t dset_dims[1];
|
|
hbool_t reading_file = FALSE;
|
|
size_t target_size;
|
|
hid_t file_id = H5I_INVALID_HID;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
hid_t dset_id = H5I_INVALID_HID;
|
|
hid_t fspace_id = H5I_INVALID_HID;
|
|
void *buf = NULL;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("reading back file with fewer MPI ranks than written with");
|
|
|
|
/*
|
|
* Skip this test for an MPI communicator size of 1,
|
|
* as the test wouldn't really be meaningful
|
|
*/
|
|
if (mpi_size == 1) {
|
|
if (MAINPROCESS)
|
|
SKIPPED();
|
|
return;
|
|
}
|
|
|
|
/* Get a default Subfiling FAPL */
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, FALSE, NULL, 0);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
/* Create file on all ranks */
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
/* Calculate target size for dataset to stripe it across available IOCs */
|
|
target_size = (stripe_size_g > 0) ? (size_t)stripe_size_g : H5FD_SUBFILING_DEFAULT_STRIPE_SIZE;
|
|
|
|
/* Nudge stripe size to be multiple of C type size */
|
|
if ((target_size % sizeof(SUBF_C_TYPE)) != 0)
|
|
target_size += sizeof(SUBF_C_TYPE) - (target_size % sizeof(SUBF_C_TYPE));
|
|
|
|
target_size *= (size_t)mpi_size;
|
|
|
|
VRFY(((target_size % sizeof(SUBF_C_TYPE)) == 0), "target size check succeeded");
|
|
|
|
if (stripe_size_g > 0) {
|
|
VRFY((target_size >= (size_t)stripe_size_g), "target size check succeeded");
|
|
}
|
|
else {
|
|
VRFY((target_size >= H5FD_SUBFILING_DEFAULT_STRIPE_SIZE), "target size check succeeded");
|
|
}
|
|
|
|
dset_dims[0] = (hsize_t)(target_size / sizeof(SUBF_C_TYPE));
|
|
|
|
fspace_id = H5Screate_simple(1, dset_dims, NULL);
|
|
VRFY((fspace_id >= 0), "H5Screate_simple succeeded");
|
|
|
|
dset_id = H5Dcreate2(file_id, "DSET", SUBF_HDF5_TYPE, fspace_id, H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset creation succeeded");
|
|
|
|
/* Select hyperslab */
|
|
count[0] = dset_dims[0] / (hsize_t)mpi_size;
|
|
start[0] = (hsize_t)mpi_rank * count[0];
|
|
VRFY((H5Sselect_hyperslab(fspace_id, H5S_SELECT_SET, start, NULL, count, NULL) >= 0),
|
|
"H5Sselect_hyperslab succeeded");
|
|
|
|
buf = HDmalloc(count[0] * sizeof(SUBF_C_TYPE));
|
|
VRFY(buf, "HDmalloc succeeded");
|
|
|
|
for (size_t i = 0; i < count[0]; i++)
|
|
((SUBF_C_TYPE *)buf)[i] = (SUBF_C_TYPE)((size_t)mpi_rank + i);
|
|
|
|
VRFY((H5Dwrite(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, fspace_id, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset write succeeded");
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
/*
|
|
* If only using 1 node, read file back with a
|
|
* few ranks from that node. Otherwise, read file
|
|
* back with 1 MPI rank per node
|
|
*/
|
|
if (num_nodes_g == 1) {
|
|
int color;
|
|
|
|
if (mpi_size < 2) {
|
|
color = 1;
|
|
}
|
|
else if (mpi_size < 4) {
|
|
color = (mpi_rank < (mpi_size / 2));
|
|
}
|
|
else {
|
|
color = (mpi_rank < (mpi_size / 4));
|
|
}
|
|
|
|
if (mpi_size > 1) {
|
|
mpi_code_g = MPI_Comm_split(comm_g, color, mpi_rank, &sub_comm);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Comm_split succeeded");
|
|
}
|
|
|
|
if (color)
|
|
reading_file = TRUE;
|
|
}
|
|
else {
|
|
if (node_local_rank == 0) {
|
|
sub_comm = ioc_comm;
|
|
reading_file = TRUE;
|
|
}
|
|
}
|
|
|
|
if (reading_file) {
|
|
fapl_id = create_subfiling_ioc_fapl(sub_comm, MPI_INFO_NULL, FALSE, NULL, 0);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
file_id = H5Fopen(SUBF_FILENAME, H5F_ACC_RDONLY, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fopen succeeded");
|
|
|
|
dset_id = H5Dopen2(file_id, "DSET", H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset open succeeded");
|
|
|
|
buf = HDcalloc(1, target_size);
|
|
VRFY(buf, "HDcalloc succeeded");
|
|
|
|
VRFY((H5Dread(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, H5S_ALL, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset read succeeded");
|
|
|
|
for (size_t i = 0; i < (size_t)mpi_size; i++) {
|
|
for (size_t j = 0; j < count[0]; j++) {
|
|
SUBF_C_TYPE buf_value = ((SUBF_C_TYPE *)buf)[(i * count[0]) + j];
|
|
|
|
VRFY((buf_value == (SUBF_C_TYPE)(j + i)), "data verification succeeded");
|
|
}
|
|
}
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
}
|
|
|
|
if ((sub_comm != MPI_COMM_NULL) && (num_nodes_g == 1)) {
|
|
mpi_code_g = MPI_Comm_free(&sub_comm);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Comm_free succeeded");
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
VRFY((H5Sclose(fspace_id) >= 0), "File dataspace close succeeded");
|
|
|
|
CHECK_PASSED();
|
|
}
|
|
#undef SUBF_FILENAME
|
|
#undef SUBF_HDF5_TYPE
|
|
#undef SUBF_C_TYPE
|
|
|
|
/*
|
|
* Test that the subfiling file can be read with the
|
|
* sec2 driver after being fused back together with
|
|
* the h5fuse utility
|
|
*/
|
|
#define SUBF_FILENAME "test_subfiling_h5fuse.h5"
|
|
#define SUBF_HDF5_TYPE H5T_NATIVE_INT
|
|
#define SUBF_C_TYPE int
|
|
static void
|
|
test_subfiling_h5fuse(void)
|
|
{
|
|
hsize_t start[1];
|
|
hsize_t count[1];
|
|
hsize_t dset_dims[1];
|
|
size_t target_size;
|
|
hid_t file_id = H5I_INVALID_HID;
|
|
hid_t fapl_id = H5I_INVALID_HID;
|
|
hid_t dset_id = H5I_INVALID_HID;
|
|
hid_t fspace_id = H5I_INVALID_HID;
|
|
void *buf = NULL;
|
|
int skip_test = 0;
|
|
|
|
curr_nerrors = nerrors;
|
|
|
|
if (MAINPROCESS)
|
|
TESTING_2("h5fuse utility");
|
|
|
|
#if defined(H5_HAVE_FORK) && defined(H5_HAVE_WAITPID)
|
|
|
|
/*
|
|
* Check if h5fuse script exists in current directory;
|
|
* Skip test if it doesn't
|
|
*/
|
|
if (MAINPROCESS) {
|
|
FILE *h5fuse_script;
|
|
|
|
h5fuse_script = HDfopen("h5fuse.sh", "r");
|
|
if (h5fuse_script)
|
|
HDfclose(h5fuse_script);
|
|
else
|
|
skip_test = 1;
|
|
}
|
|
|
|
if (mpi_size > 1) {
|
|
mpi_code_g = MPI_Bcast(&skip_test, 1, MPI_INT, 0, comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Bcast succeeded");
|
|
}
|
|
|
|
if (skip_test) {
|
|
if (MAINPROCESS)
|
|
SKIPPED();
|
|
return;
|
|
}
|
|
|
|
/* Get a default Subfiling FAPL */
|
|
fapl_id = create_subfiling_ioc_fapl(comm_g, info_g, FALSE, NULL, 0);
|
|
VRFY((fapl_id >= 0), "FAPL creation succeeded");
|
|
|
|
/* Create file on all ranks */
|
|
file_id = H5Fcreate(SUBF_FILENAME, H5F_ACC_TRUNC, H5P_DEFAULT, fapl_id);
|
|
VRFY((file_id >= 0), "H5Fcreate succeeded");
|
|
|
|
/* Calculate target size for dataset to stripe it across available IOCs */
|
|
target_size = (stripe_size_g > 0) ? (size_t)stripe_size_g : H5FD_SUBFILING_DEFAULT_STRIPE_SIZE;
|
|
|
|
/* Nudge stripe size to be multiple of C type size */
|
|
if ((target_size % sizeof(SUBF_C_TYPE)) != 0)
|
|
target_size += sizeof(SUBF_C_TYPE) - (target_size % sizeof(SUBF_C_TYPE));
|
|
|
|
target_size *= (size_t)mpi_size;
|
|
|
|
VRFY(((target_size % sizeof(SUBF_C_TYPE)) == 0), "target size check succeeded");
|
|
|
|
if (stripe_size_g > 0) {
|
|
VRFY((target_size >= (size_t)stripe_size_g), "target size check succeeded");
|
|
}
|
|
else {
|
|
VRFY((target_size >= H5FD_SUBFILING_DEFAULT_STRIPE_SIZE), "target size check succeeded");
|
|
}
|
|
|
|
dset_dims[0] = (hsize_t)(target_size / sizeof(SUBF_C_TYPE));
|
|
|
|
fspace_id = H5Screate_simple(1, dset_dims, NULL);
|
|
VRFY((fspace_id >= 0), "H5Screate_simple succeeded");
|
|
|
|
dset_id = H5Dcreate2(file_id, "DSET", SUBF_HDF5_TYPE, fspace_id, H5P_DEFAULT, H5P_DEFAULT, H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset creation succeeded");
|
|
|
|
/* Select hyperslab */
|
|
count[0] = dset_dims[0] / (hsize_t)mpi_size;
|
|
start[0] = (hsize_t)mpi_rank * count[0];
|
|
VRFY((H5Sselect_hyperslab(fspace_id, H5S_SELECT_SET, start, NULL, count, NULL) >= 0),
|
|
"H5Sselect_hyperslab succeeded");
|
|
|
|
buf = HDmalloc(count[0] * sizeof(SUBF_C_TYPE));
|
|
VRFY(buf, "HDmalloc succeeded");
|
|
|
|
for (size_t i = 0; i < count[0]; i++)
|
|
((SUBF_C_TYPE *)buf)[i] = (SUBF_C_TYPE)((size_t)mpi_rank + i);
|
|
|
|
VRFY((H5Dwrite(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, fspace_id, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset write succeeded");
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Sclose(fspace_id) >= 0), "File dataspace close succeeded");
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
|
|
if (MAINPROCESS) {
|
|
h5_stat_t file_info;
|
|
pid_t pid = 0;
|
|
pid_t tmppid;
|
|
int status;
|
|
|
|
pid = HDfork();
|
|
VRFY(pid >= 0, "HDfork succeeded");
|
|
|
|
if (pid == 0) {
|
|
char *tmp_filename;
|
|
char *args[6];
|
|
|
|
tmp_filename = HDmalloc(PATH_MAX);
|
|
VRFY(tmp_filename, "HDmalloc succeeded");
|
|
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
|
|
/* Generate name for configuration file */
|
|
HDsnprintf(tmp_filename, PATH_MAX, H5FD_SUBFILING_CONFIG_FILENAME_TEMPLATE, SUBF_FILENAME,
|
|
(uint64_t)file_info.st_ino);
|
|
|
|
args[0] = HDstrdup("env");
|
|
args[1] = HDstrdup("sh");
|
|
args[2] = HDstrdup("h5fuse.sh");
|
|
args[3] = HDstrdup("-f");
|
|
args[4] = tmp_filename;
|
|
args[5] = NULL;
|
|
|
|
/* Call h5fuse script from MPI rank 0 */
|
|
HDexecvp("env", args);
|
|
}
|
|
else {
|
|
tmppid = HDwaitpid(pid, &status, 0);
|
|
VRFY(tmppid >= 0, "HDwaitpid succeeded");
|
|
|
|
if (WIFEXITED(status)) {
|
|
int ret;
|
|
|
|
if ((ret = WEXITSTATUS(status)) != 0) {
|
|
HDprintf("h5fuse process exited with error code %d\n", ret);
|
|
HDfflush(stdout);
|
|
MPI_Abort(comm_g, -1);
|
|
}
|
|
}
|
|
else {
|
|
HDprintf("h5fuse process terminated abnormally\n");
|
|
HDfflush(stdout);
|
|
MPI_Abort(comm_g, -1);
|
|
}
|
|
}
|
|
|
|
/* Verify the size of the fused file */
|
|
VRFY((HDstat(SUBF_FILENAME, &file_info) >= 0), "HDstat succeeded");
|
|
VRFY(((size_t)file_info.st_size >= target_size), "File size verification succeeded");
|
|
|
|
/* Re-open file with sec2 driver and verify the data */
|
|
file_id = H5Fopen(SUBF_FILENAME, H5F_ACC_RDONLY, H5P_DEFAULT);
|
|
VRFY((file_id >= 0), "H5Fopen succeeded");
|
|
|
|
dset_id = H5Dopen2(file_id, "DSET", H5P_DEFAULT);
|
|
VRFY((dset_id >= 0), "Dataset open succeeded");
|
|
|
|
buf = HDcalloc(1, target_size);
|
|
VRFY(buf, "HDcalloc succeeded");
|
|
|
|
VRFY((H5Dread(dset_id, SUBF_HDF5_TYPE, H5S_BLOCK, H5S_ALL, H5P_DEFAULT, buf) >= 0),
|
|
"Dataset read succeeded");
|
|
|
|
for (size_t i = 0; i < (size_t)mpi_size; i++) {
|
|
for (size_t j = 0; j < count[0]; j++) {
|
|
SUBF_C_TYPE buf_value = ((SUBF_C_TYPE *)buf)[(i * count[0]) + j];
|
|
|
|
VRFY((buf_value == (SUBF_C_TYPE)(j + i)), "data verification succeeded");
|
|
}
|
|
}
|
|
|
|
HDfree(buf);
|
|
buf = NULL;
|
|
|
|
VRFY((H5Dclose(dset_id) >= 0), "Dataset close succeeded");
|
|
VRFY((H5Fclose(file_id) >= 0), "File close succeeded");
|
|
}
|
|
|
|
mpi_code_g = MPI_Barrier(comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Barrier succeeded");
|
|
|
|
H5E_BEGIN_TRY
|
|
{
|
|
H5Fdelete(SUBF_FILENAME, fapl_id);
|
|
}
|
|
H5E_END_TRY;
|
|
|
|
VRFY((H5Pclose(fapl_id) >= 0), "FAPL close succeeded");
|
|
|
|
CHECK_PASSED();
|
|
#else
|
|
SKIPPED();
|
|
#endif
|
|
}
|
|
#undef SUBF_FILENAME
|
|
#undef SUBF_HDF5_TYPE
|
|
#undef SUBF_C_TYPE
|
|
|
|
static void
|
|
parse_subfiling_env_vars(void)
|
|
{
|
|
char *env_value;
|
|
|
|
if (NULL != (env_value = HDgetenv(H5FD_SUBFILING_STRIPE_SIZE))) {
|
|
stripe_size_g = HDstrtoll(env_value, NULL, 0);
|
|
if ((ERANGE == errno) || (stripe_size_g <= 0))
|
|
stripe_size_g = -1;
|
|
}
|
|
|
|
if (NULL != (env_value = HDgetenv(H5FD_SUBFILING_IOC_PER_NODE))) {
|
|
ioc_per_node_g = HDstrtol(env_value, NULL, 0);
|
|
if ((ERANGE == errno) || (ioc_per_node_g <= 0))
|
|
ioc_per_node_g = -1;
|
|
else if (ioc_per_node_g * num_nodes_g > mpi_size)
|
|
/*
|
|
* If the number of IOCs per node from the environment
|
|
* causes the total number of IOCs to exceed the number
|
|
* of MPI ranks available, the Subfiling VFD will simply
|
|
* use all of the MPI ranks on a node as IOCs
|
|
*/
|
|
ioc_per_node_g = node_local_size;
|
|
}
|
|
|
|
if (NULL != (env_value = HDgetenv(H5FD_IOC_THREAD_POOL_SIZE))) {
|
|
ioc_thread_pool_size_g = HDatoi(env_value);
|
|
if (ioc_thread_pool_size_g <= 0)
|
|
ioc_thread_pool_size_g = -1;
|
|
}
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
unsigned seed;
|
|
int required = MPI_THREAD_MULTIPLE;
|
|
int provided = 0;
|
|
|
|
HDcompile_assert(SUBFILING_MIN_STRIPE_SIZE <= H5FD_SUBFILING_DEFAULT_STRIPE_SIZE);
|
|
|
|
/* Initialize MPI */
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Init_thread(&argc, &argv, required, &provided))) {
|
|
HDprintf("MPI_Init_thread failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_rank(comm_g, &mpi_rank))) {
|
|
HDprintf("MPI_Comm_rank failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
|
|
if (provided != required) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI doesn't support MPI_Init_thread with MPI_THREAD_MULTIPLE\n");
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_size(comm_g, &mpi_size))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Comm_size failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
|
|
/* Split communicator according to node-local ranks */
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_split_type(comm_g, MPI_COMM_TYPE_SHARED, mpi_rank,
|
|
MPI_INFO_NULL, &node_local_comm))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Comm_split_type failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_size(node_local_comm, &node_local_size))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Comm_size failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_rank(node_local_comm, &node_local_rank))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Comm_rank failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
|
|
/* Get the number of nodes being run on */
|
|
num_nodes_g = (node_local_rank == 0) ? 1 : 0;
|
|
if (MPI_SUCCESS !=
|
|
(mpi_code_g = MPI_Allreduce(MPI_IN_PLACE, &num_nodes_g, 1, MPI_INT, MPI_SUM, comm_g))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Allreduce failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
|
|
/*
|
|
* Split communicator according to rank value across nodes.
|
|
* If the SELECT_IOC_ONE_PER_NODE IOC selection strategy is
|
|
* used, each rank with a node local rank value of 0 will
|
|
* be an IOC in the new communicator.
|
|
*/
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_split(comm_g, node_local_rank, mpi_rank, &ioc_comm))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Comm_split failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_size(ioc_comm, &ioc_comm_size))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Comm_size failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Comm_rank(ioc_comm, &ioc_comm_rank))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Comm_rank failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
|
|
if (H5dont_atexit() < 0) {
|
|
if (MAINPROCESS)
|
|
HDprintf("Failed to turn off atexit processing. Continue.\n");
|
|
}
|
|
|
|
H5open();
|
|
|
|
/* Enable selection I/O using internal temporary workaround */
|
|
H5_use_selection_io_g = TRUE;
|
|
|
|
if (MAINPROCESS) {
|
|
HDprintf("Testing Subfiling VFD functionality\n");
|
|
}
|
|
|
|
TestAlarmOn();
|
|
|
|
/*
|
|
* Obtain and broadcast seed value since ranks
|
|
* aren't guaranteed to arrive here at exactly
|
|
* the same time and could end up out of sync
|
|
* with each other in regards to random number
|
|
* generation
|
|
*/
|
|
if (mpi_rank == 0)
|
|
seed = (unsigned)time(NULL);
|
|
|
|
if (mpi_size > 1) {
|
|
if (MPI_SUCCESS != (mpi_code_g = MPI_Bcast(&seed, 1, MPI_UNSIGNED, 0, comm_g))) {
|
|
if (MAINPROCESS)
|
|
HDprintf("MPI_Bcast failed with error code %d\n", mpi_code_g);
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
}
|
|
|
|
srand(seed);
|
|
|
|
if (MAINPROCESS)
|
|
HDprintf("Using seed: %u\n\n", seed);
|
|
|
|
/* Grab values from environment variables if set */
|
|
parse_subfiling_env_vars();
|
|
|
|
/*
|
|
* Assume that we use the "one IOC per node" selection
|
|
* strategy by default, with a possibly modified
|
|
* number of IOCs per node value
|
|
*/
|
|
num_iocs_g = (ioc_per_node_g > 0) ? (int)ioc_per_node_g * num_nodes_g : num_nodes_g;
|
|
if (num_iocs_g > mpi_size)
|
|
num_iocs_g = mpi_size;
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(tests); i++) {
|
|
if (MPI_SUCCESS == (mpi_code_g = MPI_Barrier(comm_g))) {
|
|
(*tests[i])();
|
|
}
|
|
else {
|
|
if (MAINPROCESS)
|
|
MESG("MPI_Barrier failed");
|
|
nerrors++;
|
|
}
|
|
}
|
|
|
|
if (MAINPROCESS)
|
|
HDputs("");
|
|
|
|
/*
|
|
* Set any unset Subfiling environment variables and re-run
|
|
* the tests as a quick smoke check of whether those are
|
|
* working correctly
|
|
*/
|
|
if (stripe_size_g < 0) {
|
|
int64_t stripe_size;
|
|
char tmp[64];
|
|
|
|
/*
|
|
* Choose a random Subfiling stripe size between
|
|
* the smallest allowed value and the default value
|
|
*/
|
|
if (mpi_rank == 0) {
|
|
stripe_size = (rand() % (H5FD_SUBFILING_DEFAULT_STRIPE_SIZE - SUBFILING_MIN_STRIPE_SIZE + 1)) +
|
|
SUBFILING_MIN_STRIPE_SIZE;
|
|
}
|
|
|
|
if (mpi_size > 1) {
|
|
mpi_code_g = MPI_Bcast(&stripe_size, 1, MPI_INT64_T, 0, comm_g);
|
|
VRFY((mpi_code_g == MPI_SUCCESS), "MPI_Bcast succeeded");
|
|
}
|
|
|
|
HDsnprintf(tmp, sizeof(tmp), "%" PRId64, stripe_size);
|
|
|
|
if (HDsetenv(H5FD_SUBFILING_STRIPE_SIZE, tmp, 1) < 0) {
|
|
if (MAINPROCESS)
|
|
HDprintf("HDsetenv failed\n");
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
}
|
|
if (ioc_per_node_g < 0) {
|
|
const char *ioc_per_node_str;
|
|
|
|
if (2 * num_nodes_g <= mpi_size)
|
|
ioc_per_node_str = "2";
|
|
else
|
|
ioc_per_node_str = "1";
|
|
|
|
if (HDsetenv(H5FD_SUBFILING_IOC_PER_NODE, ioc_per_node_str, 1) < 0) {
|
|
if (MAINPROCESS)
|
|
HDprintf("HDsetenv failed\n");
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
}
|
|
if (ioc_thread_pool_size_g < 0) {
|
|
if (HDsetenv(H5FD_IOC_THREAD_POOL_SIZE, "2", 1) < 0) {
|
|
if (MAINPROCESS)
|
|
HDprintf("HDsetenv failed\n");
|
|
nerrors++;
|
|
goto exit;
|
|
}
|
|
}
|
|
|
|
/* Grab values from environment variables */
|
|
parse_subfiling_env_vars();
|
|
|
|
/*
|
|
* Assume that we use the "one IOC per node" selection
|
|
* strategy by default, with a possibly modified
|
|
* number of IOCs per node value
|
|
*/
|
|
num_iocs_g = (ioc_per_node_g > 0) ? (int)ioc_per_node_g * num_nodes_g : num_nodes_g;
|
|
if (num_iocs_g > mpi_size)
|
|
num_iocs_g = mpi_size;
|
|
|
|
if (MAINPROCESS) {
|
|
HDprintf("Re-running tests with environment variables set\n");
|
|
}
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(tests); i++) {
|
|
if (MPI_SUCCESS == (mpi_code_g = MPI_Barrier(comm_g))) {
|
|
(*tests[i])();
|
|
}
|
|
else {
|
|
if (MAINPROCESS)
|
|
MESG("MPI_Barrier failed");
|
|
nerrors++;
|
|
}
|
|
}
|
|
|
|
if (MAINPROCESS)
|
|
HDputs("");
|
|
|
|
if (nerrors)
|
|
goto exit;
|
|
|
|
if (MAINPROCESS)
|
|
HDputs("All Subfiling VFD tests passed\n");
|
|
|
|
exit:
|
|
if (nerrors) {
|
|
if (MAINPROCESS)
|
|
HDprintf("*** %d TEST ERROR%s OCCURRED ***\n", nerrors, nerrors > 1 ? "S" : "");
|
|
}
|
|
|
|
TestAlarmOff();
|
|
|
|
H5close();
|
|
|
|
if (MPI_COMM_WORLD != ioc_comm)
|
|
MPI_Comm_free(&ioc_comm);
|
|
if (MPI_COMM_WORLD != node_local_comm)
|
|
MPI_Comm_free(&node_local_comm);
|
|
|
|
MPI_Finalize();
|
|
|
|
HDexit(nerrors ? EXIT_FAILURE : EXIT_SUCCESS);
|
|
}
|
|
|
|
#else /* H5_HAVE_SUBFILING_VFD */
|
|
|
|
int
|
|
main(void)
|
|
{
|
|
h5_reset();
|
|
HDprintf("Testing Subfiling VFD functionality\n");
|
|
HDprintf("SKIPPED - Subfiling VFD not built\n");
|
|
HDexit(EXIT_SUCCESS);
|
|
}
|
|
|
|
#endif /* H5_HAVE_SUBFILING_VFD */
|