1
1
openmpi/oshmem/mca/spml/base/spml_base.c

149 строки
4.4 KiB
C
Исходник Обычный вид История

/*
* Copyright (c) 2012 Mellanox Technologies, Inc.
* All rights reserved.
* $COPYRIGHT$
*
* Additional copyrights may follow
*
* $HEADER$
*/
#include "oshmem_config.h"
#include "ompi/mca/bml/base/base.h"
#include "opal/datatype/opal_convertor.h"
#include "orte/include/orte/types.h"
#include "orte/runtime/orte_globals.h"
#include "oshmem/mca/spml/yoda/spml_yoda.h"
#include "oshmem/proc/proc.h"
#include "oshmem/mca/spml/base/base.h"
#include "oshmem/mca/spml/yoda/spml_yoda_putreq.h"
#include "oshmem/mca/spml/yoda/spml_yoda_getreq.h"
#include "ompi/mca/btl/btl.h"
#define SPML_BASE_DO_CMP(res, addr, op, val) \
switch((op)) { \
case SHMEM_CMP_EQ: \
res = *(addr) == (val) ? 1 : 0; \
break; \
case SHMEM_CMP_NE: \
res = *(addr) != (val) ? 1 : 0; \
break; \
case SHMEM_CMP_GT: \
res = *(addr) > (val) ? 1 : 0; \
break; \
case SHMEM_CMP_LE: \
res = *(addr) <= (val) ? 1 : 0; \
break; \
case SHMEM_CMP_LT: \
res = *(addr) < (val) ? 1: 0; \
break; \
case SHMEM_CMP_GE: \
res = *(addr) >= (val) ? 1 : 0; \
break; \
}
#define SPML_BASE_DO_WAIT(cond, val, addr, op) \
do { \
SPML_BASE_DO_CMP(cond, val,addr,op); \
opal_progress(); \
} while (cond == 0) ;
/**
* Wait for data delivery.
* Pool on a variable given in addr until it is not equal to value.
*/
int mca_spml_base_wait(void* addr, int cmp, void* value, int datatype)
{
int *int_addr, int_value;
long *long_addr, long_value;
short *short_addr, short_value;
long long *longlong_addr, longlong_value;
ompi_fortran_integer_t *fint_addr, fint_value;
ompi_fortran_integer4_t *fint4_addr, fint4_value;
ompi_fortran_integer8_t *fint8_addr, fint8_value;
int res = 0;
switch(datatype){
/* Int */
case SHMEM_INT:
int_value = *(int*)value;
int_addr = (int*)addr;
SPML_BASE_DO_WAIT(res, int_addr, cmp, int_value);
break;
/* Short */
case SHMEM_SHORT:
short_value = *(short*)value;
short_addr = (short*)addr;
SPML_BASE_DO_WAIT(res, short_addr, cmp, short_value);
break;
/* Long */
case SHMEM_LONG:
long_value = *(long*)value;
long_addr = (long*)addr;
SPML_BASE_DO_WAIT(res, long_addr, cmp, long_value);
break;
/* Long-Long */
case SHMEM_LLONG:
longlong_value = *(long long*)value;
longlong_addr = (long long*)addr;
SPML_BASE_DO_WAIT(res, longlong_addr, cmp, longlong_value);
break;
/*C equivalent of Fortran integer type */
case SHMEM_FINT:
fint_value = *(ompi_fortran_integer_t *)value;
fint_addr = (ompi_fortran_integer_t *)addr;
SPML_BASE_DO_WAIT(res, fint_addr, cmp, fint_value);
break;
/*C equivalent of Fortran int4 type*/
case SHMEM_FINT4:
fint4_value = *(ompi_fortran_integer4_t *)value;
fint4_addr = (ompi_fortran_integer4_t *)addr;
SPML_BASE_DO_WAIT(res, fint4_addr, cmp, fint4_value);
break;
/*C equivalent of Fortran int8 type*/
case SHMEM_FINT8:
fint8_value = *(ompi_fortran_integer8_t *)value;
fint8_addr = (ompi_fortran_integer8_t *)addr;
SPML_BASE_DO_WAIT(res, fint8_addr, cmp, fint8_value);
break;
}
return OSHMEM_SUCCESS;
}
/**
* Waits for completion of a non-blocking put or get issued by the calling PE.
* This function waits for completion of a single non-blocking transfer issued by
* shmem_put_nb() or shmem_get_nb() (or related functions) when called with the
* address of a completion handle.
* Completion of the call to shmem_wait_nb() ensures that a non-blocking transfer has
* completed. The source buffer may then be reused.
*/
int mca_spml_base_wait_nb(void* handle)
{
/* TODO fence is a gag for more accurate code
* Use shmem_quiet() (or a function calling shmem_quiet()) or
* shmem_wait_nb() to force completion of transfers for non-blocking operations.
*/
MCA_SPML_CALL(fence());
return OSHMEM_SUCCESS;
}
int mca_spml_base_oob_get_mkeys(int pe, uint32_t seg, mca_spml_mkey_t *mkeys)
{
return OSHMEM_ERROR;
}