Blob Blame History Raw
/* -*- Mode: C; c-basic-offset:4 ; -*- */
/*
 *
 *  (C) 2012 by Argonne National Laboratory.
 *      See COPYRIGHT in top-level directory.
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <mpi.h>
#include "mpitest.h"
#include "squelch.h"

#define ITER  100
#define NUM_COUNT 2
static int test_counts[NUM_COUNT] = { 5, 1000 };

#if defined (GACC_TYPE_SHORT)
#define TYPE_C   short
#define TYPE_MPI_BASE MPI_SHORT
#define TYPE_FMT "%d"
#elif defined (GACC_TYPE_LONG)
#define TYPE_C   long
#define TYPE_MPI_BASE MPI_LONG
#define TYPE_FMT "%ld"
#elif defined (GACC_TYPE_DOUBLE)
#define TYPE_C   double
#define TYPE_MPI_BASE MPI_DOUBLE
#define TYPE_FMT "%f"
#else
#define TYPE_C   int
#define TYPE_MPI_BASE MPI_INT
#define TYPE_FMT "%d"
#endif

#if defined(GACC_TYPE_DERIVED)
#define TYPE_MPI derived_type
#else
#define TYPE_MPI TYPE_MPI_BASE
#endif

static void reset_bufs(TYPE_C * win_ptr, TYPE_C * res_ptr, TYPE_C * val_ptr, TYPE_C value,
                       int count, MPI_Win win)
{
    int rank, nproc, i;

    MPI_Barrier(MPI_COMM_WORLD);

    MPI_Comm_rank(MPI_COMM_WORLD, &rank);
    MPI_Comm_size(MPI_COMM_WORLD, &nproc);

    MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
    memset(win_ptr, 0, sizeof(TYPE_C) * nproc * count);
    MPI_Win_unlock(rank, win);

    memset(res_ptr, -1, sizeof(TYPE_C) * nproc * count);

    for (i = 0; i < count; i++)
        val_ptr[i] = value;

    MPI_Barrier(MPI_COMM_WORLD);
}

int main(int argc, char **argv)
{
    int i, rank, nproc;
    int j, count, errors = 0;
    TYPE_C *win_ptr, *res_ptr, *val_ptr;
    MPI_Win win;
#if defined (GACC_TYPE_DERIVED)
    MPI_Datatype derived_type;
#endif

    MTest_Init(&argc, &argv);

    MPI_Comm_rank(MPI_COMM_WORLD, &rank);
    MPI_Comm_size(MPI_COMM_WORLD, &nproc);

    for (j = 0; j < NUM_COUNT; j++) {
        count = test_counts[j];
        win_ptr = malloc(sizeof(TYPE_C) * nproc * count);
        res_ptr = malloc(sizeof(TYPE_C) * nproc * count);
        val_ptr = malloc(sizeof(TYPE_C) * count);

#if defined (GACC_TYPE_DERIVED)
        MPI_Type_contiguous(1, TYPE_MPI_BASE, &derived_type);
        MPI_Type_commit(&derived_type);
#endif

        MPI_Win_create(win_ptr, sizeof(TYPE_C) * nproc * count, sizeof(TYPE_C),
                       MPI_INFO_NULL, MPI_COMM_WORLD, &win);

        /* Test self communication */

        reset_bufs(win_ptr, res_ptr, val_ptr, 1, count, win);

        for (i = 0; i < ITER; i++) {
            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
            MPI_Get_accumulate(val_ptr, count, TYPE_MPI, res_ptr, count, TYPE_MPI,
                               rank, 0, count, TYPE_MPI, MPI_SUM, win);
            MPI_Win_unlock(rank, win);
        }

        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        for (i = 0; i < count; i++) {
            if (win_ptr[i] != ITER) {
                SQUELCH(printf("%d->%d -- SELF[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                               rank, rank, i, (TYPE_C) ITER, win_ptr[i]););
                errors++;
            }
        }
        MPI_Win_unlock(rank, win);

        /* Test neighbor communication */

        reset_bufs(win_ptr, res_ptr, val_ptr, 1, count, win);

        for (i = 0; i < ITER; i++) {
            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, (rank + 1) % nproc, 0, win);
            MPI_Get_accumulate(val_ptr, count, TYPE_MPI, res_ptr, count, TYPE_MPI,
                               (rank + 1) % nproc, 0, count, TYPE_MPI, MPI_SUM, win);
            MPI_Win_unlock((rank + 1) % nproc, win);
        }

        MPI_Barrier(MPI_COMM_WORLD);

        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        for (i = 0; i < count; i++) {
            if (win_ptr[i] != ITER) {
                SQUELCH(printf("%d->%d -- NEIGHBOR[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                               (rank + 1) % nproc, rank, i, (TYPE_C) ITER, win_ptr[i]););
                errors++;
            }
        }
        MPI_Win_unlock(rank, win);

        /* Test contention */

        reset_bufs(win_ptr, res_ptr, val_ptr, 1, count, win);

        if (rank != 0) {
            for (i = 0; i < ITER; i++) {
                MPI_Win_lock(MPI_LOCK_EXCLUSIVE, 0, 0, win);
                MPI_Get_accumulate(val_ptr, count, TYPE_MPI, res_ptr, count, TYPE_MPI,
                                   0, 0, count, TYPE_MPI, MPI_SUM, win);
                MPI_Win_unlock(0, win);
            }
        }

        MPI_Barrier(MPI_COMM_WORLD);

        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        if (rank == 0 && nproc > 1) {
            for (i = 0; i < count; i++) {
                if (win_ptr[i] != ITER * (nproc - 1)) {
                    SQUELCH(printf
                            ("*->%d - CONTENTION[%d]: expected=" TYPE_FMT " val=" TYPE_FMT "\n",
                             rank, i, (TYPE_C) ITER * (nproc - 1), win_ptr[i]););
                    errors++;
                }
            }
        }
        MPI_Win_unlock(rank, win);

        /* Test all-to-all communication (fence) */

        reset_bufs(win_ptr, res_ptr, val_ptr, rank, count, win);

        for (i = 0; i < ITER; i++) {
            int j;

            MPI_Win_fence(MPI_MODE_NOPRECEDE, win);
            for (j = 0; j < nproc; j++) {
                MPI_Get_accumulate(val_ptr, count, TYPE_MPI, &res_ptr[j * count], count, TYPE_MPI,
                                   j, rank * count, count, TYPE_MPI, MPI_SUM, win);
            }
            MPI_Win_fence(MPI_MODE_NOSUCCEED, win);
            MPI_Barrier(MPI_COMM_WORLD);

            for (j = 0; j < nproc; j++) {
                int c;
                for (c = 0; c < count; c++) {
                    if (res_ptr[j * count + c] != i * rank) {
                        SQUELCH(printf
                                ("%d->%d -- ALL-TO-ALL (FENCE) [%d]: iter %d, expected result "
                                 TYPE_FMT ", got " TYPE_FMT "\n", rank, j, c, i, (TYPE_C) i * rank,
                                 res_ptr[j * count + c]););
                        errors++;
                    }
                }
            }
        }

        MPI_Barrier(MPI_COMM_WORLD);
        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        for (i = 0; i < nproc; i++) {
            int c;
            for (c = 0; c < count; c++) {
                if (win_ptr[i * count + c] != ITER * i) {
                    SQUELCH(printf
                            ("%d->%d -- ALL-TO-ALL (FENCE): expected " TYPE_FMT ", got " TYPE_FMT
                             "\n", i, rank, (TYPE_C) ITER * i, win_ptr[i * count + c]););
                    errors++;
                }
            }
        }
        MPI_Win_unlock(rank, win);

        /* Test all-to-all communication (lock-all) */

        reset_bufs(win_ptr, res_ptr, val_ptr, rank, count, win);

        for (i = 0; i < ITER; i++) {
            int j;

            MPI_Win_lock_all(0, win);
            for (j = 0; j < nproc; j++) {
                MPI_Get_accumulate(val_ptr, count, TYPE_MPI, &res_ptr[j * count], count, TYPE_MPI,
                                   j, rank * count, count, TYPE_MPI, MPI_SUM, win);
            }
            MPI_Win_unlock_all(win);
            MPI_Barrier(MPI_COMM_WORLD);

            for (j = 0; j < nproc; j++) {
                int c;
                for (c = 0; c < count; c++) {
                    if (res_ptr[j * count + c] != i * rank) {
                        SQUELCH(printf
                                ("%d->%d -- ALL-TO-ALL (LOCK-ALL) [%d]: iter %d, expected result "
                                 TYPE_FMT ", got " TYPE_FMT "\n", rank, j, c, i, (TYPE_C) i * rank,
                                 res_ptr[j * count + c]););
                        errors++;
                    }
                }
            }
        }

        MPI_Barrier(MPI_COMM_WORLD);
        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        for (i = 0; i < nproc; i++) {
            int c;
            for (c = 0; c < count; c++) {
                if (win_ptr[i * count + c] != ITER * i) {
                    SQUELCH(printf
                            ("%d->%d -- ALL-TO-ALL (LOCK-ALL): expected " TYPE_FMT ", got " TYPE_FMT
                             "\n", i, rank, (TYPE_C) ITER * i, win_ptr[i * count + c]););
                    errors++;
                }
            }
        }
        MPI_Win_unlock(rank, win);

        /* Test all-to-all communication (lock-all+flush) */

        reset_bufs(win_ptr, res_ptr, val_ptr, rank, count, win);

        for (i = 0; i < ITER; i++) {
            int j;

            MPI_Win_lock_all(0, win);
            for (j = 0; j < nproc; j++) {
                MPI_Get_accumulate(val_ptr, count, TYPE_MPI, &res_ptr[j * count], count, TYPE_MPI,
                                   j, rank * count, count, TYPE_MPI, MPI_SUM, win);
                MPI_Win_flush(j, win);
            }
            MPI_Win_unlock_all(win);
            MPI_Barrier(MPI_COMM_WORLD);

            for (j = 0; j < nproc; j++) {
                int c;
                for (c = 0; c < count; c++) {
                    if (res_ptr[j * count + c] != i * rank) {
                        SQUELCH(printf
                                ("%d->%d -- ALL-TO-ALL (LOCK-ALL+FLUSH) [%d]: iter %d, expected result "
                                 TYPE_FMT ", got " TYPE_FMT "\n", rank, j, c, i, (TYPE_C) i * rank,
                                 res_ptr[j * count + c]););
                        errors++;
                    }
                }
            }
        }

        MPI_Barrier(MPI_COMM_WORLD);
        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        for (i = 0; i < nproc; i++) {
            int c;
            for (c = 0; c < count; c++) {
                if (win_ptr[i * count + c] != ITER * i) {
                    SQUELCH(printf
                            ("%d->%d -- ALL-TO-ALL (LOCK-ALL+FLUSH): expected " TYPE_FMT ", got "
                             TYPE_FMT "\n", i, rank, (TYPE_C) ITER * i, win_ptr[i * count + c]););
                    errors++;
                }
            }
        }
        MPI_Win_unlock(rank, win);

        /* Test NO_OP (neighbor communication) */

        reset_bufs(win_ptr, res_ptr, val_ptr, 1, count, win);

        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        for (i = 0; i < count * nproc; i++)
            win_ptr[i] = (TYPE_C) rank;
        MPI_Win_unlock(rank, win);
        MPI_Barrier(MPI_COMM_WORLD);

        for (i = 0; i < ITER; i++) {
            int j, target = (rank + 1) % nproc;

            /* Test: origin_buf = NULL */
            memset(res_ptr, -1, sizeof(TYPE_C) * nproc * count);        /* reset result buffer. */

            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, target, 0, win);
            MPI_Get_accumulate(NULL, count, TYPE_MPI, res_ptr, count, TYPE_MPI,
                               target, 0, count, TYPE_MPI, MPI_NO_OP, win);
            MPI_Win_unlock(target, win);

            for (j = 0; j < count; j++) {
                if (res_ptr[j] != (TYPE_C) target) {
                    SQUELCH(printf
                            ("%d->%d -- NOP(1)[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                             target, rank, i, (TYPE_C) target, res_ptr[i]););
                    errors++;
                }
            }

            /* Test: origin_buf = NULL, origin_count = 0 */
            memset(res_ptr, -1, sizeof(TYPE_C) * nproc * count);

            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, target, 0, win);
            MPI_Get_accumulate(NULL, 0, TYPE_MPI, res_ptr, count, TYPE_MPI,
                               target, 0, count, TYPE_MPI, MPI_NO_OP, win);
            MPI_Win_unlock(target, win);

            for (j = 0; j < count; j++) {
                if (res_ptr[j] != (TYPE_C) target) {
                    SQUELCH(printf
                            ("%d->%d -- NOP(2)[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                             target, rank, i, (TYPE_C) target, res_ptr[i]););
                    errors++;
                }
            }

            /* Test: origin_buf = NULL, origin_count = 0, origin_dtype = NULL */
            memset(res_ptr, -1, sizeof(TYPE_C) * nproc * count);

            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, target, 0, win);
            MPI_Get_accumulate(NULL, 0, MPI_DATATYPE_NULL, res_ptr, count, TYPE_MPI,
                               target, 0, count, TYPE_MPI, MPI_NO_OP, win);
            MPI_Win_unlock(target, win);

            for (j = 0; j < count; j++) {
                if (res_ptr[j] != (TYPE_C) target) {
                    SQUELCH(printf
                            ("%d->%d -- NOP(2)[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                             target, rank, i, (TYPE_C) target, res_ptr[i]););
                    errors++;
                }
            }
        }

        /* Test NO_OP (self communication) */

        reset_bufs(win_ptr, res_ptr, val_ptr, 1, count, win);

        MPI_Win_lock(MPI_LOCK_EXCLUSIVE, rank, 0, win);
        for (i = 0; i < count * nproc; i++)
            win_ptr[i] = (TYPE_C) rank;
        MPI_Win_unlock(rank, win);
        MPI_Barrier(MPI_COMM_WORLD);

        for (i = 0; i < ITER; i++) {
            int j, target = rank;

            /* Test: origin_buf = NULL */
            memset(res_ptr, -1, sizeof(TYPE_C) * nproc * count);

            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, target, 0, win);
            MPI_Get_accumulate(NULL, count, TYPE_MPI, res_ptr, count, TYPE_MPI,
                               target, 0, count, TYPE_MPI, MPI_NO_OP, win);
            MPI_Win_unlock(target, win);

            for (j = 0; j < count; j++) {
                if (res_ptr[j] != (TYPE_C) target) {
                    SQUELCH(printf
                            ("%d->%d -- NOP_SELF(1)[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                             target, rank, i, (TYPE_C) target, res_ptr[i]););
                    errors++;
                }
            }

            /* Test: origin_buf = NULL, origin_count = 0 */
            memset(res_ptr, -1, sizeof(TYPE_C) * nproc * count);

            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, target, 0, win);
            MPI_Get_accumulate(NULL, 0, TYPE_MPI, res_ptr, count, TYPE_MPI,
                               target, 0, count, TYPE_MPI, MPI_NO_OP, win);
            MPI_Win_unlock(target, win);

            for (j = 0; j < count; j++) {
                if (res_ptr[j] != (TYPE_C) target) {
                    SQUELCH(printf
                            ("%d->%d -- NOP_SELF(2)[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                             target, rank, i, (TYPE_C) target, res_ptr[i]););
                    errors++;
                }
            }

            /* Test: origin_buf = NULL, origin_count = 0, origin_dtype = NULL */
            memset(res_ptr, -1, sizeof(TYPE_C) * nproc * count);

            MPI_Win_lock(MPI_LOCK_EXCLUSIVE, target, 0, win);
            MPI_Get_accumulate(NULL, 0, MPI_DATATYPE_NULL, res_ptr, count, TYPE_MPI,
                               target, 0, count, TYPE_MPI, MPI_NO_OP, win);
            MPI_Win_unlock(target, win);

            for (j = 0; j < count; j++) {
                if (res_ptr[j] != (TYPE_C) target) {
                    SQUELCH(printf
                            ("%d->%d -- NOP_SELF(2)[%d]: expected " TYPE_FMT ", got " TYPE_FMT "\n",
                             target, rank, i, (TYPE_C) target, res_ptr[i]););
                    errors++;
                }
            }
        }

        MPI_Win_free(&win);

#if defined (GACC_TYPE_DERIVED)
        MPI_Type_free(&derived_type);
#endif

        free(win_ptr);
        free(res_ptr);
        free(val_ptr);
    }

    MTest_Finalize(errors);

    return MTestReturnValue(errors);
}