|
Packit |
0848f5 |
/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil ; -*- */
|
|
Packit |
0848f5 |
/*
|
|
Packit |
0848f5 |
*
|
|
Packit |
0848f5 |
* (C) 2003 by Argonne National Laboratory.
|
|
Packit |
0848f5 |
* See COPYRIGHT in top-level directory.
|
|
Packit |
0848f5 |
*/
|
|
Packit |
0848f5 |
#include <mpi.h>
|
|
Packit |
0848f5 |
#include <stdio.h>
|
|
Packit |
0848f5 |
#include <assert.h>
|
|
Packit |
0848f5 |
#include "mpitest.h"
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
#define ITER 100
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
int main(int argc, char *argv[])
|
|
Packit |
0848f5 |
{
|
|
Packit |
0848f5 |
int rank, nproc, i;
|
|
Packit |
0848f5 |
int errors = 0, all_errors = 0;
|
|
Packit |
0848f5 |
int *buf;
|
|
Packit |
0848f5 |
MPI_Win window;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Init(&argc, &argv);
|
|
Packit |
0848f5 |
MPI_Comm_rank(MPI_COMM_WORLD, &rank;;
|
|
Packit |
0848f5 |
MPI_Comm_size(MPI_COMM_WORLD, &nproc);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (nproc < 2) {
|
|
Packit |
0848f5 |
if (rank == 0)
|
|
Packit |
0848f5 |
printf("Error: must be run with two or more processes\n");
|
|
Packit |
0848f5 |
MPI_Abort(MPI_COMM_WORLD, 1);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/** Create using MPI_Win_create() **/
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (rank == 0) {
|
|
Packit |
0848f5 |
MPI_Alloc_mem(4 * sizeof(int), MPI_INFO_NULL, &buf;;
|
|
Packit |
0848f5 |
*buf = nproc - 1;
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
else
|
|
Packit |
0848f5 |
buf = NULL;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_create(buf, 4 * sizeof(int) * (rank == 0), sizeof(int),
|
|
Packit |
0848f5 |
MPI_INFO_NULL, MPI_COMM_WORLD, &window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* PROC_NULL Communication */
|
|
Packit |
0848f5 |
{
|
|
Packit |
0848f5 |
MPI_Request pn_req[4];
|
|
Packit |
0848f5 |
int val[4], res;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_lock_all(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget_accumulate(&val[0], 1, MPI_INT, &res, 1, MPI_INT, MPI_PROC_NULL, 0, 1, MPI_INT,
|
|
Packit |
0848f5 |
MPI_REPLACE, window, &pn_req[0]);
|
|
Packit |
0848f5 |
MPI_Rget(&val[1], 1, MPI_INT, MPI_PROC_NULL, 1, 1, MPI_INT, window, &pn_req[1]);
|
|
Packit |
0848f5 |
MPI_Rput(&val[2], 1, MPI_INT, MPI_PROC_NULL, 2, 1, MPI_INT, window, &pn_req[2]);
|
|
Packit |
0848f5 |
MPI_Raccumulate(&val[3], 1, MPI_INT, MPI_PROC_NULL, 3, 1, MPI_INT, MPI_REPLACE, window,
|
|
Packit |
0848f5 |
&pn_req[3]);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
assert(pn_req[0] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[1] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[2] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[3] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_unlock_all(window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Waitall(4, pn_req, MPI_STATUSES_IGNORE);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_lock(MPI_LOCK_SHARED, 0, 0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* GET-ACC: Test third-party communication, through rank 0. */
|
|
Packit |
0848f5 |
for (i = 0; i < ITER; i++) {
|
|
Packit |
0848f5 |
MPI_Request gacc_req;
|
|
Packit |
0848f5 |
int val = -1, exp = -1;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Processes form a ring. Process 0 starts first, then passes a token
|
|
Packit |
0848f5 |
* to the right. Each process, in turn, performs third-party
|
|
Packit |
0848f5 |
* communication via process 0's window. */
|
|
Packit |
0848f5 |
if (rank > 0) {
|
|
Packit |
0848f5 |
MPI_Recv(NULL, 0, MPI_BYTE, rank - 1, 0, MPI_COMM_WORLD, MPI_STATUS_IGNORE);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget_accumulate(&rank, 1, MPI_INT, &val, 1, MPI_INT, 0, 0, 1, MPI_INT, MPI_REPLACE,
|
|
Packit |
0848f5 |
window, &gacc_req);
|
|
Packit |
0848f5 |
assert(gacc_req != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
MPI_Wait(&gacc_req, MPI_STATUS_IGNORE);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
exp = (rank + nproc - 1) % nproc;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (val != exp) {
|
|
Packit |
0848f5 |
printf("%d - Got %d, expected %d\n", rank, val, exp);
|
|
Packit |
0848f5 |
errors++;
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (rank < nproc - 1) {
|
|
Packit |
0848f5 |
MPI_Send(NULL, 0, MPI_BYTE, rank + 1, 0, MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (rank == 0)
|
|
Packit |
0848f5 |
*buf = nproc - 1;
|
|
Packit |
0848f5 |
MPI_Win_sync(window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* GET+PUT: Test third-party communication, through rank 0. */
|
|
Packit |
0848f5 |
for (i = 0; i < ITER; i++) {
|
|
Packit |
0848f5 |
MPI_Request req;
|
|
Packit |
0848f5 |
int val = -1, exp = -1;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Processes form a ring. Process 0 starts first, then passes a token
|
|
Packit |
0848f5 |
* to the right. Each process, in turn, performs third-party
|
|
Packit |
0848f5 |
* communication via process 0's window. */
|
|
Packit |
0848f5 |
if (rank > 0) {
|
|
Packit |
0848f5 |
MPI_Recv(NULL, 0, MPI_BYTE, rank - 1, 0, MPI_COMM_WORLD, MPI_STATUS_IGNORE);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget(&val, 1, MPI_INT, 0, 0, 1, MPI_INT, window, &req;;
|
|
Packit |
0848f5 |
assert(req != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
MPI_Wait(&req, MPI_STATUS_IGNORE);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Use flush to guarantee remote completion */
|
|
Packit |
0848f5 |
MPI_Put(&rank, 1, MPI_INT, 0, 0, 1, MPI_INT, window);
|
|
Packit |
0848f5 |
MPI_Win_flush(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
exp = (rank + nproc - 1) % nproc;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (val != exp) {
|
|
Packit |
0848f5 |
printf("%d - Got %d, expected %d\n", rank, val, exp);
|
|
Packit |
0848f5 |
errors++;
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (rank < nproc - 1) {
|
|
Packit |
0848f5 |
MPI_Send(NULL, 0, MPI_BYTE, rank + 1, 0, MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (rank == 0)
|
|
Packit |
0848f5 |
*buf = nproc - 1;
|
|
Packit |
0848f5 |
MPI_Win_sync(window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* GET+ACC: Test third-party communication, through rank 0. */
|
|
Packit |
0848f5 |
for (i = 0; i < ITER; i++) {
|
|
Packit |
0848f5 |
MPI_Request req;
|
|
Packit |
0848f5 |
int val = -1, exp = -1;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Processes form a ring. Process 0 starts first, then passes a token
|
|
Packit |
0848f5 |
* to the right. Each process, in turn, performs third-party
|
|
Packit |
0848f5 |
* communication via process 0's window. */
|
|
Packit |
0848f5 |
if (rank > 0) {
|
|
Packit |
0848f5 |
MPI_Recv(NULL, 0, MPI_BYTE, rank - 1, 0, MPI_COMM_WORLD, MPI_STATUS_IGNORE);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget(&val, 1, MPI_INT, 0, 0, 1, MPI_INT, window, &req;;
|
|
Packit |
0848f5 |
assert(req != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
MPI_Wait(&req, MPI_STATUS_IGNORE);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Use flush to guarantee remote completion */
|
|
Packit |
0848f5 |
MPI_Accumulate(&rank, 1, MPI_INT, 0, 0, 1, MPI_INT, MPI_REPLACE, window);
|
|
Packit |
0848f5 |
MPI_Win_flush(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
exp = (rank + nproc - 1) % nproc;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (val != exp) {
|
|
Packit |
0848f5 |
printf("%d - Got %d, expected %d\n", rank, val, exp);
|
|
Packit |
0848f5 |
errors++;
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (rank < nproc - 1) {
|
|
Packit |
0848f5 |
MPI_Send(NULL, 0, MPI_BYTE, rank + 1, 0, MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
MPI_Win_unlock(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Wait inside of an epoch */
|
|
Packit |
0848f5 |
{
|
|
Packit |
0848f5 |
MPI_Request pn_req[4];
|
|
Packit |
0848f5 |
int val[4], res;
|
|
Packit |
0848f5 |
const int target = 0;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_lock_all(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget_accumulate(&val[0], 1, MPI_INT, &res, 1, MPI_INT, target, 0, 1, MPI_INT,
|
|
Packit |
0848f5 |
MPI_REPLACE, window, &pn_req[0]);
|
|
Packit |
0848f5 |
MPI_Rget(&val[1], 1, MPI_INT, target, 1, 1, MPI_INT, window, &pn_req[1]);
|
|
Packit |
0848f5 |
MPI_Rput(&val[2], 1, MPI_INT, target, 2, 1, MPI_INT, window, &pn_req[2]);
|
|
Packit |
0848f5 |
MPI_Raccumulate(&val[3], 1, MPI_INT, target, 3, 1, MPI_INT, MPI_REPLACE, window,
|
|
Packit |
0848f5 |
&pn_req[3]);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
assert(pn_req[0] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[1] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[2] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[3] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Waitall(4, pn_req, MPI_STATUSES_IGNORE);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_unlock_all(window);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Barrier(MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Wait outside of an epoch */
|
|
Packit |
0848f5 |
{
|
|
Packit |
0848f5 |
MPI_Request pn_req[4];
|
|
Packit |
0848f5 |
int val[4], res;
|
|
Packit |
0848f5 |
const int target = 0;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_lock_all(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget_accumulate(&val[0], 1, MPI_INT, &res, 1, MPI_INT, target, 0, 1, MPI_INT,
|
|
Packit |
0848f5 |
MPI_REPLACE, window, &pn_req[0]);
|
|
Packit |
0848f5 |
MPI_Rget(&val[1], 1, MPI_INT, target, 1, 1, MPI_INT, window, &pn_req[1]);
|
|
Packit |
0848f5 |
MPI_Rput(&val[2], 1, MPI_INT, target, 2, 1, MPI_INT, window, &pn_req[2]);
|
|
Packit |
0848f5 |
MPI_Raccumulate(&val[3], 1, MPI_INT, target, 3, 1, MPI_INT, MPI_REPLACE, window,
|
|
Packit |
0848f5 |
&pn_req[3]);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
assert(pn_req[0] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[1] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[2] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[3] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_unlock_all(window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Waitall(4, pn_req, MPI_STATUSES_IGNORE);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Wait in a different epoch */
|
|
Packit |
0848f5 |
{
|
|
Packit |
0848f5 |
MPI_Request pn_req[4];
|
|
Packit |
0848f5 |
int val[4], res;
|
|
Packit |
0848f5 |
const int target = 0;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_lock_all(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget_accumulate(&val[0], 1, MPI_INT, &res, 1, MPI_INT, target, 0, 1, MPI_INT,
|
|
Packit |
0848f5 |
MPI_REPLACE, window, &pn_req[0]);
|
|
Packit |
0848f5 |
MPI_Rget(&val[1], 1, MPI_INT, target, 1, 1, MPI_INT, window, &pn_req[1]);
|
|
Packit |
0848f5 |
MPI_Rput(&val[2], 1, MPI_INT, target, 2, 1, MPI_INT, window, &pn_req[2]);
|
|
Packit |
0848f5 |
MPI_Raccumulate(&val[3], 1, MPI_INT, target, 3, 1, MPI_INT, MPI_REPLACE, window,
|
|
Packit |
0848f5 |
&pn_req[3]);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
assert(pn_req[0] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[1] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[2] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[3] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_unlock_all(window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_lock_all(0, window);
|
|
Packit |
0848f5 |
MPI_Waitall(4, pn_req, MPI_STATUSES_IGNORE);
|
|
Packit |
0848f5 |
MPI_Win_unlock_all(window);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
/* Wait in a fence epoch */
|
|
Packit |
0848f5 |
{
|
|
Packit |
0848f5 |
MPI_Request pn_req[4];
|
|
Packit |
0848f5 |
int val[4], res;
|
|
Packit |
0848f5 |
const int target = 0;
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_lock_all(0, window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Rget_accumulate(&val[0], 1, MPI_INT, &res, 1, MPI_INT, target, 0, 1, MPI_INT,
|
|
Packit |
0848f5 |
MPI_REPLACE, window, &pn_req[0]);
|
|
Packit |
0848f5 |
MPI_Rget(&val[1], 1, MPI_INT, target, 1, 1, MPI_INT, window, &pn_req[1]);
|
|
Packit |
0848f5 |
MPI_Rput(&val[2], 1, MPI_INT, target, 2, 1, MPI_INT, window, &pn_req[2]);
|
|
Packit |
0848f5 |
MPI_Raccumulate(&val[3], 1, MPI_INT, target, 3, 1, MPI_INT, MPI_REPLACE, window,
|
|
Packit |
0848f5 |
&pn_req[3]);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
assert(pn_req[0] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[1] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[2] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
assert(pn_req[3] != MPI_REQUEST_NULL);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_unlock_all(window);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_fence(0, window);
|
|
Packit |
0848f5 |
MPI_Waitall(4, pn_req, MPI_STATUSES_IGNORE);
|
|
Packit |
0848f5 |
MPI_Win_fence(0, window);
|
|
Packit |
0848f5 |
}
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Win_free(&window);
|
|
Packit |
0848f5 |
if (buf)
|
|
Packit |
0848f5 |
MPI_Free_mem(buf);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Reduce(&errors, &all_errors, 1, MPI_INT, MPI_SUM, 0, MPI_COMM_WORLD);
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
if (rank == 0 && all_errors == 0)
|
|
Packit |
0848f5 |
printf(" No Errors\n");
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
MPI_Finalize();
|
|
Packit |
0848f5 |
|
|
Packit |
0848f5 |
return 0;
|
|
Packit |
0848f5 |
}
|