In case you are right now having the same doubts as author of this post, I might have some good news for you. The following snippet of code was supposed to be part of my upcoming Weka toolbox for SPM8. It copies content of a 2-D full real MATLAB array into java.lang.Double[][] of the same dimensions given as the second argument. I ended up using another solution to pass my data around, but nevertheless, I found out how to correctly handle opaque Java objects from a MEX file using JNI. It took some debugging to discover, that the actual jobject handle is stored at (jobject*)(((char*) mxGetData(prhs[n]))+16), where n – index of your Java object argument. It means that you can use mxGetClassName(prhs[n]) to make sure that it is the expected Java type, in this case java.lang.Double[][] and then do any kind of manipulations using JNI. I didn’t discover what other secrets mxGetData(prhs[n]) holds, therefore I cannot tell you for example how to safely create new Java objects, wrap them in an mxArray and return to MATLAB, but I can propose this simple workaround. Pass a java.lang.Object[] array (or even java.util.Vector<Object>) as parameter to your MEX function and put all your result objects in there. That’it. Mystery solved. I’m pretty sure I will have some use for these findings in the future, maybe you will as well. Cheers!
PS. I tested on 64-bit MATLAB R2011b on Windows. It might be different in other versions but probably the address is the same or very near by.
24 | void mexFunction( int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[]) { |
25 | if (nrhs != 2 || mxGetNumberOfDimensions(prhs[0]) != 2 || |
26 | !mxIsNumeric(prhs[0]) || mxIsSparse(prhs[0]) || |
27 | mxIsComplex(prhs[0]) || !mxIsDouble(prhs[0]) || |
28 | strcmp (mxGetClassName(prhs[1]), "java.lang.Double[][]" ) || |
29 | mxGetM(prhs[0]) != mxGetM(prhs[1])) { |
30 | mexErrMsgTxt( "Usage: weka_fast_ml_array_to_java_array(src, dest), where src - 2D full real double array, dest - java.lang.Double[][] with the same dimensions" ); |
36 | JNI_GetCreatedJavaVMs(&vm, 1, (jsize*) &n); |
38 | mexErrMsgTxt( "Couldn't find existing Java VM" ); |
40 | vm->GetEnv(( void **) &env, JNI_VERSION_1_6); |
42 | mexErrMsgTxt( "Couldn't get Java environment" ); |
44 | char *d = ( char *) mxGetData(prhs[1]); |
45 | jobjectArray rows = *((jobjectArray*)(d + 16)); |
47 | size_t M = mxGetM(prhs[0]); |
48 | size_t N = mxGetN(prhs[0]); |
49 | const double *p = mxGetPr(prhs[0]); |
51 | jclass java_lang_Double = env->FindClass( "java/lang/Double" ); |
52 | jmethodID java_lang_Double_init_D = env->GetMethodID(java_lang_Double, "<init>" , "(D)V" ); |
54 | for ( int m = 0; m < M; m++) { |
55 | jobjectArray col = (jobjectArray) env->GetObjectArrayElement(rows, m); |
56 | if (env->GetArrayLength(col) != N) { |
57 | mexErrMsgTxt( "Wrong column length in destination" ); |
59 | for ( int n = 0; n < N; n++) { |
60 | jobject val = env->NewObject(java_lang_Double, java_lang_Double_init_D, *(p + m * N + n)); |
61 | env->SetObjectArrayElement(col, n, val); |
62 | env->DeleteLocalRef(val); |