18039b34b4
The error message emitted by ompi/java when dlopen fails is misleading and not very informative. Signed-off-by: Howard Pritchard <howardp@lanl.gov>
1110 строки
33 KiB
C
1110 строки
33 KiB
C
/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil -*- */
|
|
/*
|
|
* Copyright (c) 2004-2007 The Trustees of Indiana University and Indiana
|
|
* University Research and Technology
|
|
* Corporation. All rights reserved.
|
|
* Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
* of Tennessee Research Foundation. All rights
|
|
* reserved.
|
|
* Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
* University of Stuttgart. All rights reserved.
|
|
* Copyright (c) 2004-2005 The Regents of the University of California.
|
|
* All rights reserved.
|
|
* Copyright (c) 2015 Los Alamos National Security, LLC. All rights
|
|
* reserved.
|
|
* Copyright (c) 2015 Cisco Systems, Inc. All rights reserved.
|
|
* $COPYRIGHT$
|
|
*
|
|
* Additional copyrights may follow
|
|
*
|
|
* $HEADER$
|
|
*/
|
|
/*
|
|
* This file is almost a complete re-write for Open MPI compared to the
|
|
* original mpiJava package. Its license and copyright are listed below.
|
|
* See <path to ompi/mpi/java/README> for more information.
|
|
*/
|
|
/*
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
/*
|
|
* File : mpi_MPI.c
|
|
* Headerfile : mpi_MPI.h
|
|
* Author : SungHoon Ko, Xinying Li (contributions from MAEDA Atusi)
|
|
* Created : Thu Apr 9 12:22:15 1998
|
|
* Revision : $Revision: 1.17 $
|
|
* Updated : $Date: 2003/01/17 01:50:37 $
|
|
* Copyright: Northeast Parallel Architectures Center
|
|
* at Syracuse University 1998
|
|
*/
|
|
#include "ompi_config.h"
|
|
|
|
#include <stdio.h>
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
#include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_STDLIB_H
|
|
#include <stdlib.h>
|
|
#endif
|
|
#ifdef HAVE_STRING_H
|
|
#include <string.h>
|
|
#endif
|
|
#ifdef HAVE_TARGETCONDITIONALS_H
|
|
#include <TargetConditionals.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_STAT_H
|
|
#include <sys/stat.h>
|
|
#endif
|
|
#include <dlfcn.h>
|
|
#include <poll.h>
|
|
|
|
#include "opal/util/output.h"
|
|
#include "opal/datatype/opal_convertor.h"
|
|
#include "opal/mca/base/mca_base_var.h"
|
|
|
|
#include "mpi.h"
|
|
#include "ompi/errhandler/errcode.h"
|
|
#include "ompi/datatype/ompi_datatype.h"
|
|
#include "mpi_MPI.h"
|
|
#include "mpiJava.h"
|
|
|
|
ompi_java_globals_t ompi_java;
|
|
int ompi_mpi_java_eager = 65536;
|
|
opal_free_list_t ompi_java_buffers;
|
|
static void *libmpi = NULL;
|
|
|
|
static void bufferConstructor(ompi_java_buffer_t *item)
|
|
{
|
|
item->buffer = malloc(ompi_mpi_java_eager);
|
|
}
|
|
|
|
static void bufferDestructor(ompi_java_buffer_t *item)
|
|
{
|
|
free(item->buffer);
|
|
}
|
|
|
|
OBJ_CLASS_INSTANCE(ompi_java_buffer_t,
|
|
opal_free_list_item_t,
|
|
bufferConstructor,
|
|
bufferDestructor);
|
|
|
|
/*
|
|
* Class: mpi_MPI
|
|
* Method: loadGlobalLibraries
|
|
*
|
|
* Java implementations typically default to loading dynamic
|
|
* libraries strictly to a local namespace. This breaks the
|
|
* Open MPI model where components reference back up to the
|
|
* base libraries (e.g., libmpi) as it requires that the
|
|
* symbols in those base libraries be globally available.
|
|
*
|
|
* One option, of course, is to build with --disable-dlopen.
|
|
* However, this would preclude the ability to pickup 3rd-party
|
|
* binary plug-ins at time of execution. This is a valuable
|
|
* capability that would be a negative factor towards use of
|
|
* the Java bindings.
|
|
*
|
|
* The other option is to explicitly dlopen libmpi ourselves
|
|
* and instruct dlopen to add all those symbols to the global
|
|
* namespace. This must be done prior to calling any MPI
|
|
* function (e.g., MPI_Init) or else Java will have already
|
|
* loaded the library to the local namespace. So create a
|
|
* special JNI entry point that just loads the required libmpi
|
|
* to the global namespace and call it first (see MPI.java),
|
|
* thus making all symbols available to subsequent dlopen calls
|
|
* when opening OMPI components.
|
|
*/
|
|
jint JNI_OnLoad(JavaVM *vm, void *reserved)
|
|
{
|
|
char *env = getenv("OMPI_ATTACH");
|
|
if (NULL != env && 0 < atoi(env)) {
|
|
volatile int _dbg = 1;
|
|
while (_dbg) poll(NULL, 0, 1);
|
|
}
|
|
libmpi = dlopen("libmpi." OPAL_DYN_LIB_SUFFIX, RTLD_NOW | RTLD_GLOBAL);
|
|
|
|
if(libmpi == NULL)
|
|
{
|
|
fprintf(stderr, "Java bindings failed to load libmpi: %s\n",dlerror());
|
|
exit(1);
|
|
}
|
|
|
|
return JNI_VERSION_1_6;
|
|
}
|
|
|
|
void JNI_OnUnload(JavaVM *vm, void *reserved)
|
|
{
|
|
if(libmpi != NULL)
|
|
dlclose(libmpi);
|
|
}
|
|
|
|
static void initFreeList(void)
|
|
{
|
|
OBJ_CONSTRUCT(&ompi_java_buffers, opal_free_list_t);
|
|
|
|
int r = opal_free_list_init(&ompi_java_buffers,
|
|
sizeof(ompi_java_buffer_t),
|
|
opal_cache_line_size,
|
|
OBJ_CLASS(ompi_java_buffer_t),
|
|
0, /* payload size */
|
|
0, /* payload align */
|
|
2, /* initial elements to alloc */
|
|
-1, /* max elements */
|
|
2, /* num elements per alloc */
|
|
NULL, /* mpool */
|
|
0, /* mpool reg flags */
|
|
NULL, /* unused0 */
|
|
NULL, /* item_init */
|
|
NULL /* inem_init context */);
|
|
if(r != OPAL_SUCCESS)
|
|
{
|
|
fprintf(stderr, "Unable to initialize ompi_java_buffers.\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
static jclass findClass(JNIEnv *env, const char *className)
|
|
{
|
|
jclass c = (*env)->FindClass(env, className),
|
|
r = (*env)->NewGlobalRef(env, c);
|
|
|
|
(*env)->DeleteLocalRef(env, c);
|
|
return r;
|
|
}
|
|
|
|
static void findClasses(JNIEnv *env)
|
|
{
|
|
ompi_java.CartParmsClass = findClass(env, "mpi/CartParms");
|
|
ompi_java.ShiftParmsClass = findClass(env, "mpi/ShiftParms");
|
|
ompi_java.GraphParmsClass = findClass(env, "mpi/GraphParms");
|
|
|
|
ompi_java.DistGraphNeighborsClass = findClass(
|
|
env, "mpi/DistGraphNeighbors");
|
|
|
|
ompi_java.StatusClass = findClass(env, "mpi/Status");
|
|
ompi_java.ExceptionClass = findClass(env, "mpi/MPIException");
|
|
|
|
ompi_java.ExceptionInit = (*env)->GetMethodID(
|
|
env, ompi_java.ExceptionClass,
|
|
"<init>", "(IILjava/lang/String;)V");
|
|
|
|
ompi_java.IntegerClass = findClass(env, "java/lang/Integer");
|
|
ompi_java.LongClass = findClass(env, "java/lang/Long");
|
|
|
|
ompi_java.IntegerValueOf = (*env)->GetStaticMethodID(
|
|
env, ompi_java.IntegerClass, "valueOf", "(I)Ljava/lang/Integer;");
|
|
ompi_java.LongValueOf = (*env)->GetStaticMethodID(
|
|
env, ompi_java.LongClass, "valueOf", "(J)Ljava/lang/Long;");
|
|
}
|
|
|
|
static void deleteClasses(JNIEnv *env)
|
|
{
|
|
(*env)->DeleteGlobalRef(env, ompi_java.CartParmsClass);
|
|
(*env)->DeleteGlobalRef(env, ompi_java.ShiftParmsClass);
|
|
(*env)->DeleteGlobalRef(env, ompi_java.GraphParmsClass);
|
|
(*env)->DeleteGlobalRef(env, ompi_java.DistGraphNeighborsClass);
|
|
(*env)->DeleteGlobalRef(env, ompi_java.StatusClass);
|
|
(*env)->DeleteGlobalRef(env, ompi_java.ExceptionClass);
|
|
(*env)->DeleteGlobalRef(env, ompi_java.IntegerClass);
|
|
(*env)->DeleteGlobalRef(env, ompi_java.LongClass);
|
|
}
|
|
|
|
JNIEXPORT jobject JNICALL Java_mpi_MPI_newInt2(JNIEnv *env, jclass clazz)
|
|
{
|
|
struct { int a; int b; } s;
|
|
int iOff = (int)((MPI_Aint)(&(s.b)) - (MPI_Aint)(&(s.a)));
|
|
jclass c = (*env)->FindClass(env, "mpi/Int2");
|
|
jmethodID m = (*env)->GetMethodID(env, c, "<init>", "(II)V");
|
|
return (*env)->NewObject(env, c, m, iOff, sizeof(int));
|
|
}
|
|
|
|
JNIEXPORT jobject JNICALL Java_mpi_MPI_newShortInt(JNIEnv *env, jclass clazz)
|
|
{
|
|
struct { short a; int b; } s;
|
|
int iOff = (int)((MPI_Aint)(&(s.b)) - (MPI_Aint)(&(s.a)));
|
|
jclass c = (*env)->FindClass(env, "mpi/ShortInt");
|
|
jmethodID m = (*env)->GetMethodID(env, c, "<init>", "(III)V");
|
|
return (*env)->NewObject(env, c, m, sizeof(short), iOff, sizeof(int));
|
|
}
|
|
|
|
JNIEXPORT jobject JNICALL Java_mpi_MPI_newLongInt(JNIEnv *env, jclass clazz)
|
|
{
|
|
struct { long a; int b; } s;
|
|
int iOff = (int)((MPI_Aint)(&(s.b)) - (MPI_Aint)(&(s.a)));
|
|
jclass c = (*env)->FindClass(env, "mpi/LongInt");
|
|
jmethodID m = (*env)->GetMethodID(env, c, "<init>", "(III)V");
|
|
return (*env)->NewObject(env, c, m, sizeof(long), iOff, sizeof(int));
|
|
}
|
|
|
|
JNIEXPORT jobject JNICALL Java_mpi_MPI_newFloatInt(JNIEnv *env, jclass clazz)
|
|
{
|
|
struct { float a; int b; } s;
|
|
int iOff = (int)((MPI_Aint)(&(s.b)) - (MPI_Aint)(&(s.a)));
|
|
jclass c = (*env)->FindClass(env, "mpi/FloatInt");
|
|
jmethodID m = (*env)->GetMethodID(env, c, "<init>", "(II)V");
|
|
return (*env)->NewObject(env, c, m, iOff, sizeof(int));
|
|
}
|
|
|
|
JNIEXPORT jobject JNICALL Java_mpi_MPI_newDoubleInt(JNIEnv *env, jclass clazz)
|
|
{
|
|
struct { double a; int b; } s;
|
|
int iOff = (int)((MPI_Aint)(&(s.b)) - (MPI_Aint)(&(s.a)));
|
|
jclass c = (*env)->FindClass(env, "mpi/DoubleInt");
|
|
jmethodID m = (*env)->GetMethodID(env, c, "<init>", "(II)V");
|
|
return (*env)->NewObject(env, c, m, iOff, sizeof(int));
|
|
}
|
|
|
|
JNIEXPORT jobjectArray JNICALL Java_mpi_MPI_Init_1jni(
|
|
JNIEnv *env, jclass clazz, jobjectArray argv)
|
|
{
|
|
jsize i;
|
|
jclass string;
|
|
jobject value;
|
|
|
|
int len = (*env)->GetArrayLength(env, argv);
|
|
char **sargs = (char**)calloc(len+1, sizeof(char*));
|
|
|
|
for(i = 0; i < len; i++)
|
|
{
|
|
jstring jc = (jstring)(*env)->GetObjectArrayElement(env, argv, i);
|
|
const char *s = (*env)->GetStringUTFChars(env, jc, NULL);
|
|
sargs[i] = strdup(s);
|
|
(*env)->ReleaseStringUTFChars(env, jc, s);
|
|
(*env)->DeleteLocalRef(env, jc);
|
|
}
|
|
|
|
int rc = MPI_Init(&len, &sargs);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
mca_base_var_register("ompi", "mpi", "java", "eager",
|
|
"Java buffers eager size",
|
|
MCA_BASE_VAR_TYPE_INT, NULL, 0, 0,
|
|
OPAL_INFO_LVL_5,
|
|
MCA_BASE_VAR_SCOPE_READONLY,
|
|
&ompi_mpi_java_eager);
|
|
|
|
string = (*env)->FindClass(env, "java/lang/String");
|
|
value = (*env)->NewObjectArray(env, len, string, NULL);
|
|
|
|
for(i = 0; i < len; i++)
|
|
{
|
|
jstring jc = (*env)->NewStringUTF(env, sargs[i]);
|
|
(*env)->SetObjectArrayElement(env, value, i, jc);
|
|
(*env)->DeleteLocalRef(env, jc);
|
|
}
|
|
|
|
findClasses(env);
|
|
initFreeList();
|
|
return value;
|
|
}
|
|
|
|
JNIEXPORT jint JNICALL Java_mpi_MPI_InitThread_1jni(
|
|
JNIEnv *env, jclass clazz, jobjectArray argv, jint required)
|
|
{
|
|
jsize i;
|
|
int len = (*env)->GetArrayLength(env,argv);
|
|
char **sargs = (char**)calloc(len+1, sizeof(char*));
|
|
|
|
for(i = 0; i < len; i++)
|
|
{
|
|
jstring jc = (jstring)(*env)->GetObjectArrayElement(env, argv, i);
|
|
const char *s = (*env)->GetStringUTFChars(env, jc, 0);
|
|
sargs[i] = strdup(s);
|
|
(*env)->ReleaseStringUTFChars(env, jc, s);
|
|
(*env)->DeleteLocalRef(env, jc);
|
|
}
|
|
|
|
int provided;
|
|
int rc = MPI_Init_thread(&len, &sargs, required, &provided);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
|
|
findClasses(env);
|
|
initFreeList();
|
|
return provided;
|
|
}
|
|
|
|
JNIEXPORT jint JNICALL Java_mpi_MPI_queryThread_1jni(JNIEnv *env, jclass clazz)
|
|
{
|
|
int provided;
|
|
int rc = MPI_Query_thread(&provided);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
return provided;
|
|
}
|
|
|
|
JNIEXPORT jboolean JNICALL Java_mpi_MPI_isThreadMain_1jni(
|
|
JNIEnv *env, jclass clazz)
|
|
{
|
|
int flag;
|
|
int rc = MPI_Is_thread_main(&flag);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
return flag ? JNI_TRUE : JNI_FALSE;
|
|
}
|
|
|
|
JNIEXPORT void JNICALL Java_mpi_MPI_Finalize_1jni(JNIEnv *env, jclass obj)
|
|
{
|
|
OBJ_DESTRUCT(&ompi_java_buffers);
|
|
int rc = MPI_Finalize();
|
|
ompi_java_exceptionCheck(env, rc);
|
|
deleteClasses(env);
|
|
}
|
|
|
|
JNIEXPORT jint JNICALL Java_mpi_MPI_getProcessorName(
|
|
JNIEnv *env, jclass obj, jbyteArray buf)
|
|
{
|
|
int len;
|
|
jbyte* bufc = (jbyte*)((*env)->GetByteArrayElements(env, buf, NULL));
|
|
int rc = MPI_Get_processor_name((char*)bufc, &len);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
(*env)->ReleaseByteArrayElements(env, buf, bufc, 0);
|
|
return len;
|
|
}
|
|
|
|
JNIEXPORT jdouble JNICALL Java_mpi_MPI_wtime_1jni(JNIEnv *env, jclass jthis)
|
|
{
|
|
return MPI_Wtime();
|
|
}
|
|
|
|
JNIEXPORT jdouble JNICALL Java_mpi_MPI_wtick_1jni(JNIEnv *env, jclass jthis)
|
|
{
|
|
return MPI_Wtick();
|
|
}
|
|
|
|
JNIEXPORT jboolean JNICALL Java_mpi_MPI_isInitialized(JNIEnv *env, jclass jthis)
|
|
{
|
|
int flag;
|
|
int rc = MPI_Initialized(&flag);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
return flag ? JNI_TRUE : JNI_FALSE;
|
|
}
|
|
|
|
JNIEXPORT jboolean JNICALL Java_mpi_MPI_isFinalized(JNIEnv *env, jclass jthis)
|
|
{
|
|
int flag;
|
|
int rc = MPI_Finalized(&flag);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
return flag ? JNI_TRUE : JNI_FALSE;
|
|
}
|
|
|
|
JNIEXPORT void JNICALL Java_mpi_MPI_attachBuffer_1jni(
|
|
JNIEnv *env, jclass jthis, jbyteArray buf)
|
|
{
|
|
int size=(*env)->GetArrayLength(env,buf);
|
|
jbyte* bufptr = (*env)->GetByteArrayElements(env, buf, NULL);
|
|
int rc = MPI_Buffer_attach(bufptr,size);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
}
|
|
|
|
JNIEXPORT void JNICALL Java_mpi_MPI_detachBuffer_1jni(
|
|
JNIEnv *env, jclass jthis, jbyteArray buf)
|
|
{
|
|
int size;
|
|
jbyte* bufptr;
|
|
int rc = MPI_Buffer_detach(&bufptr, &size);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
|
|
if(buf != NULL)
|
|
(*env)->ReleaseByteArrayElements(env,buf,bufptr,0);
|
|
}
|
|
|
|
void* ompi_java_getArrayCritical(void** bufBase, JNIEnv *env,
|
|
jobject buf, int offset)
|
|
{
|
|
*bufBase = (jbyte*)(*env)->GetPrimitiveArrayCritical(env, buf, NULL);
|
|
return ((jbyte*)*bufBase) + offset;
|
|
}
|
|
|
|
void* ompi_java_getDirectBufferAddress(JNIEnv *env, jobject buf)
|
|
{
|
|
/* Allow NULL buffers to send/recv 0 items as control messages. */
|
|
return buf == NULL ? NULL : (*env)->GetDirectBufferAddress(env, buf);
|
|
}
|
|
|
|
static int getTypeExtent(JNIEnv *env, MPI_Datatype type)
|
|
{
|
|
MPI_Aint lb, extent;
|
|
int rc = MPI_Type_get_extent(type, &lb, &extent);
|
|
ompi_java_exceptionCheck(env, rc);
|
|
int value = extent;
|
|
assert(((MPI_Aint)value) == extent);
|
|
return value;
|
|
}
|
|
|
|
static void getArrayRegion(JNIEnv *env, jobject buf, int baseType,
|
|
int offset, int length, void *ptr)
|
|
{
|
|
switch(baseType)
|
|
{
|
|
case 0:
|
|
break;
|
|
case 1:
|
|
(*env)->GetByteArrayRegion(env, buf, offset, length, ptr);
|
|
break;
|
|
case 2:
|
|
(*env)->GetCharArrayRegion(env, buf, offset / 2, length / 2, ptr);
|
|
break;
|
|
case 3:
|
|
(*env)->GetShortArrayRegion(env, buf, offset / 2, length / 2, ptr);
|
|
break;
|
|
case 4:
|
|
(*env)->GetBooleanArrayRegion(env, buf, offset, length, ptr);
|
|
break;
|
|
case 5:
|
|
(*env)->GetIntArrayRegion(env, buf, offset / 4, length / 4, ptr);
|
|
break;
|
|
case 6:
|
|
(*env)->GetLongArrayRegion(env, buf, offset / 8, length / 8, ptr);
|
|
break;
|
|
case 7:
|
|
(*env)->GetFloatArrayRegion(env, buf, offset / 4, length / 4, ptr);
|
|
break;
|
|
case 8:
|
|
(*env)->GetDoubleArrayRegion(env, buf, offset / 8, length / 8, ptr);
|
|
break;
|
|
case 9:
|
|
(*env)->GetByteArrayRegion(env, buf, offset, length, ptr);
|
|
break;
|
|
default:
|
|
assert(0);
|
|
}
|
|
}
|
|
|
|
static void setArrayRegion(JNIEnv *env, jobject buf, int baseType,
|
|
int offset, int length, void *ptr)
|
|
{
|
|
switch(baseType)
|
|
{
|
|
case 0:
|
|
break;
|
|
case 1:
|
|
(*env)->SetByteArrayRegion(env, buf, offset, length, ptr);
|
|
break;
|
|
case 2:
|
|
(*env)->SetCharArrayRegion(env, buf, offset / 2, length / 2, ptr);
|
|
break;
|
|
case 3:
|
|
(*env)->SetShortArrayRegion(env, buf, offset / 2, length / 2, ptr);
|
|
break;
|
|
case 4:
|
|
(*env)->SetBooleanArrayRegion(env, buf, offset, length, ptr);
|
|
break;
|
|
case 5:
|
|
(*env)->SetIntArrayRegion(env, buf, offset / 4, length / 4, ptr);
|
|
break;
|
|
case 6:
|
|
(*env)->SetLongArrayRegion(env, buf, offset / 8, length / 8, ptr);
|
|
break;
|
|
case 7:
|
|
(*env)->SetFloatArrayRegion(env, buf, offset / 4, length / 4, ptr);
|
|
break;
|
|
case 8:
|
|
(*env)->SetDoubleArrayRegion(env, buf, offset / 8, length / 8, ptr);
|
|
break;
|
|
case 9:
|
|
(*env)->SetByteArrayRegion(env, buf, offset, length, ptr);
|
|
break;
|
|
default:
|
|
assert(0);
|
|
}
|
|
}
|
|
|
|
static void* getBuffer(JNIEnv *env, ompi_java_buffer_t **item, int size)
|
|
{
|
|
if(size > ompi_mpi_java_eager)
|
|
{
|
|
*item = NULL;
|
|
return malloc(size);
|
|
}
|
|
else
|
|
{
|
|
opal_free_list_item_t *freeListItem;
|
|
freeListItem = opal_free_list_get (&ompi_java_buffers);
|
|
|
|
ompi_java_exceptionCheck(env, NULL == freeListItem ? OMPI_ERROR :
|
|
OMPI_SUCCESS);
|
|
|
|
*item = (ompi_java_buffer_t*)freeListItem;
|
|
return (*item)->buffer;
|
|
}
|
|
}
|
|
|
|
static void releaseBuffer(void *ptr, ompi_java_buffer_t *item)
|
|
{
|
|
if(item == NULL)
|
|
{
|
|
free(ptr);
|
|
}
|
|
else
|
|
{
|
|
assert(item->buffer == ptr);
|
|
opal_free_list_return (&ompi_java_buffers, (opal_free_list_item_t*)item);
|
|
}
|
|
}
|
|
|
|
static int getCountv(int *counts, int *displs, int size)
|
|
{
|
|
/* Maybe displs is not ordered. */
|
|
int i, max = 0;
|
|
|
|
for(i = 1; i < size; i++)
|
|
{
|
|
if(displs[max] < displs[i])
|
|
max = i;
|
|
}
|
|
|
|
return displs[max] * counts[max];
|
|
}
|
|
|
|
static void* getReadPtr(ompi_java_buffer_t **item, JNIEnv *env, jobject buf,
|
|
int offset, int count, MPI_Datatype type, int baseType)
|
|
{
|
|
int length = count * getTypeExtent(env, type);
|
|
void *ptr = getBuffer(env, item, length);
|
|
|
|
if(opal_datatype_is_contiguous_memory_layout(&type->super, count))
|
|
{
|
|
getArrayRegion(env, buf, baseType, offset, length, ptr);
|
|
}
|
|
else
|
|
{
|
|
void *inBuf, *inBase;
|
|
inBuf = ompi_java_getArrayCritical(&inBase, env, buf, offset);
|
|
|
|
int rc = opal_datatype_copy_content_same_ddt(
|
|
&type->super, count, ptr, inBuf);
|
|
|
|
ompi_java_exceptionCheck(env,
|
|
rc==OPAL_SUCCESS ? OMPI_SUCCESS : OMPI_ERROR);
|
|
|
|
(*env)->ReleasePrimitiveArrayCritical(env, buf, inBase, JNI_ABORT);
|
|
}
|
|
|
|
return ptr;
|
|
}
|
|
|
|
static void* getReadPtrRank(
|
|
ompi_java_buffer_t **item, JNIEnv *env, jobject buf, int offset,
|
|
int count, int size, int rank, MPI_Datatype type, int baseType)
|
|
{
|
|
int extent = getTypeExtent(env, type),
|
|
rLen = extent * count,
|
|
length = rLen * size,
|
|
rDispl = rLen * rank,
|
|
rOff = offset + rDispl;
|
|
void *ptr = getBuffer(env, item, length);
|
|
void *rPtr = (char*)ptr + rDispl;
|
|
|
|
if(opal_datatype_is_contiguous_memory_layout(&type->super, count))
|
|
{
|
|
getArrayRegion(env, buf, baseType, rOff, rLen, rPtr);
|
|
}
|
|
else
|
|
{
|
|
void *bufPtr, *bufBase;
|
|
bufPtr = ompi_java_getArrayCritical(&bufBase, env, buf, rOff);
|
|
|
|
int rc = opal_datatype_copy_content_same_ddt(
|
|
&type->super, count, rPtr, bufPtr);
|
|
|
|
ompi_java_exceptionCheck(env,
|
|
rc==OPAL_SUCCESS ? OMPI_SUCCESS : OMPI_ERROR);
|
|
|
|
(*env)->ReleasePrimitiveArrayCritical(env, buf, bufBase, JNI_ABORT);
|
|
}
|
|
|
|
return ptr;
|
|
}
|
|
|
|
static void* getReadPtrvRank(
|
|
ompi_java_buffer_t **item, JNIEnv *env, jobject buf,
|
|
int offset, int *counts, int *displs, int size,
|
|
int rank, MPI_Datatype type, int baseType)
|
|
{
|
|
int extent = getTypeExtent(env, type),
|
|
length = extent * getCountv(counts, displs, size);
|
|
void *ptr = getBuffer(env, item, length);
|
|
int rootOff = offset + extent * displs[rank];
|
|
|
|
if(opal_datatype_is_contiguous_memory_layout(&type->super, counts[rank]))
|
|
{
|
|
int rootLength = extent * counts[rank];
|
|
void *rootPtr = (char*)ptr + extent * displs[rank];
|
|
getArrayRegion(env, buf, baseType, rootOff, rootLength, rootPtr);
|
|
}
|
|
else
|
|
{
|
|
void *inBuf, *inBase;
|
|
inBuf = ompi_java_getArrayCritical(&inBase, env, buf, rootOff);
|
|
|
|
int rc = opal_datatype_copy_content_same_ddt(
|
|
&type->super, counts[rank], ptr, inBuf);
|
|
|
|
ompi_java_exceptionCheck(env,
|
|
rc==OPAL_SUCCESS ? OMPI_SUCCESS : OMPI_ERROR);
|
|
|
|
(*env)->ReleasePrimitiveArrayCritical(env, buf, inBase, JNI_ABORT);
|
|
}
|
|
|
|
return ptr;
|
|
}
|
|
|
|
static void* getReadPtrvAll(
|
|
ompi_java_buffer_t **item, JNIEnv *env, jobject buf,
|
|
int offset, int *counts, int *displs, int size,
|
|
MPI_Datatype type, int baseType)
|
|
{
|
|
int i,
|
|
extent = getTypeExtent(env, type),
|
|
length = extent * getCountv(counts, displs, size);
|
|
void *ptr = getBuffer(env, item, length);
|
|
|
|
if(opal_datatype_is_contiguous_memory_layout(&type->super, 2))
|
|
{
|
|
for(i = 0; i < size; i++)
|
|
{
|
|
int iOff = offset + extent * displs[i],
|
|
iLen = extent * counts[i];
|
|
void *iPtr = (char*)ptr + extent * displs[i];
|
|
getArrayRegion(env, buf, baseType, iOff, iLen, iPtr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
void *bufPtr, *bufBase;
|
|
bufPtr = ompi_java_getArrayCritical(&bufBase, env, buf, offset);
|
|
|
|
for(i = 0; i < size; i++)
|
|
{
|
|
int iOff = extent * displs[i];
|
|
char *iBuf = iOff + (char*)bufPtr,
|
|
*iPtr = iOff + (char*)ptr;
|
|
|
|
int rc = opal_datatype_copy_content_same_ddt(
|
|
&type->super, counts[i], iPtr, iBuf);
|
|
|
|
ompi_java_exceptionCheck(env,
|
|
rc==OPAL_SUCCESS ? OMPI_SUCCESS : OMPI_ERROR);
|
|
}
|
|
|
|
(*env)->ReleasePrimitiveArrayCritical(env, buf, bufBase, JNI_ABORT);
|
|
}
|
|
|
|
return ptr;
|
|
}
|
|
|
|
static void* getWritePtr(ompi_java_buffer_t **item, JNIEnv *env,
|
|
int count, MPI_Datatype type)
|
|
{
|
|
int extent = getTypeExtent(env, type),
|
|
length = count * extent;
|
|
|
|
return getBuffer(env, item, length);
|
|
}
|
|
|
|
static void* getWritePtrv(ompi_java_buffer_t **item, JNIEnv *env,
|
|
int *counts, int *displs, int size, MPI_Datatype type)
|
|
{
|
|
int extent = getTypeExtent(env, type),
|
|
count = getCountv(counts, displs, size),
|
|
length = extent * count;
|
|
|
|
return getBuffer(env, item, length);
|
|
}
|
|
|
|
void ompi_java_getReadPtr(
|
|
void **ptr, ompi_java_buffer_t **item, JNIEnv *env, jobject buf,
|
|
jboolean db, int offset, int count, MPI_Datatype type, int baseType)
|
|
{
|
|
if(buf == NULL || baseType == 0)
|
|
{
|
|
/* Allow NULL buffers to send/recv 0 items as control messages. */
|
|
*ptr = NULL;
|
|
*item = NULL;
|
|
}
|
|
else if(db)
|
|
{
|
|
assert(offset == 0);
|
|
*ptr = (*env)->GetDirectBufferAddress(env, buf);
|
|
*item = NULL;
|
|
}
|
|
else
|
|
{
|
|
*ptr = getReadPtr(item, env, buf, offset, count, type, baseType);
|
|
}
|
|
}
|
|
|
|
void ompi_java_getReadPtrRank(
|
|
void **ptr, ompi_java_buffer_t **item, JNIEnv *env,
|
|
jobject buf, jboolean db, int offset, int count, int size,
|
|
int rank, MPI_Datatype type, int baseType)
|
|
{
|
|
if(buf == NULL || baseType == 0)
|
|
{
|
|
/* Allow NULL buffers to send/recv 0 items as control messages. */
|
|
*ptr = NULL;
|
|
*item = NULL;
|
|
}
|
|
else if(db)
|
|
{
|
|
assert(offset == 0);
|
|
*ptr = (*env)->GetDirectBufferAddress(env, buf);
|
|
*item = NULL;
|
|
}
|
|
else
|
|
{
|
|
*ptr = getReadPtrRank(item, env, buf, offset, count,
|
|
size, rank, type, baseType);
|
|
}
|
|
}
|
|
|
|
void ompi_java_getReadPtrv(
|
|
void **ptr, ompi_java_buffer_t **item, JNIEnv *env,
|
|
jobject buf, jboolean db, int offset, int *counts, int *displs,
|
|
int size, int rank, MPI_Datatype type, int baseType)
|
|
{
|
|
if(buf == NULL)
|
|
{
|
|
/* Allow NULL buffers to send/recv 0 items as control messages. */
|
|
*ptr = NULL;
|
|
*item = NULL;
|
|
}
|
|
else if(db)
|
|
{
|
|
assert(offset == 0);
|
|
*ptr = (*env)->GetDirectBufferAddress(env, buf);
|
|
*item = NULL;
|
|
}
|
|
else if(rank == -1)
|
|
{
|
|
*ptr = getReadPtrvAll(item, env, buf, offset, counts,
|
|
displs, size, type, baseType);
|
|
}
|
|
else
|
|
{
|
|
*ptr = getReadPtrvRank(item, env, buf, offset, counts,
|
|
displs, size, rank, type, baseType);
|
|
}
|
|
}
|
|
|
|
void ompi_java_releaseReadPtr(
|
|
void *ptr, ompi_java_buffer_t *item, jobject buf, jboolean db)
|
|
{
|
|
if(!db && buf && ptr)
|
|
releaseBuffer(ptr, item);
|
|
}
|
|
|
|
void ompi_java_getWritePtr(
|
|
void **ptr, ompi_java_buffer_t **item, JNIEnv *env,
|
|
jobject buf, jboolean db, int count, MPI_Datatype type)
|
|
{
|
|
if(buf == NULL)
|
|
{
|
|
/* Allow NULL buffers to send/recv 0 items as control messages. */
|
|
*ptr = NULL;
|
|
*item = NULL;
|
|
}
|
|
else if(db)
|
|
{
|
|
*ptr = (*env)->GetDirectBufferAddress(env, buf);
|
|
*item = NULL;
|
|
}
|
|
else
|
|
{
|
|
*ptr = getWritePtr(item, env, count, type);
|
|
}
|
|
}
|
|
|
|
void ompi_java_getWritePtrv(
|
|
void **ptr, ompi_java_buffer_t **item, JNIEnv *env, jobject buf,
|
|
jboolean db, int *counts, int *displs, int size, MPI_Datatype type)
|
|
{
|
|
if(buf == NULL)
|
|
{
|
|
/* Allow NULL buffers to send/recv 0 items as control messages. */
|
|
*ptr = NULL;
|
|
*item = NULL;
|
|
}
|
|
else if(db)
|
|
{
|
|
*ptr = (*env)->GetDirectBufferAddress(env, buf);
|
|
*item = NULL;
|
|
}
|
|
else
|
|
{
|
|
*ptr = getWritePtrv(item, env, counts, displs, size, type);
|
|
}
|
|
}
|
|
|
|
void ompi_java_releaseWritePtr(
|
|
void *ptr, ompi_java_buffer_t *item, JNIEnv *env, jobject buf,
|
|
jboolean db, int offset, int count, MPI_Datatype type, int baseType)
|
|
{
|
|
if(db || !buf || !ptr)
|
|
return;
|
|
|
|
if(opal_datatype_is_contiguous_memory_layout(&type->super, count))
|
|
{
|
|
int length = count * getTypeExtent(env, type);
|
|
setArrayRegion(env, buf, baseType, offset, length, ptr);
|
|
}
|
|
else
|
|
{
|
|
void *inBuf, *inBase;
|
|
inBuf = ompi_java_getArrayCritical(&inBase, env, buf, offset);
|
|
|
|
int rc = opal_datatype_copy_content_same_ddt(
|
|
&type->super, count, inBuf, ptr);
|
|
|
|
ompi_java_exceptionCheck(env,
|
|
rc==OPAL_SUCCESS ? OMPI_SUCCESS : OMPI_ERROR);
|
|
|
|
(*env)->ReleasePrimitiveArrayCritical(env, buf, inBase, 0);
|
|
}
|
|
|
|
releaseBuffer(ptr, item);
|
|
}
|
|
|
|
void ompi_java_releaseWritePtrv(
|
|
void *ptr, ompi_java_buffer_t *item, JNIEnv *env,
|
|
jobject buf, jboolean db, int offset, int *counts, int *displs,
|
|
int size, MPI_Datatype type, int baseType)
|
|
{
|
|
if(db || !buf || !ptr)
|
|
return;
|
|
|
|
int i;
|
|
int extent = getTypeExtent(env, type);
|
|
|
|
if(opal_datatype_is_contiguous_memory_layout(&type->super, 2))
|
|
{
|
|
for(i = 0; i < size; i++)
|
|
{
|
|
int iOff = offset + extent * displs[i],
|
|
iLen = extent * counts[i];
|
|
void *iPtr = (char*)ptr + extent * displs[i];
|
|
setArrayRegion(env, buf, baseType, iOff, iLen, iPtr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
void *bufPtr, *bufBase;
|
|
bufPtr = ompi_java_getArrayCritical(&bufBase, env, buf, offset);
|
|
|
|
for(i = 0; i < size; i++)
|
|
{
|
|
int iOff = extent * displs[i];
|
|
char *iBuf = iOff + (char*)bufPtr,
|
|
*iPtr = iOff + (char*)ptr;
|
|
|
|
int rc = opal_datatype_copy_content_same_ddt(
|
|
&type->super, counts[i], iBuf, iPtr);
|
|
|
|
ompi_java_exceptionCheck(env,
|
|
rc==OPAL_SUCCESS ? OMPI_SUCCESS : OMPI_ERROR);
|
|
}
|
|
|
|
(*env)->ReleasePrimitiveArrayCritical(env, buf, bufBase, 0);
|
|
}
|
|
|
|
releaseBuffer(ptr, item);
|
|
}
|
|
|
|
jobject ompi_java_Integer_valueOf(JNIEnv *env, jint i)
|
|
{
|
|
return (*env)->CallStaticObjectMethod(env,
|
|
ompi_java.IntegerClass, ompi_java.IntegerValueOf, i);
|
|
}
|
|
|
|
jobject ompi_java_Long_valueOf(JNIEnv *env, jlong i)
|
|
{
|
|
return (*env)->CallStaticObjectMethod(env,
|
|
ompi_java.LongClass, ompi_java.LongValueOf, i);
|
|
}
|
|
|
|
void ompi_java_getIntArray(JNIEnv *env, jintArray array,
|
|
jint **jptr, int **cptr)
|
|
{
|
|
jint *jInts = (*env)->GetIntArrayElements(env, array, NULL);
|
|
*jptr = jInts;
|
|
|
|
if(sizeof(int) == sizeof(jint))
|
|
{
|
|
*cptr = (int*)jInts;
|
|
}
|
|
else
|
|
{
|
|
int i, length = (*env)->GetArrayLength(env, array);
|
|
int *cInts = calloc(length, sizeof(int));
|
|
|
|
for(i = 0; i < length; i++)
|
|
cInts[i] = jInts[i];
|
|
|
|
*cptr = cInts;
|
|
}
|
|
}
|
|
|
|
void ompi_java_releaseIntArray(JNIEnv *env, jintArray array,
|
|
jint *jptr, int *cptr)
|
|
{
|
|
if(jptr != cptr)
|
|
{
|
|
int i, length = (*env)->GetArrayLength(env, array);
|
|
|
|
for(i = 0; i < length; i++)
|
|
jptr[i] = cptr[i];
|
|
|
|
free(cptr);
|
|
}
|
|
|
|
(*env)->ReleaseIntArrayElements(env, array, jptr, 0);
|
|
}
|
|
|
|
void ompi_java_forgetIntArray(JNIEnv *env, jintArray array,
|
|
jint *jptr, int *cptr)
|
|
{
|
|
if(jptr != cptr)
|
|
free(cptr);
|
|
|
|
(*env)->ReleaseIntArrayElements(env, array, jptr, JNI_ABORT);
|
|
}
|
|
|
|
void ompi_java_getBooleanArray(JNIEnv *env, jbooleanArray array,
|
|
jboolean **jptr, int **cptr)
|
|
{
|
|
int i, length = (*env)->GetArrayLength(env, array);
|
|
jboolean *jb = (*env)->GetBooleanArrayElements(env, array, NULL);
|
|
int *cb = (int*)calloc(length, sizeof(int));
|
|
|
|
for(i = 0; i < length; i++)
|
|
cb[i] = jb[i];
|
|
|
|
*jptr = jb;
|
|
*cptr = cb;
|
|
}
|
|
|
|
void ompi_java_releaseBooleanArray(JNIEnv *env, jbooleanArray array,
|
|
jboolean *jptr, int *cptr)
|
|
{
|
|
int i, length = (*env)->GetArrayLength(env, array);
|
|
|
|
for(i = 0; i < length; i++)
|
|
jptr[i] = cptr[i] ? JNI_TRUE : JNI_FALSE;
|
|
|
|
free(cptr);
|
|
(*env)->ReleaseBooleanArrayElements(env, array, jptr, 0);
|
|
}
|
|
|
|
void ompi_java_forgetBooleanArray(JNIEnv *env, jbooleanArray array,
|
|
jboolean *jptr, int *cptr)
|
|
{
|
|
free(cptr);
|
|
(*env)->ReleaseBooleanArrayElements(env, array, jptr, JNI_ABORT);
|
|
}
|
|
|
|
void ompi_java_getPtrArray(JNIEnv *env, jlongArray array,
|
|
jlong **jptr, void ***cptr)
|
|
{
|
|
jlong *jp = *jptr = (*env)->GetLongArrayElements(env, array, NULL);
|
|
|
|
if(sizeof(jlong) == sizeof(void*))
|
|
{
|
|
*cptr = (void**)jp;
|
|
}
|
|
else
|
|
{
|
|
int i, length = (*env)->GetArrayLength(env, array);
|
|
void **cp = *cptr = calloc(length, sizeof(void*));
|
|
|
|
for(i = 0; i < length; i++)
|
|
cp[i] = (void*)jp[i];
|
|
}
|
|
}
|
|
|
|
void ompi_java_releasePtrArray(JNIEnv *env, jlongArray array,
|
|
jlong *jptr, void **cptr)
|
|
{
|
|
if(jptr != (jlong*)cptr)
|
|
{
|
|
int i, length = (*env)->GetArrayLength(env, array);
|
|
|
|
for(i = 0; i < length; i++)
|
|
jptr[i] = (jlong)cptr[i];
|
|
|
|
free(cptr);
|
|
}
|
|
|
|
(*env)->ReleaseLongArrayElements(env, array, jptr, 0);
|
|
}
|
|
|
|
jboolean ompi_java_exceptionCheck(JNIEnv *env, int rc)
|
|
{
|
|
if(MPI_SUCCESS == rc)
|
|
{
|
|
return JNI_FALSE;
|
|
}
|
|
else if((*env)->ExceptionCheck(env))
|
|
{
|
|
return JNI_TRUE;
|
|
}
|
|
else
|
|
{
|
|
int errClass = ompi_mpi_errcode_get_class(rc);
|
|
char *message = ompi_mpi_errnum_get_string(rc);
|
|
jstring jmessage = (*env)->NewStringUTF(env, (const char*)message);
|
|
|
|
jobject mpiex = (*env)->NewObject(env, ompi_java.ExceptionClass,
|
|
ompi_java.ExceptionInit,
|
|
rc, errClass, jmessage);
|
|
(*env)->Throw(env, mpiex);
|
|
(*env)->DeleteLocalRef(env, mpiex);
|
|
(*env)->DeleteLocalRef(env, jmessage);
|
|
return JNI_TRUE;
|
|
}
|
|
}
|
|
|
|
void* ompi_java_attrSet(JNIEnv *env, jbyteArray jval)
|
|
{
|
|
int length = (*env)->GetArrayLength(env, jval);
|
|
void *cval = malloc(sizeof(int) + length);
|
|
*((int*)cval) = length;
|
|
|
|
(*env)->GetByteArrayRegion(env, jval,
|
|
0, length, (jbyte*)cval + sizeof(int));
|
|
|
|
return cval;
|
|
}
|
|
|
|
jbyteArray ompi_java_attrGet(JNIEnv *env, void *cval)
|
|
{
|
|
int length = *((int*)cval);
|
|
jbyteArray jval = (*env)->NewByteArray(env, length);
|
|
|
|
(*env)->SetByteArrayRegion(env, jval,
|
|
0, length, (jbyte*)cval + sizeof(int));
|
|
|
|
return jval;
|
|
}
|
|
|
|
int ompi_java_attrCopy(void *attrValIn, void *attrValOut, int *flag)
|
|
{
|
|
int length = *((int*)attrValIn) + sizeof(int);
|
|
*((void**)attrValOut) = malloc(length);
|
|
memcpy(*((void**)attrValOut), attrValIn, length);
|
|
*flag = 1;
|
|
return MPI_SUCCESS;
|
|
}
|
|
|
|
int ompi_java_attrDelete(void *attrVal)
|
|
{
|
|
free(attrVal);
|
|
return MPI_SUCCESS;
|
|
}
|