Yet Another eXchange Tool 0.11.1
Loading...
Searching...
No Matches
test_redist_collection.c
/*
* Keywords:
* Maintainer: Moritz Hanke <hanke@dkrz.de>
* Thomas Jahns <jahns@dkrz.de>
* URL: https://dkrz-sw.gitlab-pages.dkrz.de/yaxt/
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* Neither the name of the DKRZ GmbH nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
* IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
* PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
* OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <mpi.h>
#include <yaxt.h>
#include "tests.h"
#include "ctest_common.h"
#include "test_redist_common.h"
static void
simple_test(MPI_Comm comm, Xt_config config);
static void
test_empty_redist(MPI_Comm comm, Xt_config config);
static void
test_repeated_redist(MPI_Comm comm, Xt_config config, int cache_size);
static void
test_displacement_variations(MPI_Comm comm, Xt_config config);
int main(int argc, char **argv) {
test_init_mpi(&argc, &argv, MPI_COMM_WORLD);
Xt_config config = redist_exchanger_option(&argc, &argv);
simple_test(MPI_COMM_WORLD, config);
test_empty_redist(MPI_COMM_WORLD, config);
// test with one redist used three times (with two different input data
// displacements -> test of cache) (with default cache size)
// set up data
test_repeated_redist(MPI_COMM_WORLD, config, -1);
// test with one redist used three times (with two different input data
// displacements -> test of cache) (with cache size == 0)
// set up data
test_repeated_redist(MPI_COMM_WORLD, config, 0);
test_displacement_variations(MPI_COMM_WORLD, config);
MPI_Finalize();
return TEST_EXIT_CODE;
}
static void
simple_test(MPI_Comm comm, Xt_config config)
{ // general test with one redist
// set up data
enum { nvalues = 5, nselect = (nvalues + 1)/2 };
Xt_xmap xmap = build_odd_selection_xmap(nvalues, comm);
Xt_redist redist = xt_redist_p2p_new(xmap, MPI_DOUBLE);
// generate redist_collection
Xt_redist redist_coll
= xt_redist_collection_custom_new(&redist, 1, -1, comm, config);
// test communicator of redist
if (!communicators_are_congruent(xt_redist_get_MPI_Comm(redist_coll), comm))
PUT_ERR("error in xt_redist_get_MPI_Comm\n");
// test exchange
static const double src_data[nvalues] = {1,2,3,4,5};
double dst_data[nselect];
static const double ref_dst_data[nselect] = {1,3,5};
check_redist(redist_coll, src_data, nselect,
dst_data, fill_array_double, NULL,
ref_dst_data, MPI_DOUBLE, MPI_DOUBLE);
Xt_redist redist_coll_copy = xt_redist_copy(redist_coll);
check_redist(redist_coll_copy, src_data, nselect,
dst_data, fill_array_double, NULL,
ref_dst_data, MPI_DOUBLE, MPI_DOUBLE);
// clean up
xt_redist_delete(redist_coll_copy);
xt_redist_delete(redist_coll);
}
static void
test_empty_redist(MPI_Comm comm, Xt_config config)
{ // test empty redist
Xt_idxlist src_idxlist = xt_idxempty_new();
Xt_idxlist dst_idxlist = xt_idxempty_new();
Xt_xmap xmap = xt_xmap_all2all_new(src_idxlist, dst_idxlist, comm);
xt_idxlist_delete(src_idxlist);
xt_idxlist_delete(dst_idxlist);
Xt_redist redist = xt_redist_p2p_new(xmap, MPI_DOUBLE);
// generate redist_collection
Xt_redist redist_coll
= xt_redist_collection_custom_new(&redist, 1, -1, comm, config);
// test communicator of redist
if (!communicators_are_congruent(xt_redist_get_MPI_Comm(redist_coll), comm))
PUT_ERR("error in xt_redist_get_MPI_Comm\n");
// test exchange
static const double src_data[1] = {-1};
double dst_data[1] = {-2};
xt_redist_s_exchange1(redist_coll, src_data, dst_data);
static const double ref_dst_data[1] = {-2};
if (ref_dst_data[0] != dst_data[0])
PUT_ERR("error in xt_redist_s_exchange\n");
Xt_redist redist_coll_copy = xt_redist_copy(redist_coll);
dst_data[0] = -2;
xt_redist_s_exchange1(redist_coll_copy, src_data, dst_data);
if (ref_dst_data[0] != dst_data[0])
PUT_ERR("error in xt_redist_s_exchange\n");
// clean up
xt_redist_delete(redist_coll_copy);
xt_redist_delete(redist_coll);
}
static void
test_repeated_redist(MPI_Comm comm, Xt_config config, int cache_size)
{
enum { num_slice = 3,
src_slice_len = 5, dst_slice_len = (src_slice_len+1)/2 };
Xt_xmap xmap = build_odd_selection_xmap(src_slice_len, comm);
Xt_redist redist = xt_redist_p2p_new(xmap, MPI_DOUBLE);
// generate redist_collection
Xt_redist redists[num_slice] = {redist, redist, redist};
Xt_redist redist_coll
= xt_redist_collection_custom_new(redists, num_slice, cache_size, comm,
config);
// test communicator of redist
if (!communicators_are_congruent(xt_redist_get_MPI_Comm(redist_coll), comm))
PUT_ERR("error in xt_redist_get_MPI_Comm\n");
// test exchange
{
static const double src_data[num_slice][src_slice_len]
= {{1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15}};
static const double ref_dst_data[num_slice][dst_slice_len]
= {{1,3,5},{6,8,10},{11,13,15}};
double dst_data[num_slice][dst_slice_len];
static const void *const src_data_p[num_slice]
= { src_data[0], src_data[1], src_data[2]};
void *dst_data_p[num_slice] = { dst_data[0], dst_data[1], dst_data[2] };
check_redist_coll(redist_coll, sync_mode_test_all,
num_slice, (const void **)src_data_p,
num_slice * dst_slice_len, dst_data_p, dst_data[0],
fill_array_double, NULL, ref_dst_data, MPI_DOUBLE,
MPI_DOUBLE);
}
// test exchange with changed displacements
{
static const double src_data[num_slice][src_slice_len]
= {{1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15}};
static const double ref_dst_data[num_slice][dst_slice_len]
= {{1,3,5},{6,8,10},{11,13,15}};
double dst_data[num_slice][dst_slice_len];
static const void *const src_data_p[num_slice]
= {src_data[1],src_data[0],src_data[2]};
void *dst_data_p[num_slice] = { dst_data[1], dst_data[0], dst_data[2] };
check_redist_coll(redist_coll, sync_mode_test_all,
num_slice, (const void **)src_data_p,
num_slice * dst_slice_len, dst_data_p, dst_data[0],
fill_array_double, NULL, ref_dst_data, MPI_DOUBLE,
MPI_DOUBLE);
}
// test exchange with original displacements
{
static const double src_data[num_slice][src_slice_len]
= {{1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15}};
static const double ref_dst_data[num_slice][dst_slice_len]
= {{1,3,5},{6,8,10},{11,13,15}};
double dst_data[num_slice][3];
static const void *const src_data_p[num_slice]
= { src_data[0], src_data[1], src_data[2] };
void *dst_data_p[num_slice] = { dst_data[0], dst_data[1], dst_data[2] };
check_redist_coll(redist_coll, sync_mode_test_all,
num_slice, (const void **)src_data_p,
num_slice * dst_slice_len, dst_data_p, dst_data[0],
fill_array_double, NULL, ref_dst_data, MPI_DOUBLE,
MPI_DOUBLE);
}
// clean up
xt_redist_delete(redist_coll);
}
enum { num_redists = 3 };
enum { nvalues = 5, nselect = nvalues/2+(nvalues&1) };
static void
run_displacement_check(Xt_redist redist_coll, int sync)
{
static const double src_data[num_redists][nvalues]
= {{1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15}};
enum { cache_size = 16, cache_overrun = 2 };
double src_data_[nvalues + cache_size + cache_overrun];
enum {
num_dst_elems = num_redists * nselect + cache_size + cache_overrun
};
double dst_data[num_dst_elems];
const void *src_data_p[num_redists] = {src_data[0],src_data[1],NULL};
void *dst_data_p[num_redists] = {dst_data+0,dst_data+nselect,NULL};
double ref_dst_data[num_dst_elems];
for (size_t j = 0; j < num_redists-1; ++j)
for (size_t i = 0; i < nselect; ++i)
ref_dst_data[j*nselect + i] = src_data[j][i*2];
for (size_t k = 0; k < cache_size + cache_overrun; ++k) {
memcpy(src_data_+k, src_data[2], nvalues * sizeof(*src_data_));
for (size_t i = 0; i < num_dst_elems; ++i)
dst_data[i] = -1;
src_data_p[2] = src_data_ + k;
dst_data_p[2] = dst_data + nselect*2 + k;
/* put every second value from src_data[3] into ref_dst_data
* starting at (num_redists-1)*nselect + k, i.e. with offset k
* vs. a contiguous transformation */
for (size_t i = 0; i < k; ++i)
ref_dst_data[(num_redists-1)*nselect + i] = -1;
for (size_t i = 0; i < nselect; ++i)
ref_dst_data[(num_redists-1)*nselect + k + i]
= src_data[num_redists-1][i*2];
for (size_t i = num_redists*nselect + k; i < num_dst_elems; ++i)
ref_dst_data[i] = -1;
check_redist_coll(redist_coll,
sync ? sync_mode_test_s : sync_mode_test_a,
num_redists, src_data_p, num_dst_elems, dst_data_p,
dst_data, fill_array_double, NULL, ref_dst_data,
MPI_DOUBLE, MPI_DOUBLE);
}
}
static void
test_displacement_variations(MPI_Comm comm, Xt_config config)
{
// test with one redist used three times (with different input
// data displacements until the cache is full)
// set up data
Xt_xmap xmap = build_odd_selection_xmap(nvalues, comm);
Xt_redist redist = xt_redist_p2p_new(xmap, MPI_DOUBLE);
// generate redist_collection
Xt_redist redists[num_redists] = {redist, redist, redist};
Xt_redist redist_coll
= xt_redist_collection_custom_new(redists, num_redists, -1, comm, config);
// test communicator of redist
if (!communicators_are_congruent(xt_redist_get_MPI_Comm(redist_coll), comm))
PUT_ERR("error in xt_redist_get_MPI_Comm\n");
// test exchange
run_displacement_check(redist_coll, 0);
run_displacement_check(redist_coll, 1);
Xt_redist redist_coll_copy = xt_redist_copy(redist_coll);
run_displacement_check(redist_coll_copy, 0);
run_displacement_check(redist_coll_copy, 1);
// clean up
xt_redist_delete(redist_coll_copy);
xt_redist_delete(redist_coll);
}
/*
* Local Variables:
* c-basic-offset: 2
* coding: utf-8
* indent-tabs-mode: nil
* show-trailing-whitespace: t
* require-trailing-newline: t
* End:
*/
@ MPI_COMM_WORLD
Definition core.h:73
int MPI_Comm
Definition core.h:64
add versions of standard API functions not returning on error
void xt_config_delete(Xt_config config)
Definition xt_config.c:84
void xt_initialize(MPI_Comm default_comm)
Definition xt_init.c:70
void xt_finalize(void)
Definition xt_init.c:92
Xt_idxlist xt_idxempty_new(void)
void xt_idxlist_delete(Xt_idxlist idxlist)
Definition xt_idxlist.c:75
void xt_redist_delete(Xt_redist redist)
Definition xt_redist.c:73
Xt_redist xt_redist_copy(Xt_redist redist)
Definition xt_redist.c:68
MPI_Comm xt_redist_get_MPI_Comm(Xt_redist redist)
Definition xt_redist.c:129
void xt_redist_s_exchange1(Xt_redist redist, const void *src_data, void *dst_data)
Definition xt_redist.c:91
Xt_redist xt_redist_collection_custom_new(Xt_redist *redists, int num_redists, int cache_size, MPI_Comm comm, Xt_config config)
Xt_redist xt_redist_p2p_new(Xt_xmap xmap, MPI_Datatype datatype)
void xt_xmap_delete(Xt_xmap xmap)
Definition xt_xmap.c:86
Xt_xmap xt_xmap_all2all_new(Xt_idxlist src_idxlist, Xt_idxlist dst_idxlist, MPI_Comm comm)