/* * Copyright (C) 2011-2012 The Android Open Source Project * * 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. */ #define LOG_TAG "libRS_jni" #include <dlfcn.h> #include <fcntl.h> #include <math.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <android/bitmap.h> #include <android/log.h> #include <rsEnv.h> #include "rsDispatch.h" //#define LOG_API ALOG #define LOG_API(...) #define LOG_ERR(...) __android_log_print(ANDROID_LOG_ERROR, "RenderScript JNI", __VA_ARGS__); #define RS_JNI_VERSION 2301 #define NELEM(m) (sizeof(m) / sizeof((m)[0])) template <typename... T> void UNUSED(T... t) {} #define PER_ARRAY_TYPE(flag, fnc, readonly, ...) { \ jint len = 0; \ void *ptr = nullptr; \ void *srcPtr = nullptr; \ size_t typeBytes = 0; \ jint relFlag = 0; \ if (readonly) { \ /* The on-release mode should only be JNI_ABORT for read-only accesses. */ \ /* readonly = true, also indicates we are copying to the allocation . */ \ relFlag = JNI_ABORT; \ } \ switch(dataType) { \ case RS_TYPE_FLOAT_32: \ len = _env->GetArrayLength((jfloatArray)data); \ ptr = _env->GetFloatArrayElements((jfloatArray)data, flag); \ typeBytes = 4; \ if (usePadding) { \ srcPtr = ptr; \ len = len / 3 * 4; \ if (count == 0) { \ count = len / 4; \ } \ ptr = malloc (len * typeBytes); \ if (readonly) { \ copyWithPadding(ptr, srcPtr, mSize, count); \ fnc(__VA_ARGS__); \ } else { \ fnc(__VA_ARGS__); \ copyWithUnPadding(srcPtr, ptr, mSize, count); \ } \ free(ptr); \ ptr = srcPtr; \ } else { \ fnc(__VA_ARGS__); \ } \ _env->ReleaseFloatArrayElements((jfloatArray)data, (jfloat *)ptr, relFlag); \ return; \ case RS_TYPE_FLOAT_64: \ len = _env->GetArrayLength((jdoubleArray)data); \ ptr = _env->GetDoubleArrayElements((jdoubleArray)data, flag); \ typeBytes = 8; \ if (usePadding) { \ srcPtr = ptr; \ len = len / 3 * 4; \ if (count == 0) { \ count = len / 4; \ } \ ptr = malloc (len * typeBytes); \ if (readonly) { \ copyWithPadding(ptr, srcPtr, mSize, count); \ fnc(__VA_ARGS__); \ } else { \ fnc(__VA_ARGS__); \ copyWithUnPadding(srcPtr, ptr, mSize, count); \ } \ free(ptr); \ ptr = srcPtr; \ } else { \ fnc(__VA_ARGS__); \ } \ _env->ReleaseDoubleArrayElements((jdoubleArray)data, (jdouble *)ptr, relFlag); \ return; \ case RS_TYPE_SIGNED_8: \ case RS_TYPE_UNSIGNED_8: \ len = _env->GetArrayLength((jbyteArray)data); \ ptr = _env->GetByteArrayElements((jbyteArray)data, flag); \ typeBytes = 1; \ if (usePadding) { \ srcPtr = ptr; \ len = len / 3 * 4; \ if (count == 0) { \ count = len / 4; \ } \ ptr = malloc (len * typeBytes); \ if (readonly) { \ copyWithPadding(ptr, srcPtr, mSize, count); \ fnc(__VA_ARGS__); \ } else { \ fnc(__VA_ARGS__); \ copyWithUnPadding(srcPtr, ptr, mSize, count); \ } \ free(ptr); \ ptr = srcPtr; \ } else { \ fnc(__VA_ARGS__); \ } \ _env->ReleaseByteArrayElements((jbyteArray)data, (jbyte*)ptr, relFlag); \ return; \ case RS_TYPE_SIGNED_16: \ case RS_TYPE_UNSIGNED_16: \ len = _env->GetArrayLength((jshortArray)data); \ ptr = _env->GetShortArrayElements((jshortArray)data, flag); \ typeBytes = 2; \ if (usePadding) { \ srcPtr = ptr; \ len = len / 3 * 4; \ if (count == 0) { \ count = len / 4; \ } \ ptr = malloc (len * typeBytes); \ if (readonly) { \ copyWithPadding(ptr, srcPtr, mSize, count); \ fnc(__VA_ARGS__); \ } else { \ fnc(__VA_ARGS__); \ copyWithUnPadding(srcPtr, ptr, mSize, count); \ } \ free(ptr); \ ptr = srcPtr; \ } else { \ fnc(__VA_ARGS__); \ } \ _env->ReleaseShortArrayElements((jshortArray)data, (jshort *)ptr, relFlag); \ return; \ case RS_TYPE_SIGNED_32: \ case RS_TYPE_UNSIGNED_32: \ len = _env->GetArrayLength((jintArray)data); \ ptr = _env->GetIntArrayElements((jintArray)data, flag); \ typeBytes = 4; \ if (usePadding) { \ srcPtr = ptr; \ len = len / 3 * 4; \ if (count == 0) { \ count = len / 4; \ } \ ptr = malloc (len * typeBytes); \ if (readonly) { \ copyWithPadding(ptr, srcPtr, mSize, count); \ fnc(__VA_ARGS__); \ } else { \ fnc(__VA_ARGS__); \ copyWithUnPadding(srcPtr, ptr, mSize, count); \ } \ free(ptr); \ ptr = srcPtr; \ } else { \ fnc(__VA_ARGS__); \ } \ _env->ReleaseIntArrayElements((jintArray)data, (jint *)ptr, relFlag); \ return; \ case RS_TYPE_SIGNED_64: \ case RS_TYPE_UNSIGNED_64: \ len = _env->GetArrayLength((jlongArray)data); \ ptr = _env->GetLongArrayElements((jlongArray)data, flag); \ typeBytes = 8; \ if (usePadding) { \ srcPtr = ptr; \ len = len / 3 * 4; \ if (count == 0) { \ count = len / 4; \ } \ ptr = malloc (len * typeBytes); \ if (readonly) { \ copyWithPadding(ptr, srcPtr, mSize, count); \ fnc(__VA_ARGS__); \ } else { \ fnc(__VA_ARGS__); \ copyWithUnPadding(srcPtr, ptr, mSize, count); \ } \ free(ptr); \ ptr = srcPtr; \ } else { \ fnc(__VA_ARGS__); \ } \ _env->ReleaseLongArrayElements((jlongArray)data, (jlong *)ptr, relFlag); \ return; \ default: \ break; \ } \ UNUSED(len, ptr, srcPtr, typeBytes, relFlag); \ } class AutoJavaStringToUTF8 { public: AutoJavaStringToUTF8(JNIEnv* env, jstring str) : fEnv(env), fJStr(str) { fCStr = env->GetStringUTFChars(str, NULL); fLength = env->GetStringUTFLength(str); } ~AutoJavaStringToUTF8() { fEnv->ReleaseStringUTFChars(fJStr, fCStr); } const char* c_str() const { return fCStr; } jsize length() const { return fLength; } private: JNIEnv* fEnv; jstring fJStr; const char* fCStr; jsize fLength; }; class AutoJavaStringArrayToUTF8 { public: AutoJavaStringArrayToUTF8(JNIEnv* env, jobjectArray strings, jsize stringsLength) : mEnv(env), mStrings(strings), mStringsLength(stringsLength) { mCStrings = NULL; mSizeArray = NULL; if (stringsLength > 0) { mCStrings = (const char **)calloc(stringsLength, sizeof(char *)); mSizeArray = (size_t*)calloc(stringsLength, sizeof(size_t)); for (jsize ct = 0; ct < stringsLength; ct ++) { jstring s = (jstring)mEnv->GetObjectArrayElement(mStrings, ct); mCStrings[ct] = mEnv->GetStringUTFChars(s, NULL); mSizeArray[ct] = mEnv->GetStringUTFLength(s); } } } ~AutoJavaStringArrayToUTF8() { for (jsize ct=0; ct < mStringsLength; ct++) { jstring s = (jstring)mEnv->GetObjectArrayElement(mStrings, ct); mEnv->ReleaseStringUTFChars(s, mCStrings[ct]); } free(mCStrings); free(mSizeArray); } const char **c_str() const { return mCStrings; } size_t *c_str_len() const { return mSizeArray; } jsize length() const { return mStringsLength; } private: JNIEnv *mEnv; jobjectArray mStrings; const char **mCStrings; size_t *mSizeArray; jsize mStringsLength; }; // --------------------------------------------------------------------------- static dispatchTable dispatchTab; // Incremental Support lib static dispatchTable dispatchTabInc; static jboolean nLoadSO(JNIEnv *_env, jobject _this, jboolean useNative, jint targetApi, jstring libPath) { void* handle = NULL; if (useNative) { handle = dlopen("libRS.so", RTLD_LAZY | RTLD_LOCAL); } else { // For API 9+, dlopen the full path of libRSSupport. if (libPath != NULL) { const char * libPathJni = _env->GetStringUTFChars(libPath, JNI_FALSE); handle = dlopen(libPathJni, RTLD_LAZY | RTLD_LOCAL); _env->ReleaseStringUTFChars(libPath, libPathJni); } else { handle = dlopen("libRSSupport.so", RTLD_LAZY | RTLD_LOCAL); } } if (handle == NULL) { LOG_ERR("couldn't dlopen %s; librsjni version: %d", dlerror(), RS_JNI_VERSION); return false; } if (loadSymbols(handle, dispatchTab, targetApi) == false) { LOG_ERR("Dispatch table init failed! librsjni version: %d", RS_JNI_VERSION); dlclose(handle); return false; } LOG_API("Successfully loaded runtime"); return true; } static ioSuppDT ioDispatch; static jboolean nLoadIOSO(JNIEnv *_env, jobject _this) { void* handleIO = NULL; handleIO = dlopen("libRSSupportIO.so", RTLD_LAZY | RTLD_LOCAL); if (handleIO == NULL) { LOG_ERR("Couldn't load libRSSupportIO.so, librsjni version: %d", RS_JNI_VERSION); return false; } if (loadIOSuppSyms(handleIO, ioDispatch) == false) { LOG_ERR("libRSSupportIO init failed! librsjni version: %d", RS_JNI_VERSION); return false; } return true; } // --------------------------------------------------------------------------- static void copyWithPadding(void* ptr, void* srcPtr, int mSize, int count) { int sizeBytesPad = mSize * 4; int sizeBytes = mSize * 3; uint8_t *dst = static_cast<uint8_t *>(ptr); uint8_t *src = static_cast<uint8_t *>(srcPtr); for (int i = 0; i < count; i++) { memcpy(dst, src, sizeBytes); dst += sizeBytesPad; src += sizeBytes; } } static void copyWithUnPadding(void* ptr, void* srcPtr, int mSize, int count) { int sizeBytesPad = mSize * 4; int sizeBytes = mSize * 3; uint8_t *dst = static_cast<uint8_t *>(ptr); uint8_t *src = static_cast<uint8_t *>(srcPtr); for (int i = 0; i < count; i++) { memcpy(dst, src, sizeBytes); dst += sizeBytes; src += sizeBytesPad; } } // --------------------------------------------------------------------------- static void nContextFinish(JNIEnv *_env, jobject _this, jlong con) { LOG_API("nContextFinish, con(%p)", (RsContext)con); dispatchTab.ContextFinish((RsContext)con); } static jlong nClosureCreate(JNIEnv *_env, jobject _this, jlong con, jlong kernelID, jlong returnValue, jlongArray fieldIDArray, jlongArray valueArray, jintArray sizeArray, jlongArray depClosureArray, jlongArray depFieldIDArray) { jlong ret = 0; jlong* jFieldIDs = _env->GetLongArrayElements(fieldIDArray, nullptr); jsize fieldIDs_length = _env->GetArrayLength(fieldIDArray); jlong* jValues = _env->GetLongArrayElements(valueArray, nullptr); jsize values_length = _env->GetArrayLength(valueArray); jint* jSizes = _env->GetIntArrayElements(sizeArray, nullptr); jsize sizes_length = _env->GetArrayLength(sizeArray); jlong* jDepClosures = _env->GetLongArrayElements(depClosureArray, nullptr); jsize depClosures_length = _env->GetArrayLength(depClosureArray); jlong* jDepFieldIDs = _env->GetLongArrayElements(depFieldIDArray, nullptr); jsize depFieldIDs_length = _env->GetArrayLength(depFieldIDArray); size_t numValues, numDependencies; RsScriptFieldID* fieldIDs; RsClosure* depClosures; RsScriptFieldID* depFieldIDs; if (fieldIDs_length != values_length || values_length != sizes_length) { LOG_ERR("Unmatched field IDs, values, and sizes in closure creation."); goto exit; } numValues = (size_t)fieldIDs_length; if (depClosures_length != depFieldIDs_length) { LOG_ERR("Unmatched closures and field IDs for dependencies in closure creation."); goto exit; } numDependencies = (size_t)depClosures_length; if (numDependencies > numValues) { LOG_ERR("Unexpected number of dependencies in closure creation"); goto exit; } if (numValues > RS_CLOSURE_MAX_NUMBER_ARGS_AND_BINDINGS) { LOG_ERR("Too many arguments or globals in closure creation"); goto exit; } if (numValues > 0) { fieldIDs = (RsScriptFieldID*)alloca(sizeof(RsScriptFieldID) * numValues); if (fieldIDs == nullptr) { goto exit; } } else { // numValues == 0 // alloca(0) implementation is platform dependent fieldIDs = nullptr; } for (size_t i = 0; i < numValues; i++) { fieldIDs[i] = (RsScriptFieldID)jFieldIDs[i]; } if (numDependencies > 0) { depClosures = (RsClosure*)alloca(sizeof(RsClosure) * numDependencies); if (depClosures == nullptr) { goto exit; } for (size_t i = 0; i < numDependencies; i++) { depClosures[i] = (RsClosure)jDepClosures[i]; } depFieldIDs = (RsScriptFieldID*)alloca(sizeof(RsScriptFieldID) * numDependencies); if (depFieldIDs == nullptr) { goto exit; } for (size_t i = 0; i < numDependencies; i++) { depFieldIDs[i] = (RsClosure)jDepFieldIDs[i]; } } else { // numDependencies == 0 // alloca(0) implementation is platform dependent depClosures = nullptr; depFieldIDs = nullptr; } ret = (jlong)(uintptr_t)dispatchTab.ClosureCreate( (RsContext)con, (RsScriptKernelID)kernelID, (RsAllocation)returnValue, fieldIDs, numValues, jValues, numValues, (int*)jSizes, numValues, depClosures, numDependencies, depFieldIDs, numDependencies); exit: _env->ReleaseLongArrayElements(depFieldIDArray, jDepFieldIDs, JNI_ABORT); _env->ReleaseLongArrayElements(depClosureArray, jDepClosures, JNI_ABORT); _env->ReleaseIntArrayElements (sizeArray, jSizes, JNI_ABORT); _env->ReleaseLongArrayElements(valueArray, jValues, JNI_ABORT); _env->ReleaseLongArrayElements(fieldIDArray, jFieldIDs, JNI_ABORT); return ret; } static jlong nInvokeClosureCreate(JNIEnv *_env, jobject _this, jlong con, jlong invokeID, jbyteArray paramArray, jlongArray fieldIDArray, jlongArray valueArray, jintArray sizeArray) { jlong ret = 0; jbyte* jParams = _env->GetByteArrayElements(paramArray, nullptr); jsize jParamLength = _env->GetArrayLength(paramArray); jlong* jFieldIDs = _env->GetLongArrayElements(fieldIDArray, nullptr); jsize fieldIDs_length = _env->GetArrayLength(fieldIDArray); jlong* jValues = _env->GetLongArrayElements(valueArray, nullptr); jsize values_length = _env->GetArrayLength(valueArray); jint* jSizes = _env->GetIntArrayElements(sizeArray, nullptr); jsize sizes_length = _env->GetArrayLength(sizeArray); size_t numValues; RsScriptFieldID* fieldIDs; if (fieldIDs_length != values_length || values_length != sizes_length) { LOG_ERR("Unmatched field IDs, values, and sizes in closure creation."); goto exit; } numValues = (size_t) fieldIDs_length; if (numValues > RS_CLOSURE_MAX_NUMBER_ARGS_AND_BINDINGS) { LOG_ERR("Too many arguments or globals in closure creation"); goto exit; } fieldIDs = (RsScriptFieldID*)alloca(sizeof(RsScriptFieldID) * numValues); if (fieldIDs == nullptr) { goto exit; } for (size_t i = 0; i < numValues; i++) { fieldIDs[i] = (RsScriptFieldID)jFieldIDs[i]; } ret = (jlong)(uintptr_t)dispatchTab.InvokeClosureCreate( (RsContext)con, (RsScriptInvokeID)invokeID, jParams, jParamLength, fieldIDs, numValues, jValues, numValues, (int*)jSizes, numValues); exit: _env->ReleaseIntArrayElements (sizeArray, jSizes, JNI_ABORT); _env->ReleaseLongArrayElements(valueArray, jValues, JNI_ABORT); _env->ReleaseLongArrayElements(fieldIDArray, jFieldIDs, JNI_ABORT); _env->ReleaseByteArrayElements(paramArray, jParams, JNI_ABORT); return ret; } static void nClosureSetArg(JNIEnv *_env, jobject _this, jlong con, jlong closureID, jint index, jlong value, jint size) { // Size is signed with -1 indicating the values is an Allocation dispatchTab.ClosureSetArg((RsContext)con, (RsClosure)closureID, (uint32_t)index, (uintptr_t)value, size); } static void nClosureSetGlobal(JNIEnv *_env, jobject _this, jlong con, jlong closureID, jlong fieldID, jlong value, jint size) { // Size is signed with -1 indicating the values is an Allocation dispatchTab.ClosureSetGlobal((RsContext)con, (RsClosure)closureID, (RsScriptFieldID)fieldID, (int64_t)value, size); } static long nScriptGroup2Create(JNIEnv *_env, jobject _this, jlong con, jstring name, jstring cacheDir, jlongArray closureArray) { jlong ret = 0; AutoJavaStringToUTF8 nameUTF(_env, name); AutoJavaStringToUTF8 cacheDirUTF(_env, cacheDir); jlong* jClosures = _env->GetLongArrayElements(closureArray, nullptr); jsize numClosures = _env->GetArrayLength(closureArray); RsClosure* closures; if (numClosures > (jsize) RS_SCRIPT_GROUP_MAX_NUMBER_CLOSURES) { LOG_ERR("Too many closures in script group"); goto exit; } closures = (RsClosure*)alloca(sizeof(RsClosure) * numClosures); if (closures == nullptr) { goto exit; } for (int i = 0; i < numClosures; i++) { closures[i] = (RsClosure)jClosures[i]; } ret = (jlong)(uintptr_t)dispatchTab.ScriptGroup2Create( (RsContext)con, nameUTF.c_str(), nameUTF.length(), cacheDirUTF.c_str(), cacheDirUTF.length(), closures, numClosures); exit: _env->ReleaseLongArrayElements(closureArray, jClosures, JNI_ABORT); return ret; } static void nScriptGroup2Execute(JNIEnv *_env, jobject _this, jlong con, jlong groupID) { dispatchTab.ScriptGroupExecute((RsContext)con, (RsScriptGroup2)groupID); } static void nObjDestroy(JNIEnv *_env, jobject _this, jlong con, jlong obj) { LOG_API("nObjDestroy, con(%p) obj(%p)", (RsContext)con, (void *)obj); dispatchTab.ObjDestroy((RsContext)con, (void *)obj); } static void nScriptIntrinsicBLAS_Single(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong id, jint func, jint TransA, jint TransB, jint Side, jint Uplo, jint Diag, jint M, jint N, jint K, jfloat alpha, jlong A, jlong B, jfloat beta, jlong C, jint incX, jint incY, jint KL, jint KU, jboolean mUseInc) { RsBlasCall call; memset(&call, 0, sizeof(call)); call.func = (RsBlasFunction)func; call.transA = (RsBlasTranspose)TransA; call.transB = (RsBlasTranspose)TransB; call.side = (RsBlasSide)Side; call.uplo = (RsBlasUplo)Uplo; call.diag = (RsBlasDiag)Diag; call.M = M; call.N = N; call.K = K; call.alpha.f = alpha; call.beta.f = beta; call.incX = incX; call.incY = incY; call.KL = KL; call.KU = KU; RsAllocation in_allocs[3]; in_allocs[0] = (RsAllocation)A; in_allocs[1] = (RsAllocation)B; in_allocs[2] = (RsAllocation)C; if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEachMulti((RsContext)incCon, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } else { dispatchTab.ScriptForEachMulti((RsContext)con, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } } static void nScriptIntrinsicBLAS_Double(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong id, jint func, jint TransA, jint TransB, jint Side, jint Uplo, jint Diag, jint M, jint N, jint K, jdouble alpha, jlong A, jlong B, jdouble beta, jlong C, jint incX, jint incY, jint KL, jint KU, jboolean mUseInc) { RsBlasCall call; memset(&call, 0, sizeof(call)); call.func = (RsBlasFunction)func; call.transA = (RsBlasTranspose)TransA; call.transB = (RsBlasTranspose)TransB; call.side = (RsBlasSide)Side; call.uplo = (RsBlasUplo)Uplo; call.diag = (RsBlasDiag)Diag; call.M = M; call.N = N; call.K = K; call.alpha.d = alpha; call.beta.d = beta; call.incX = incX; call.incY = incY; call.KL = KL; call.KU = KU; RsAllocation in_allocs[3]; in_allocs[0] = (RsAllocation)A; in_allocs[1] = (RsAllocation)B; in_allocs[2] = (RsAllocation)C; if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEachMulti((RsContext)incCon, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } else { dispatchTab.ScriptForEachMulti((RsContext)con, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } } static void nScriptIntrinsicBLAS_Complex(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong id, jint func, jint TransA, jint TransB, jint Side, jint Uplo, jint Diag, jint M, jint N, jint K, jfloat alphaX, jfloat alphaY, jlong A, jlong B, jfloat betaX, jfloat betaY, jlong C, jint incX, jint incY, jint KL, jint KU, jboolean mUseInc) { RsBlasCall call; memset(&call, 0, sizeof(call)); call.func = (RsBlasFunction)func; call.transA = (RsBlasTranspose)TransA; call.transB = (RsBlasTranspose)TransB; call.side = (RsBlasSide)Side; call.uplo = (RsBlasUplo)Uplo; call.diag = (RsBlasDiag)Diag; call.M = M; call.N = N; call.K = K; call.alpha.c.r = alphaX; call.alpha.c.i = alphaY; call.beta.c.r = betaX; call.beta.c.i = betaY; call.incX = incX; call.incY = incY; call.KL = KL; call.KU = KU; RsAllocation in_allocs[3]; in_allocs[0] = (RsAllocation)A; in_allocs[1] = (RsAllocation)B; in_allocs[2] = (RsAllocation)C; if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEachMulti((RsContext)incCon, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } else { dispatchTab.ScriptForEachMulti((RsContext)con, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } } static void nScriptIntrinsicBLAS_Z(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong id, jint func, jint TransA, jint TransB, jint Side, jint Uplo, jint Diag, jint M, jint N, jint K, jdouble alphaX, jdouble alphaY, jlong A, jlong B, jdouble betaX, jdouble betaY, jlong C, jint incX, jint incY, jint KL, jint KU, jboolean mUseInc) { RsBlasCall call; memset(&call, 0, sizeof(call)); call.func = (RsBlasFunction)func; call.transA = (RsBlasTranspose)TransA; call.transB = (RsBlasTranspose)TransB; call.side = (RsBlasSide)Side; call.uplo = (RsBlasUplo)Uplo; call.diag = (RsBlasDiag)Diag; call.M = M; call.N = N; call.K = K; call.alpha.z.r = alphaX; call.alpha.z.i = alphaY; call.beta.z.r = betaX; call.beta.z.i = betaY; call.incX = incX; call.incY = incY; call.KL = KL; call.KU = KU; RsAllocation in_allocs[3]; in_allocs[0] = (RsAllocation)A; in_allocs[1] = (RsAllocation)B; in_allocs[2] = (RsAllocation)C; if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEachMulti((RsContext)incCon, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } else { dispatchTab.ScriptForEachMulti((RsContext)con, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } } static void nScriptIntrinsicBLAS_BNNM(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong id, jint M, jint N, jint K, jlong A, jint a_offset, jlong B, jint b_offset, jlong C, jint c_offset, jint c_mult_int, jboolean mUseInc) { RsBlasCall call; memset(&call, 0, sizeof(call)); call.func = RsBlas_bnnm; call.M = M; call.N = N; call.K = K; call.a_offset = a_offset & 0xFF; call.b_offset = b_offset & 0xFF; call.c_offset = c_offset; call.c_mult_int = c_mult_int; RsAllocation in_allocs[3]; in_allocs[0] = (RsAllocation)A; in_allocs[1] = (RsAllocation)B; in_allocs[2] = (RsAllocation)C; if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEachMulti((RsContext)incCon, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } else { dispatchTab.ScriptForEachMulti((RsContext)con, (RsScript)id, 0, in_allocs, NELEM(in_allocs), nullptr, &call, sizeof(call), nullptr, 0); } } // --------------------------------------------------------------------------- static jlong nDeviceCreate(JNIEnv *_env, jobject _this) { LOG_API("nDeviceCreate"); return (jlong)(uintptr_t)dispatchTab.DeviceCreate(); } static void nDeviceDestroy(JNIEnv *_env, jobject _this, jlong dev) { LOG_API("nDeviceDestroy"); return dispatchTab.DeviceDestroy((RsDevice)dev); } static void nDeviceSetConfig(JNIEnv *_env, jobject _this, jlong dev, jint p, jint value) { LOG_API("nDeviceSetConfig dev(%p), param(%i), value(%i)", (void *)dev, p, value); return dispatchTab.DeviceSetConfig((RsDevice)dev, (RsDeviceParam)p, value); } static jlong nContextCreate(JNIEnv *_env, jobject _this, jlong dev, jint ver, jint sdkVer, jint ct, jstring nativeLibDirJava) { LOG_API("nContextCreate"); // Access the NativeLibDir in the Java Context. const char * nativeLibDir = _env->GetStringUTFChars(nativeLibDirJava, JNI_FALSE); size_t length = (size_t)_env->GetStringUTFLength(nativeLibDirJava); jlong id = (jlong)(uintptr_t)dispatchTab.ContextCreate((RsDevice)dev, ver, sdkVer, (RsContextType)ct, 0); if (dispatchTab.SetNativeLibDir) { dispatchTab.SetNativeLibDir((RsContext)id, nativeLibDir, length); } _env->ReleaseStringUTFChars(nativeLibDirJava, nativeLibDir); return id; } static void nContextSetPriority(JNIEnv *_env, jobject _this, jlong con, jint p) { LOG_API("ContextSetPriority, con(%p), priority(%i)", (RsContext)con, p); dispatchTab.ContextSetPriority((RsContext)con, p); } static void nContextDestroy(JNIEnv *_env, jobject _this, jlong con) { LOG_API("nContextDestroy, con(%p)", (RsContext)con); dispatchTab.ContextDestroy((RsContext)con); } static void nContextDump(JNIEnv *_env, jobject _this, jlong con, jint bits) { LOG_API("nContextDump, con(%p) bits(%i)", (RsContext)con, bits); dispatchTab.ContextDump((RsContext)con, bits); } static jstring nContextGetErrorMessage(JNIEnv *_env, jobject _this, jlong con) { LOG_API("nContextGetErrorMessage, con(%p)", (RsContext)con); char buf[1024]; size_t receiveLen; uint32_t subID; int id = dispatchTab.ContextGetMessage((RsContext)con, buf, sizeof(buf), &receiveLen, sizeof(receiveLen), &subID, sizeof(subID)); if (!id && receiveLen) { // __android_log_print(ANDROID_LOG_VERBOSE, LOG_TAG, // "message receive buffer too small. %zu", receiveLen); } return _env->NewStringUTF(buf); } static jint nContextGetUserMessage(JNIEnv *_env, jobject _this, jlong con, jintArray data) { jint len = _env->GetArrayLength(data); LOG_API("nContextGetMessage, con(%p), len(%i)", (RsContext)con, len); jint *ptr = _env->GetIntArrayElements(data, NULL); size_t receiveLen; uint32_t subID; int id = dispatchTab.ContextGetMessage((RsContext)con, ptr, len * 4, &receiveLen, sizeof(receiveLen), &subID, sizeof(subID)); if (!id && receiveLen) { // __android_log_print(ANDROID_LOG_VERBOSE, LOG_TAG, // "message receive buffer too small. %zu", receiveLen); } _env->ReleaseIntArrayElements(data, ptr, 0); return (jint)id; } static jint nContextPeekMessage(JNIEnv *_env, jobject _this, jlong con, jintArray auxData) { LOG_API("nContextPeekMessage, con(%p)", (RsContext)con); jint *auxDataPtr = _env->GetIntArrayElements(auxData, NULL); size_t receiveLen; uint32_t subID; int id = dispatchTab.ContextPeekMessage((RsContext)con, &receiveLen, sizeof(receiveLen), &subID, sizeof(subID)); auxDataPtr[0] = (jint)subID; auxDataPtr[1] = (jint)receiveLen; _env->ReleaseIntArrayElements(auxData, auxDataPtr, 0); return (jint)id; } static void nContextInitToClient(JNIEnv *_env, jobject _this, jlong con) { LOG_API("nContextInitToClient, con(%p)", (RsContext)con); dispatchTab.ContextInitToClient((RsContext)con); } static void nContextDeinitToClient(JNIEnv *_env, jobject _this, jlong con) { LOG_API("nContextDeinitToClient, con(%p)", (RsContext)con); dispatchTab.ContextDeinitToClient((RsContext)con); } static void nContextSendMessage(JNIEnv *_env, jobject _this, jlong con, jint id, jintArray data) { jint *ptr = NULL; jint len = 0; if (data) { len = _env->GetArrayLength(data); ptr = _env->GetIntArrayElements(data, NULL); } LOG_API("nContextSendMessage, con(%p), id(%i), len(%i)", (RsContext)con, id, len); dispatchTab.ContextSendMessage((RsContext)con, id, (const uint8_t *)ptr, len * sizeof(int)); if (data) { _env->ReleaseIntArrayElements(data, ptr, JNI_ABORT); } } static jlong nElementCreate(JNIEnv *_env, jobject _this, jlong con, jlong type, jint kind, jboolean norm, jint size) { LOG_API("nElementCreate, con(%p), type(%i), kind(%i), norm(%i), size(%i)", (RsContext)con, type, kind, norm, size); return (jlong)(uintptr_t)dispatchTab.ElementCreate((RsContext)con, (RsDataType)type, (RsDataKind)kind, norm, size); } static jlong nElementCreate2(JNIEnv *_env, jobject _this, jlong con, jlongArray _ids, jobjectArray _names, jintArray _arraySizes) { int fieldCount = _env->GetArrayLength(_ids); LOG_API("nElementCreate2, con(%p)", (RsContext)con); jlong *jIds = _env->GetLongArrayElements(_ids, NULL); jint *jArraySizes = _env->GetIntArrayElements(_arraySizes, NULL); RsElement *ids = (RsElement*)malloc(fieldCount * sizeof(RsElement)); uint32_t *arraySizes = (uint32_t *)malloc(fieldCount * sizeof(uint32_t)); for(int i = 0; i < fieldCount; i ++) { ids[i] = (RsElement)jIds[i]; arraySizes[i] = (uint32_t)jArraySizes[i]; } AutoJavaStringArrayToUTF8 names(_env, _names, fieldCount); const char **nameArray = names.c_str(); size_t *sizeArray = names.c_str_len(); jlong id = (jlong)(uintptr_t)dispatchTab.ElementCreate2((RsContext)con, (RsElement *)ids, fieldCount, nameArray, fieldCount * sizeof(size_t), sizeArray, (const uint32_t *)arraySizes, fieldCount); free(ids); free(arraySizes); _env->ReleaseLongArrayElements(_ids, jIds, JNI_ABORT); _env->ReleaseIntArrayElements(_arraySizes, jArraySizes, JNI_ABORT); return id; } static void nElementGetSubElements(JNIEnv *_env, jobject _this, jlong con, jlong id, jlongArray _IDs, jobjectArray _names, jintArray _arraySizes) { uint32_t dataSize = _env->GetArrayLength(_IDs); LOG_API("nElementGetSubElements, con(%p)", (RsContext)con); uintptr_t *ids = (uintptr_t *)malloc(dataSize * sizeof(uintptr_t)); const char **names = (const char **)malloc((uint32_t)dataSize * sizeof(const char *)); size_t *arraySizes = (size_t *)malloc(dataSize * sizeof(size_t)); dispatchTab.ElementGetSubElements((RsContext)con, (RsElement)id, ids, names, arraySizes, (uint32_t)dataSize); for(uint32_t i = 0; i < dataSize; i++) { const jlong id = (jlong)(uintptr_t)ids[i]; const jint arraySize = (jint)arraySizes[i]; _env->SetObjectArrayElement(_names, i, _env->NewStringUTF(names[i])); _env->SetLongArrayRegion(_IDs, i, 1, &id); _env->SetIntArrayRegion(_arraySizes, i, 1, &arraySize); } free(ids); free(names); free(arraySizes); } // ----------------------------------- static jlong nTypeCreate(JNIEnv *_env, jobject _this, jlong con, jlong eid, jint dimx, jint dimy, jint dimz, jboolean mips, jboolean faces, jint yuv) { LOG_API("nTypeCreate, con(%p) eid(%p), x(%i), y(%i), z(%i), mips(%i), faces(%i), yuv(%i)", (RsContext)con, eid, dimx, dimy, dimz, mips, faces, yuv); return (jlong)(uintptr_t)dispatchTab.TypeCreate((RsContext)con, (RsElement)eid, dimx, dimy, dimz, mips, faces, yuv); } // ----------------------------------- static jlong nAllocationCreateTyped(JNIEnv *_env, jobject _this, jlong con, jlong type, jint mips, jint usage, jlong pointer) { LOG_API("nAllocationCreateTyped, con(%p), type(%p), mip(%i), usage(%i), ptr(%p)", (RsContext)con, (RsElement)type, mips, usage, (void *)pointer); return (jlong)(uintptr_t) dispatchTab.AllocationCreateTyped((RsContext)con, (RsType)type, (RsAllocationMipmapControl)mips, (uint32_t)usage, (uintptr_t)pointer); } static void nAllocationSyncAll(JNIEnv *_env, jobject _this, jlong con, jlong a, jint bits) { LOG_API("nAllocationSyncAll, con(%p), a(%p), bits(0x%08x)", (RsContext)con, (RsAllocation)a, bits); dispatchTab.AllocationSyncAll((RsContext)con, (RsAllocation)a, (RsAllocationUsageType)bits); } static void nAllocationSetSurface(JNIEnv *_env, jobject _this, jlong con, jlong alloc, jobject sur) { ioDispatch.sAllocationSetSurface(_env, _this, (RsContext)con, (RsAllocation)alloc, sur, dispatchTab); } static void nAllocationIoSend(JNIEnv *_env, jobject _this, jlong con, jlong alloc) { dispatchTab.AllocationIoSend((RsContext)con, (RsAllocation)alloc); } static void nAllocationGenerateMipmaps(JNIEnv *_env, jobject _this, jlong con, jlong alloc) { LOG_API("nAllocationGenerateMipmaps, con(%p), a(%p)", (RsContext)con, (RsAllocation)alloc); dispatchTab.AllocationGenerateMipmaps((RsContext)con, (RsAllocation)alloc); } static size_t GetBitmapSize(JNIEnv *env, jobject jbitmap) { AndroidBitmapInfo info; memset(&info, 0, sizeof(info)); AndroidBitmap_getInfo(env, jbitmap, &info); size_t s = info.width * info.height; switch (info.format) { case ANDROID_BITMAP_FORMAT_RGBA_8888: s *= 4; break; case ANDROID_BITMAP_FORMAT_RGB_565: s *= 2; break; case ANDROID_BITMAP_FORMAT_RGBA_4444: s *= 2; break; } return s; } static jlong nAllocationCreateFromBitmap(JNIEnv *_env, jobject _this, jlong con, jlong type, jint mip, jobject jbitmap, jint usage) { jlong id = 0; void *pixels = NULL; AndroidBitmap_lockPixels(_env, jbitmap, &pixels); if (pixels != NULL) { id = (jlong)(uintptr_t)dispatchTab.AllocationCreateFromBitmap((RsContext)con, (RsType)type, (RsAllocationMipmapControl)mip, pixels, GetBitmapSize(_env, jbitmap), usage); AndroidBitmap_unlockPixels(_env, jbitmap); } return id; } static jlong nAllocationCreateBitmapBackedAllocation(JNIEnv *_env, jobject _this, jlong con, jlong type, jint mip, jobject jbitmap, jint usage) { jlong id = 0; void *pixels = NULL; AndroidBitmap_lockPixels(_env, jbitmap, &pixels); if (pixels != NULL) { id = (jlong)(uintptr_t)dispatchTab.AllocationCreateTyped((RsContext)con, (RsType)type, (RsAllocationMipmapControl)mip, (uint32_t)usage, (uintptr_t)pixels); AndroidBitmap_unlockPixels(_env, jbitmap); } return id; } static jlong nAllocationCubeCreateFromBitmap(JNIEnv *_env, jobject _this, jlong con, jlong type, jint mip, jobject jbitmap, jint usage) { void *pixels = NULL; AndroidBitmap_lockPixels(_env, jbitmap, &pixels); jlong id = 0; if (pixels != NULL) { id = (jlong)(uintptr_t)dispatchTab.AllocationCubeCreateFromBitmap((RsContext)con, (RsType)type, (RsAllocationMipmapControl)mip, pixels, GetBitmapSize(_env, jbitmap), usage); AndroidBitmap_unlockPixels(_env, jbitmap); } return id; } static void nAllocationCopyFromBitmap(JNIEnv *_env, jobject _this, jlong con, jlong alloc, jobject jbitmap) { AndroidBitmapInfo info; memset(&info, 0, sizeof(info)); AndroidBitmap_getInfo(_env, jbitmap, &info); void *pixels = NULL; AndroidBitmap_lockPixels(_env, jbitmap, &pixels); if (pixels != NULL) { dispatchTab.Allocation2DData((RsContext)con, (RsAllocation)alloc, 0, 0, 0, RS_ALLOCATION_CUBEMAP_FACE_POSITIVE_X, info.width, info.height, pixels, GetBitmapSize(_env, jbitmap), 0); AndroidBitmap_unlockPixels(_env, jbitmap); } } static void nAllocationCopyToBitmap(JNIEnv *_env, jobject _this, jlong con, jlong alloc, jobject jbitmap) { AndroidBitmapInfo info; memset(&info, 0, sizeof(info)); AndroidBitmap_getInfo(_env, jbitmap, &info); void *pixels = NULL; AndroidBitmap_lockPixels(_env, jbitmap, &pixels); if (pixels != NULL) { dispatchTab.AllocationCopyToBitmap((RsContext)con, (RsAllocation)alloc, pixels, GetBitmapSize(_env, jbitmap)); AndroidBitmap_unlockPixels(_env, jbitmap); } //bitmap.notifyPixelsChanged(); } // Copies from the Java object data into the Allocation pointed to by _alloc. static void nAllocationData1D(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jint offset, jint lod, jint count, jobject data, jint sizeBytes, jint dataType, jint mSize, jboolean usePadding) { RsAllocation *alloc = (RsAllocation *)_alloc; LOG_API("nAllocation1DData, con(%p), adapter(%p), offset(%i), count(%i), sizeBytes(%i), " "dataType(%i)", (RsContext)con, (RsAllocation)alloc, offset, count, sizeBytes, dataType); PER_ARRAY_TYPE(nullptr, dispatchTab.Allocation1DData, true, (RsContext)con, alloc, offset, lod, count, ptr, sizeBytes); } static void nAllocationElementData1D(JNIEnv *_env, jobject _this, jlong con, jlong alloc, jint xoff, jint lod, jint compIdx, jbyteArray data, jint sizeBytes) { LOG_API("nAllocationElementData1D, con(%p), alloc(%p), xoff(%i), comp(%i), len(%i), " "sizeBytes(%i)", (RsContext)con, (RsAllocation)alloc, xoff, compIdx, _env->GetArrayLength(data), sizeBytes); jbyte *ptr = _env->GetByteArrayElements(data, nullptr); dispatchTab.Allocation1DElementData((RsContext)con, (RsAllocation)alloc, xoff, lod, ptr, sizeBytes, compIdx); _env->ReleaseByteArrayElements(data, ptr, JNI_ABORT); } /* static void nAllocationElementData(JNIEnv *_env, jobject _this, jlong con, jlong alloc, jint xoff, jint yoff, jint zoff, jint lod, jint compIdx, jbyteArray data, jint sizeBytes) { jint len = _env->GetArrayLength(data); LOG_API("nAllocationElementData, con(%p), alloc(%p), xoff(%i), yoff(%i), zoff(%i), comp(%i), len(%i), " "sizeBytes(%i)", (RsContext)con, (RsAllocation)alloc, xoff, yoff, zoff, compIdx, len, sizeBytes); jbyte *ptr = _env->GetByteArrayElements(data, nullptr); dispatchTab.AllocationElementData((RsContext)con, (RsAllocation)alloc, xoff, yoff, zoff, lod, ptr, sizeBytes, compIdx); _env->ReleaseByteArrayElements(data, ptr, JNI_ABORT); } */ // Copies from the Java object data into the Allocation pointed to by _alloc. static void nAllocationData2D(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jint xoff, jint yoff, jint lod, jint _face, jint w, jint h, jobject data, jint sizeBytes, jint dataType, jint mSize, jboolean usePadding) { RsAllocation *alloc = (RsAllocation *)_alloc; RsAllocationCubemapFace face = (RsAllocationCubemapFace)_face; LOG_API("nAllocation2DData, con(%p), adapter(%p), xoff(%i), yoff(%i), w(%i), h(%i), len(%i) " "type(%i)", (RsContext)con, alloc, xoff, yoff, w, h, sizeBytes, dataType); int count = w * h; PER_ARRAY_TYPE(nullptr, dispatchTab.Allocation2DData, true, (RsContext)con, alloc, xoff, yoff, lod, face, w, h, ptr, sizeBytes, 0); } static void nAllocationData2D_alloc(JNIEnv *_env, jobject _this, jlong con, jlong dstAlloc, jint dstXoff, jint dstYoff, jint dstMip, jint dstFace, jint width, jint height, jlong srcAlloc, jint srcXoff, jint srcYoff, jint srcMip, jint srcFace) { LOG_API("nAllocation2DData_s, con(%p), dstAlloc(%p), dstXoff(%i), dstYoff(%i)," " dstMip(%i), dstFace(%i), width(%i), height(%i)," " srcAlloc(%p), srcXoff(%i), srcYoff(%i), srcMip(%i), srcFace(%i)", (RsContext)con, (RsAllocation)dstAlloc, dstXoff, dstYoff, dstMip, dstFace, width, height, (RsAllocation)srcAlloc, srcXoff, srcYoff, srcMip, srcFace); dispatchTab.AllocationCopy2DRange((RsContext)con, (RsAllocation)dstAlloc, dstXoff, dstYoff, dstMip, dstFace, width, height, (RsAllocation)srcAlloc, srcXoff, srcYoff, srcMip, srcFace); } // Copies from the Java object data into the Allocation pointed to by _alloc. static void nAllocationData3D(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jint xoff, jint yoff, jint zoff, jint lod, jint w, jint h, jint d, jobject data, jint sizeBytes, jint dataType, jint mSize, jboolean usePadding) { RsAllocation *alloc = (RsAllocation *)_alloc; LOG_API("nAllocation3DData, con(%p), alloc(%p), xoff(%i), yoff(%i), zoff(%i), lod(%i), w(%i)," " h(%i), d(%i), sizeBytes(%i)", (RsContext)con, (RsAllocation)alloc, xoff, yoff, zoff, lod, w, h, d, sizeBytes); int count = w * h * d; PER_ARRAY_TYPE(nullptr, dispatchTab.Allocation3DData, true, (RsContext)con, alloc, xoff, yoff, zoff, lod, w, h, d, ptr, sizeBytes, 0); } static void nAllocationData3D_alloc(JNIEnv *_env, jobject _this, jlong con, jlong dstAlloc, jint dstXoff, jint dstYoff, jint dstZoff, jint dstMip, jint width, jint height, jint depth, jlong srcAlloc, jint srcXoff, jint srcYoff, jint srcZoff, jint srcMip) { LOG_API("nAllocationData3D_alloc, con(%p), dstAlloc(%p), dstXoff(%i), dstYoff(%i)," " dstMip(%i), width(%i), height(%i)," " srcAlloc(%p), srcXoff(%i), srcYoff(%i), srcMip(%i)", (RsContext)con, (RsAllocation)dstAlloc, dstXoff, dstYoff, dstMip, dstFace, width, height, (RsAllocation)srcAlloc, srcXoff, srcYoff, srcMip, srcFace); dispatchTab.AllocationCopy3DRange((RsContext)con, (RsAllocation)dstAlloc, dstXoff, dstYoff, dstZoff, dstMip, width, height, depth, (RsAllocation)srcAlloc, srcXoff, srcYoff, srcZoff, srcMip); } // Copies from the Allocation pointed to by _alloc into the Java object data. static void nAllocationRead(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jobject data, jint dataType, jint mSize, jboolean usePadding) { RsAllocation *alloc = (RsAllocation *)_alloc; LOG_API("nAllocationRead, con(%p), alloc(%p)", (RsContext)con, (RsAllocation)alloc); int count = 0; PER_ARRAY_TYPE(0, dispatchTab.AllocationRead, false, (RsContext)con, alloc, ptr, len * typeBytes); } // Copies from the Allocation pointed to by _alloc into the Java object data. static void nAllocationRead1D(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jint offset, jint lod, jint count, jobject data, jint sizeBytes, jint dataType, jint mSize, jboolean usePadding) { RsAllocation *alloc = (RsAllocation *)_alloc; LOG_API("nAllocation1DRead, con(%p), adapter(%p), offset(%i), count(%i), sizeBytes(%i), " "dataType(%i)", (RsContext)con, alloc, offset, count, sizeBytes, dataType); PER_ARRAY_TYPE(0, dispatchTab.Allocation1DRead, false, (RsContext)con, alloc, offset, lod, count, ptr, sizeBytes); } // Copies from the Element in the Allocation pointed to by _alloc into the Java array data. /* static void nAllocationElementRead(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jint xoff, jint yoff, jint zoff, jint lod, jint compIdx, jobject data, jint sizeBytes) { jint len = _env->GetArrayLength(data); LOG_API("nAllocationElementRead, con(%p), alloc(%p), xoff(%i), yoff(%i), zoff(%i), comp(%i), len(%i), " "sizeBytes(%i)", (RsContext)con, (RsAllocation)alloc, xoff, yoff, zoff, compIdx, len, sizeBytes); jbyte *ptr = _env->GetByteArrayElements(data, nullptr); dispatchTab.AllocationElementRead((RsContext)con, (RsAllocation)alloc, xoff, yoff, zoff, lod, ptr, sizeBytes, compIdx); _env->ReleaseByteArrayElements(data, ptr, JNI_ABORT); } */ // Copies from the Allocation pointed to by _alloc into the Java object data. static void nAllocationRead2D(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jint xoff, jint yoff, jint lod, jint _face, jint w, jint h, jobject data, jint sizeBytes, jint dataType, jint mSize, jboolean usePadding) { RsAllocation *alloc = (RsAllocation *)_alloc; RsAllocationCubemapFace face = (RsAllocationCubemapFace)_face; LOG_API("nAllocation2DRead, con(%p), adapter(%p), xoff(%i), yoff(%i), w(%i), h(%i), len(%i) " "type(%i)", (RsContext)con, alloc, xoff, yoff, w, h, sizeBytes, dataType); int count = w * h; PER_ARRAY_TYPE(0, dispatchTab.Allocation2DRead, false, (RsContext)con, alloc, xoff, yoff, lod, face, w, h, ptr, sizeBytes, 0); } // Copies from the Allocation pointed to by _alloc into the Java object data. /* static void nAllocationRead3D(JNIEnv *_env, jobject _this, jlong con, jlong _alloc, jint xoff, jint yoff, jint zoff, jint lod, jint w, jint h, jint d, jobject data, int sizeBytes, int dataType, jint mSize, jboolean usePadding) { RsAllocation *alloc = (RsAllocation *)_alloc; LOG_API("nAllocation3DRead, con(%p), alloc(%p), xoff(%i), yoff(%i), zoff(%i), lod(%i), w(%i)," " h(%i), d(%i), sizeBytes(%i)", (RsContext)con, (RsAllocation)alloc, xoff, yoff, zoff, lod, w, h, d, sizeBytes); int count = w * h * d; PER_ARRAY_TYPE(nullptr, dispatchTab.Allocation3DRead, false, (RsContext)con, alloc, xoff, yoff, zoff, lod, w, h, d, ptr, sizeBytes, 0); } */ static jlong nAllocationGetType(JNIEnv *_env, jobject _this, jlong con, jlong a) { LOG_API("nAllocationGetType, con(%p), a(%p)", (RsContext)con, (RsAllocation)a); return (jlong)(uintptr_t) dispatchTab.AllocationGetType((RsContext)con, (RsAllocation)a); } static void nAllocationResize1D(JNIEnv *_env, jobject _this, jlong con, jlong alloc, jint dimX) { LOG_API("nAllocationResize1D, con(%p), alloc(%p), sizeX(%i)", (RsContext)con, (RsAllocation)alloc, dimX); dispatchTab.AllocationResize1D((RsContext)con, (RsAllocation)alloc, dimX); } // ----------------------------------- static void nScriptBindAllocation(JNIEnv *_env, jobject _this, jlong con, jlong script, jlong alloc, jint slot, jboolean mUseInc) { LOG_API("nScriptBindAllocation, con(%p), script(%p), alloc(%p), slot(%i)", (RsContext)con, (RsScript)script, (RsAllocation)alloc, slot); if (mUseInc) { dispatchTabInc.ScriptBindAllocation((RsContext)con, (RsScript)script, (RsAllocation)alloc, slot); } else { dispatchTab.ScriptBindAllocation((RsContext)con, (RsScript)script, (RsAllocation)alloc, slot); } } static void nScriptSetVarI(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jint val, jboolean mUseInc) { LOG_API("nScriptSetVarI, con(%p), s(%p), slot(%i), val(%i)", (RsContext)con, (void *)script, slot, val); if (mUseInc) { dispatchTabInc.ScriptSetVarI((RsContext)con, (RsScript)script, slot, val); } else { dispatchTab.ScriptSetVarI((RsContext)con, (RsScript)script, slot, val); } } static void nScriptSetVarObj(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jlong val, jboolean mUseInc) { LOG_API("nScriptSetVarObj, con(%p), s(%p), slot(%i), val(%i)", (RsContext)con, (void *)script, slot, val); if (mUseInc) { dispatchTabInc.ScriptSetVarObj((RsContext)con, (RsScript)script, slot, (RsObjectBase)val); } else { dispatchTab.ScriptSetVarObj((RsContext)con, (RsScript)script, slot, (RsObjectBase)val); } } static void nScriptSetVarJ(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jlong val, jboolean mUseInc) { LOG_API("nScriptSetVarJ, con(%p), s(%p), slot(%i), val(%lli)", (RsContext)con, (void *)script, slot, val); if (mUseInc) { dispatchTabInc.ScriptSetVarJ((RsContext)con, (RsScript)script, slot, val); } else { dispatchTab.ScriptSetVarJ((RsContext)con, (RsScript)script, slot, val); } } static void nScriptSetVarF(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, float val, jboolean mUseInc) { LOG_API("nScriptSetVarF, con(%p), s(%p), slot(%i), val(%f)", (RsContext)con, (void *)script, slot, val); if (mUseInc) { dispatchTabInc.ScriptSetVarF((RsContext)con, (RsScript)script, slot, val); } else { dispatchTab.ScriptSetVarF((RsContext)con, (RsScript)script, slot, val); } } static void nScriptSetVarD(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, double val, jboolean mUseInc) { LOG_API("nScriptSetVarD, con(%p), s(%p), slot(%i), val(%lf)", (RsContext)con, (void *)script, slot, val); if (mUseInc) { dispatchTabInc.ScriptSetVarD((RsContext)con, (RsScript)script, slot, val); } else { dispatchTab.ScriptSetVarD((RsContext)con, (RsScript)script, slot, val); } } static void nScriptSetVarV(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jbyteArray data, jboolean mUseInc) { LOG_API("nScriptSetVarV, con(%p), s(%p), slot(%i)", (RsContext)con, (void *)script, slot); jint len = _env->GetArrayLength(data); jbyte *ptr = _env->GetByteArrayElements(data, NULL); if (mUseInc) { dispatchTabInc.ScriptSetVarV((RsContext)con, (RsScript)script, slot, ptr, len); } else { dispatchTab.ScriptSetVarV((RsContext)con, (RsScript)script, slot, ptr, len); } _env->ReleaseByteArrayElements(data, ptr, JNI_ABORT); } static void nScriptSetVarVE(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jbyteArray data, jlong elem, jintArray dims, jboolean mUseInc) { LOG_API("nScriptSetVarVE, con(%p), s(%p), slot(%i)", (RsContext)con, (void *)script, slot); jint len = _env->GetArrayLength(data); jbyte *ptr = _env->GetByteArrayElements(data, NULL); jint dimsLen = _env->GetArrayLength(dims) * sizeof(int); jint *dimsPtr = _env->GetIntArrayElements(dims, NULL); if (mUseInc) { dispatchTabInc.ScriptSetVarVE((RsContext)con, (RsScript)script, slot, ptr, len, (RsElement)elem, (const uint32_t *)dimsPtr, dimsLen); } else { dispatchTab.ScriptSetVarVE((RsContext)con, (RsScript)script, slot, ptr, len, (RsElement)elem, (const uint32_t *)dimsPtr, dimsLen); } _env->ReleaseByteArrayElements(data, ptr, JNI_ABORT); _env->ReleaseIntArrayElements(dims, dimsPtr, JNI_ABORT); } static void nScriptSetTimeZone(JNIEnv *_env, jobject _this, jlong con, jlong script, jbyteArray timeZone, jboolean mUseInc) { LOG_API("nScriptCSetTimeZone, con(%p), s(%p), timeZone(%s)", (RsContext)con, (void *)script, (const char *)timeZone); jint length = _env->GetArrayLength(timeZone); jbyte* timeZone_ptr; timeZone_ptr = (jbyte *) _env->GetPrimitiveArrayCritical(timeZone, (jboolean *)0); if (mUseInc) { dispatchTabInc.ScriptSetTimeZone((RsContext)con, (RsScript)script, (const char *)timeZone_ptr, length); } else { dispatchTab.ScriptSetTimeZone((RsContext)con, (RsScript)script, (const char *)timeZone_ptr, length); } if (timeZone_ptr) { _env->ReleasePrimitiveArrayCritical(timeZone, timeZone_ptr, 0); } } static void nScriptInvoke(JNIEnv *_env, jobject _this, jlong con, jlong obj, jint slot, jboolean mUseInc) { LOG_API("nScriptInvoke, con(%p), script(%p)", (RsContext)con, (void *)obj); if (mUseInc) { dispatchTabInc.ScriptInvoke((RsContext)con, (RsScript)obj, slot); } else { dispatchTab.ScriptInvoke((RsContext)con, (RsScript)obj, slot); } } static void nScriptInvokeV(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jbyteArray data, jboolean mUseInc) { LOG_API("nScriptInvokeV, con(%p), s(%p), slot(%i)", (RsContext)con, (void *)script, slot); jint len = _env->GetArrayLength(data); jbyte *ptr = _env->GetByteArrayElements(data, NULL); if (mUseInc) { dispatchTabInc.ScriptInvokeV((RsContext)con, (RsScript)script, slot, ptr, len); } else { dispatchTab.ScriptInvokeV((RsContext)con, (RsScript)script, slot, ptr, len); } _env->ReleaseByteArrayElements(data, ptr, JNI_ABORT); } static void nScriptForEach(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong script, jint slot, jlong ain, jlong aout, jboolean mUseInc) { LOG_API("nScriptForEach, con(%p), s(%p), slot(%i)", (RsContext)con, (void *)script, slot); if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEach((RsContext)incCon, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, NULL, 0, NULL, 0); } else { dispatchTab.ScriptForEach((RsContext)con, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, NULL, 0, NULL, 0); } } static void nScriptForEachV(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong script, jint slot, jlong ain, jlong aout, jbyteArray params, jboolean mUseInc) { LOG_API("nScriptForEach, con(%p), s(%p), slot(%i)", (RsContext)con, (void *)script, slot); jint len = _env->GetArrayLength(params); jbyte *ptr = _env->GetByteArrayElements(params, NULL); if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEach((RsContext)incCon, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, ptr, len, NULL, 0); } else { dispatchTab.ScriptForEach((RsContext)con, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, ptr, len, NULL, 0); } _env->ReleaseByteArrayElements(params, ptr, JNI_ABORT); } static void nScriptForEachClipped(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong script, jint slot, jlong ain, jlong aout, jint xstart, jint xend, jint ystart, jint yend, jint zstart, jint zend, jboolean mUseInc) { LOG_API("nScriptForEachClipped, con(%p), s(%p), slot(%i)", (RsContext)con, (void *)script, slot); RsScriptCall sc; sc.xStart = xstart; sc.xEnd = xend; sc.yStart = ystart; sc.yEnd = yend; sc.zStart = zstart; sc.zEnd = zend; sc.strategy = RS_FOR_EACH_STRATEGY_DONT_CARE; sc.arrayStart = 0; sc.arrayEnd = 0; sc.array2Start = 0; sc.array2End = 0; sc.array3Start = 0; sc.array3End = 0; sc.array4Start = 0; sc.array4End = 0; if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEach((RsContext)incCon, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, NULL, 0, &sc, sizeof(sc)); } else { dispatchTab.ScriptForEach((RsContext)con, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, NULL, 0, &sc, sizeof(sc)); } } static void nScriptForEachClippedV(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong script, jint slot, jlong ain, jlong aout, jbyteArray params, jint xstart, jint xend, jint ystart, jint yend, jint zstart, jint zend, jboolean mUseInc) { LOG_API("nScriptForEachClipped, con(%p), s(%p), slot(%i)", (RsContext)con, (void *)script, slot); jint len = _env->GetArrayLength(params); jbyte *ptr = _env->GetByteArrayElements(params, NULL); RsScriptCall sc; sc.xStart = xstart; sc.xEnd = xend; sc.yStart = ystart; sc.yEnd = yend; sc.zStart = zstart; sc.zEnd = zend; sc.strategy = RS_FOR_EACH_STRATEGY_DONT_CARE; sc.arrayStart = 0; sc.arrayEnd = 0; sc.array2Start = 0; sc.array2End = 0; sc.array3Start = 0; sc.array3End = 0; sc.array4Start = 0; sc.array4End = 0; if (mUseInc) { dispatchTab.ContextFinish((RsContext)con); dispatchTabInc.ScriptForEach((RsContext)incCon, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, ptr, len, &sc, sizeof(sc)); } else { dispatchTab.ScriptForEach((RsContext)con, (RsScript)script, slot, (RsAllocation)ain, (RsAllocation)aout, ptr, len, &sc, sizeof(sc)); } _env->ReleaseByteArrayElements(params, ptr, JNI_ABORT); } static void nScriptForEachMulti(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jlongArray ains, jlong aout, jbyteArray params, jintArray limits) { LOG_API("nScriptForEach, con(%p), s(%p), slot(%i) ains(%p) aout(%" PRId64 ")", (RsContext)con, (void *)script, slot, ains, aout); jint in_len = 0; jlong *in_ptr = nullptr; RsAllocation *in_allocs = nullptr; if (ains != nullptr) { in_len = _env->GetArrayLength(ains); if (in_len > (jint)RS_KERNEL_MAX_ARGUMENTS) { LOG_ERR("Too many arguments in kernel launch."); // TODO (b/20758983): Report back to Java and throw an exception return; } // TODO (b/20760800): Check in_ptr is not null in_ptr = _env->GetLongArrayElements(ains, nullptr); if (sizeof(RsAllocation) == sizeof(jlong)) { in_allocs = (RsAllocation*)in_ptr; } else { // Convert from 64-bit jlong types to the native pointer type. in_allocs = (RsAllocation*)alloca(in_len * sizeof(RsAllocation)); if (in_allocs == nullptr) { LOG_ERR("Failed launching kernel for lack of memory."); _env->ReleaseLongArrayElements(ains, in_ptr, JNI_ABORT); return; } for (int index = in_len; --index >= 0;) { in_allocs[index] = (RsAllocation)in_ptr[index]; } } } jint param_len = 0; jbyte *param_ptr = nullptr; if (params != nullptr) { param_len = _env->GetArrayLength(params); param_ptr = _env->GetByteArrayElements(params, nullptr); } RsScriptCall sc, *sca = nullptr; uint32_t sc_size = 0; jint limit_len = 0; jint *limit_ptr = nullptr; if (limits != nullptr) { limit_len = _env->GetArrayLength(limits); limit_ptr = _env->GetIntArrayElements(limits, nullptr); if (limit_len != 6) { LOG_ERR("LaunchOptions cannot be recognized."); goto exit; } sc.xStart = limit_ptr[0]; sc.xEnd = limit_ptr[1]; sc.yStart = limit_ptr[2]; sc.yEnd = limit_ptr[3]; sc.zStart = limit_ptr[4]; sc.zEnd = limit_ptr[5]; sc.strategy = RS_FOR_EACH_STRATEGY_DONT_CARE; sc.arrayStart = 0; sc.arrayEnd = 0; sc.array2Start = 0; sc.array2End = 0; sc.array3Start = 0; sc.array3End = 0; sc.array4Start = 0; sc.array4End = 0; sca = ≻ } dispatchTab.ScriptForEachMulti((RsContext)con, (RsScript)script, slot, in_allocs, in_len, (RsAllocation)aout, param_ptr, param_len, sca, sc_size); exit: if (ains != nullptr) { _env->ReleaseLongArrayElements(ains, in_ptr, JNI_ABORT); } if (params != nullptr) { _env->ReleaseByteArrayElements(params, param_ptr, JNI_ABORT); } if (limits != nullptr) { _env->ReleaseIntArrayElements(limits, limit_ptr, JNI_ABORT); } } static void nScriptReduce(JNIEnv *_env, jobject _this, jlong con, jlong script, jint slot, jlongArray ains, jlong aout, jintArray limits) { LOG_API("nScriptReduce, con(%p), s(%p), slot(%i) ains(%p) aout(%" PRId64 ")", (RsContext)con, (void *)script, slot, ains, aout); if (ains == nullptr) { LOG_ERR("At least one input required."); // TODO (b/20758983): Report back to Java and throw an exception return; } jint in_len = _env->GetArrayLength(ains); if (in_len > (jint)RS_KERNEL_MAX_ARGUMENTS) { LOG_ERR("Too many arguments in kernel launch."); // TODO (b/20758983): Report back to Java and throw an exception return; } jlong *in_ptr = _env->GetLongArrayElements(ains, nullptr); if (in_ptr == nullptr) { LOG_ERR("Failed to get Java array elements"); // TODO (b/20758983): Report back to Java and throw an exception return; } RsAllocation *in_allocs = nullptr; if (sizeof(RsAllocation) == sizeof(jlong)) { in_allocs = (RsAllocation*)in_ptr; } else { // Convert from 64-bit jlong types to the native pointer type. in_allocs = (RsAllocation*)alloca(in_len * sizeof(RsAllocation)); if (in_allocs == nullptr) { LOG_ERR("Failed launching kernel for lack of memory."); // TODO (b/20758983): Report back to Java and throw an exception _env->ReleaseLongArrayElements(ains, in_ptr, JNI_ABORT); return; } for (int index = in_len; --index >= 0;) { in_allocs[index] = (RsAllocation)in_ptr[index]; } } RsScriptCall sc, *sca = nullptr; uint32_t sc_size = 0; jint limit_len = 0; jint *limit_ptr = nullptr; if (limits != nullptr) { limit_len = _env->GetArrayLength(limits); limit_ptr = _env->GetIntArrayElements(limits, nullptr); if (limit_ptr == nullptr) { LOG_ERR("Failed to get Java array elements"); // TODO (b/20758983): Report back to Java and throw an exception _env->ReleaseLongArrayElements(ains, in_ptr, JNI_ABORT); return; } if (limit_len != 6) { LOG_ERR("LaunchOptions cannot be recognized"); // TODO (b/20758983): Report back to Java and throw an exception _env->ReleaseLongArrayElements(ains, in_ptr, JNI_ABORT); return; } sc.xStart = limit_ptr[0]; sc.xEnd = limit_ptr[1]; sc.yStart = limit_ptr[2]; sc.yEnd = limit_ptr[3]; sc.zStart = limit_ptr[4]; sc.zEnd = limit_ptr[5]; sc.strategy = RS_FOR_EACH_STRATEGY_DONT_CARE; sc.arrayStart = 0; sc.arrayEnd = 0; sc.array2Start = 0; sc.array2End = 0; sc.array3Start = 0; sc.array3End = 0; sc.array4Start = 0; sc.array4End = 0; sca = ≻ sc_size = sizeof(sc); } dispatchTab.ScriptReduce((RsContext)con, (RsScript)script, slot, in_allocs, in_len, (RsAllocation)aout, sca, sc_size); _env->ReleaseLongArrayElements(ains, in_ptr, JNI_ABORT); if (limits != nullptr) { _env->ReleaseIntArrayElements(limits, limit_ptr, JNI_ABORT); } } // ----------------------------------- static jlong nScriptCCreate(JNIEnv *_env, jobject _this, jlong con, jstring resName, jstring cacheDir, jbyteArray scriptRef, jint length) { LOG_API("nScriptCCreate, con(%p)", (RsContext)con); AutoJavaStringToUTF8 resNameUTF(_env, resName); AutoJavaStringToUTF8 cacheDirUTF(_env, cacheDir); jlong ret = 0; jbyte* script_ptr = NULL; jint _exception = 0; jint remaining; if (!scriptRef) { _exception = 1; //jniThrowException(_env, "java/lang/IllegalArgumentException", "script == null"); goto exit; } if (length < 0) { _exception = 1; //jniThrowException(_env, "java/lang/IllegalArgumentException", "length < 0"); goto exit; } remaining = _env->GetArrayLength(scriptRef); if (remaining < length) { _exception = 1; //jniThrowException(_env, "java/lang/IllegalArgumentException", // "length > script.length - offset"); goto exit; } script_ptr = (jbyte *) _env->GetPrimitiveArrayCritical(scriptRef, (jboolean *)0); //rsScriptCSetText(con, (const char *)script_ptr, length); ret = (jlong)(uintptr_t)dispatchTab.ScriptCCreate((RsContext)con, resNameUTF.c_str(), resNameUTF.length(), cacheDirUTF.c_str(), cacheDirUTF.length(), (const char *)script_ptr, length); exit: if (script_ptr) { _env->ReleasePrimitiveArrayCritical(scriptRef, script_ptr, _exception ? JNI_ABORT: 0); } return (jlong)(uintptr_t)ret; } static jlong nScriptIntrinsicCreate(JNIEnv *_env, jobject _this, jlong con, jint id, jlong eid, jboolean mUseInc) { LOG_API("nScriptIntrinsicCreate, con(%p) id(%i) element(%p)", (RsContext)con, id, (void *)eid); if (mUseInc) { return (jlong)(uintptr_t)dispatchTabInc.ScriptIntrinsicCreate((RsContext)con, id, (RsElement)eid); } else { return (jlong)(uintptr_t)dispatchTab.ScriptIntrinsicCreate((RsContext)con, id, (RsElement)eid); } } static jlong nScriptKernelIDCreate(JNIEnv *_env, jobject _this, jlong con, jlong sid, jint slot, jint sig, jboolean mUseInc) { LOG_API("nScriptKernelIDCreate, con(%p) script(%p), slot(%i), sig(%i)", (RsContext)con, (void *)sid, slot, sig); if (mUseInc) { return (jlong)(uintptr_t)dispatchTabInc.ScriptKernelIDCreate((RsContext)con, (RsScript)sid, slot, sig); } else { return (jlong)(uintptr_t)dispatchTab.ScriptKernelIDCreate((RsContext)con, (RsScript)sid, slot, sig); } } static jlong nScriptInvokeIDCreate(JNIEnv *_env, jobject _this, jlong con, jlong sid, jint slot) { LOG_API("nScriptInvokeIDCreate, con(%p) script(%p), slot(%i), sig(%i)", con, (void *)sid, slot); return (jlong)dispatchTab.ScriptInvokeIDCreate((RsContext)con, (RsScript)sid, slot); } static jlong nScriptFieldIDCreate(JNIEnv *_env, jobject _this, jlong con, jlong sid, jint slot, jboolean mUseInc) { LOG_API("nScriptFieldIDCreate, con(%p) script(%p), slot(%i)", (RsContext)con, (void *)sid, slot); if (mUseInc) { return (jlong)(uintptr_t)dispatchTabInc.ScriptFieldIDCreate((RsContext)con, (RsScript)sid, slot); } else { return (jlong)(uintptr_t)dispatchTab.ScriptFieldIDCreate((RsContext)con, (RsScript)sid, slot); } } static jlong nScriptGroupCreate(JNIEnv *_env, jobject _this, jlong con, jlongArray _kernels, jlongArray _src, jlongArray _dstk, jlongArray _dstf, jlongArray _types) { LOG_API("nScriptGroupCreate, con(%p)", (RsContext)con); jlong id = 0; RsScriptKernelID* kernelsPtr; jint kernelsLen = _env->GetArrayLength(_kernels); jlong *jKernelsPtr = _env->GetLongArrayElements(_kernels, nullptr); RsScriptKernelID* srcPtr; jint srcLen = _env->GetArrayLength(_src); jlong *jSrcPtr = _env->GetLongArrayElements(_src, nullptr); RsScriptKernelID* dstkPtr; jint dstkLen = _env->GetArrayLength(_dstk); jlong *jDstkPtr = _env->GetLongArrayElements(_dstk, nullptr); RsScriptKernelID* dstfPtr; jint dstfLen = _env->GetArrayLength(_dstf); jlong *jDstfPtr = _env->GetLongArrayElements(_dstf, nullptr); RsType* typesPtr; jint typesLen = _env->GetArrayLength(_types); jlong *jTypesPtr = _env->GetLongArrayElements(_types, nullptr); if (jKernelsPtr == nullptr) { LOG_ERR("Failed to get Java array elements: kernels"); goto cleanup; } if (jSrcPtr == nullptr) { LOG_ERR("Failed to get Java array elements: src"); goto cleanup; } if (jDstkPtr == nullptr) { LOG_ERR("Failed to get Java array elements: dstk"); goto cleanup; } if (jDstfPtr == nullptr) { LOG_ERR("Failed to get Java array elements: dstf"); goto cleanup; } if (jTypesPtr == nullptr) { LOG_ERR("Failed to get Java array elements: types"); goto cleanup; } kernelsPtr = (RsScriptKernelID*) malloc(sizeof(RsScriptKernelID) * kernelsLen); for(int i = 0; i < kernelsLen; ++i) { kernelsPtr[i] = (RsScriptKernelID)jKernelsPtr[i]; } srcPtr = (RsScriptKernelID*) malloc(sizeof(RsScriptKernelID) * srcLen); for(int i = 0; i < srcLen; ++i) { srcPtr[i] = (RsScriptKernelID)jSrcPtr[i]; } dstkPtr = (RsScriptKernelID*) malloc(sizeof(RsScriptKernelID) * dstkLen); for(int i = 0; i < dstkLen; ++i) { dstkPtr[i] = (RsScriptKernelID)jDstkPtr[i]; } dstfPtr = (RsScriptKernelID*) malloc(sizeof(RsScriptKernelID) * dstfLen); for(int i = 0; i < dstfLen; ++i) { dstfPtr[i] = (RsScriptKernelID)jDstfPtr[i]; } typesPtr = (RsType*) malloc(sizeof(RsType) * typesLen); for(int i = 0; i < typesLen; ++i) { typesPtr[i] = (RsType)jTypesPtr[i]; } id = (jlong)(uintptr_t) dispatchTab.ScriptGroupCreate((RsContext)con, (RsScriptKernelID *)kernelsPtr, kernelsLen * sizeof(RsScriptKernelID), (RsScriptKernelID *)srcPtr, srcLen * sizeof(RsScriptKernelID), (RsScriptKernelID *)dstkPtr, dstkLen * sizeof(RsScriptKernelID), (RsScriptFieldID *)dstfPtr, dstfLen * sizeof(RsScriptKernelID), (RsType *)typesPtr, typesLen * sizeof(RsType)); free(kernelsPtr); free(srcPtr); free(dstkPtr); free(dstfPtr); free(typesPtr); cleanup: if (jKernelsPtr != nullptr) { _env->ReleaseLongArrayElements(_kernels, jKernelsPtr, 0); } if (jSrcPtr != nullptr) { _env->ReleaseLongArrayElements(_src, jSrcPtr, 0); } if (jDstkPtr != nullptr) { _env->ReleaseLongArrayElements(_dstk, jDstkPtr, 0); } if (jDstfPtr != nullptr) { _env->ReleaseLongArrayElements(_dstf, jDstfPtr, 0); } if (jTypesPtr != nullptr) { _env->ReleaseLongArrayElements(_types, jTypesPtr, 0); } return id; } static void nScriptGroupSetInput(JNIEnv *_env, jobject _this, jlong con, jlong gid, jlong kid, jlong alloc) { LOG_API("nScriptGroupSetInput, con(%p) group(%p), kernelId(%p), alloc(%p)", (RsContext)con, (void *)gid, (void *)kid, (void *)alloc); dispatchTab.ScriptGroupSetInput((RsContext)con, (RsScriptGroup)gid, (RsScriptKernelID)kid, (RsAllocation)alloc); } static void nScriptGroupSetOutput(JNIEnv *_env, jobject _this, jlong con, jlong gid, jlong kid, jlong alloc) { LOG_API("nScriptGroupSetOutput, con(%p) group(%p), kernelId(%p), alloc(%p)", (RsContext)con, (void *)gid, (void *)kid, (void *)alloc); dispatchTab.ScriptGroupSetOutput((RsContext)con, (RsScriptGroup)gid, (RsScriptKernelID)kid, (RsAllocation)alloc); } static void nScriptGroupExecute(JNIEnv *_env, jobject _this, jlong con, jlong gid) { LOG_API("nScriptGroupSetOutput, con(%p) group(%p)", (RsContext)con, (void *)gid); dispatchTab.ScriptGroupExecute((RsContext)con, (RsScriptGroup)gid); } // --------------------------------------------------------------------------- static jlong nSamplerCreate(JNIEnv *_env, jobject _this, jlong con, jint magFilter, jint minFilter, jint wrapS, jint wrapT, jint wrapR, jfloat aniso) { LOG_API("nSamplerCreate, con(%p)", (RsContext)con); return (jlong)(uintptr_t)dispatchTab.SamplerCreate((RsContext)con, (RsSamplerValue)magFilter, (RsSamplerValue)minFilter, (RsSamplerValue)wrapS, (RsSamplerValue)wrapT, (RsSamplerValue)wrapR, aniso); } static jint nSystemGetPointerSize(JNIEnv *_env, jobject _this) { return (jint)sizeof(void*); } // --------------------------------------------------------------------------- // For Incremental Intrinsic Support static jboolean nIncLoadSO(JNIEnv *_env, jobject _this, jint deviceApi, jstring libPath) { void* handle = NULL; // For API 9+, dlopen the full path of libRSSupport. if (libPath != NULL) { const char * libPathJni = _env->GetStringUTFChars(libPath, JNI_FALSE); handle = dlopen(libPathJni, RTLD_LAZY | RTLD_LOCAL); _env->ReleaseStringUTFChars(libPath, libPathJni); } else { handle = dlopen("libRSSupport.so", RTLD_LAZY | RTLD_LOCAL); } if (handle == NULL) { LOG_ERR("couldn't dlopen %s; librsjni version: %d", dlerror(), RS_JNI_VERSION); return false; } if (loadSymbols(handle, dispatchTabInc, deviceApi) == false) { LOG_ERR("Dispatch Table init failed! librsjni version: %d", RS_JNI_VERSION); dlclose(handle); return false; } dispatchTabInc.AllocationCreateStrided = (AllocationCreateStridedFnPtr)dlsym(handle, "rsAllocationCreateStrided"); if (dispatchTabInc.AllocationCreateStrided == NULL) { LOG_ERR("Couldn't initialize dispatchTabInc.AllocationCreateStrided"); dlclose(handle); return false; } LOG_API("Successfully loaded compat runtime"); return true; } // ----------------------------------- // To create/destroy a dummy context static void nIncObjDestroy(JNIEnv *_env, jobject _this, jlong con, jlong obj) { LOG_API("nObjDestroy, con(%p) obj(%p)", (RsContext)con, (void *)obj); dispatchTabInc.ObjDestroy((RsContext)con, (void *)obj); } static jlong nIncDeviceCreate(JNIEnv *_env, jobject _this) { LOG_API("nDeviceCreate"); return (jlong)(uintptr_t)dispatchTabInc.DeviceCreate(); } static void nIncDeviceDestroy(JNIEnv *_env, jobject _this, jlong dev) { LOG_API("nDeviceDestroy"); return dispatchTabInc.DeviceDestroy((RsDevice)dev); } static jlong nIncContextCreate(JNIEnv *_env, jobject _this, jlong dev, jint ver, jint sdkVer, jint ct) { LOG_API("nContextCreate"); //The compat context for incremental support will be synchronous. return (jlong)(uintptr_t)dispatchTabInc.ContextCreate((RsDevice)dev, ver, sdkVer, (RsContextType)ct, RS_CONTEXT_SYNCHRONOUS); } static void nIncContextFinish(JNIEnv *_env, jobject _this, jlong con) { LOG_API("nContextFinish, con(%p)", (RsContext)con); dispatchTabInc.ContextFinish((RsContext)con); } static void nIncContextDestroy(JNIEnv *_env, jobject _this, jlong con) { LOG_API("nContextDestroy, con(%p)", (RsContext)con); dispatchTabInc.ContextDestroy((RsContext)con); } // ----------------------------------- // Create dummy Element static jlong nIncElementCreate(JNIEnv *_env, jobject _this, jlong con, jlong type, jint kind, jboolean norm, jint size) { LOG_API("nElementCreate, con(%p), type(%i), kind(%i), norm(%i), size(%i)", (RsContext)con, type, kind, norm, size); return (jlong)(uintptr_t)dispatchTabInc.ElementCreate((RsContext)con, (RsDataType)type, (RsDataKind)kind, norm, size); } // ----------------------------------- // Create dummy Type static jlong nIncTypeCreate(JNIEnv *_env, jobject _this, jlong con, jlong eid, jint dimx, jint dimy, jint dimz, jboolean mips, jboolean faces, jint yuv) { LOG_API("nTypeCreate, con(%p) eid(%p), x(%i), y(%i), z(%i), mips(%i), faces(%i), yuv(%i)", incCon, eid, dimx, dimy, dimz, mips, faces, yuv); return (jlong)(uintptr_t)dispatchTabInc.TypeCreate((RsContext)con, (RsElement)eid, dimx, dimy, dimz, mips, faces, yuv); } // ----------------------------------- // Create Allocation from pointer static jlong nIncAllocationCreateTyped(JNIEnv *_env, jobject _this, jlong con, jlong incCon, jlong alloc, jlong type, jint xBytesSize) { LOG_API("nAllocationCreateTyped, con(%p), type(%p), mip(%i), usage(%i), ptr(%p)", incCon, (RsElement)type, mips, usage, (void *)pointer); size_t strideIn; void* pIn = NULL; RsAllocation ainI = NULL; if (alloc != 0) { pIn = dispatchTab.AllocationGetPointer((RsContext)con, (RsAllocation)alloc, 0, RS_ALLOCATION_CUBEMAP_FACE_POSITIVE_X, 0, 0, &strideIn, sizeof(size_t)); /* * By definition stride is a roundup of xBytesSize with requiredAlignment, so requiredAlignment must * be strictly larger than the difference of (stride - xBytesSize). * * We can prove that as long as requiredAlignment satisfies the following two conditions, the * memory layout will be identical : * 1. Smaller or equal than stride; * 2. Larger than minRequiredAlignment. * * In this case we can simply choose the first power of 2 that satisfies both conditions. */ size_t requiredAlignment = 16; size_t minRequiredAlignment = strideIn - xBytesSize; while (requiredAlignment <= minRequiredAlignment) { requiredAlignment <<= 1; } ainI = dispatchTabInc.AllocationCreateStrided((RsContext)incCon, (RsType)type, RS_ALLOCATION_MIPMAP_NONE, RS_ALLOCATION_USAGE_INCREMENTAL_SUPPORT | RS_ALLOCATION_USAGE_SHARED, (uintptr_t)pIn, requiredAlignment); } return (jlong)(uintptr_t) ainI; } static jobject nAllocationGetByteBuffer(JNIEnv *_env, jobject _this, jlong con, jlong alloc, jint xBytesSize, jint dimY, jint dimZ) { LOG_API("nAllocationGetByteBuffer, con(%p), alloc(%p)", (RsContext)con, (RsAllocation)alloc); size_t strideIn = xBytesSize; void* ptr = NULL; if (alloc != 0 && dispatchTab.AllocationGetPointer != nullptr) { ptr = dispatchTab.AllocationGetPointer((RsContext)con, (RsAllocation)alloc, 0, RS_ALLOCATION_CUBEMAP_FACE_POSITIVE_X, dimZ, 0, &strideIn, sizeof(size_t)); } if (ptr != NULL) { size_t bufferSize = strideIn; if (dimY > 0) { bufferSize *= dimY; } if (dimZ > 0) { bufferSize *= dimZ; } jobject byteBuffer = _env->NewDirectByteBuffer(ptr, (jlong) bufferSize); return byteBuffer; } else { return NULL; } } static jlong nAllocationGetStride(JNIEnv *_env, jobject _this, jlong con, jlong alloc) { LOG_API("nAllocationGetStride, con(%p), alloc(%p)", (RsContext)con, (RsAllocation)alloc); size_t strideIn = 0; if (alloc != 0 && dispatchTab.AllocationGetPointer != nullptr) { dispatchTab.AllocationGetPointer((RsContext)con, (RsAllocation)alloc, 0, RS_ALLOCATION_CUBEMAP_FACE_POSITIVE_X, 0, 0, &strideIn, sizeof(size_t)); } return (jlong)strideIn; } // --------------------------------------------------------------------------- static const char *classPathName = "androidx/renderscript/RenderScript"; static JNINativeMethod methods[] = { {"nLoadSO", "(ZILjava/lang/String;)Z", (bool*)nLoadSO }, {"nLoadIOSO", "()Z", (bool*)nLoadIOSO }, {"nDeviceCreate", "()J", (void*)nDeviceCreate }, {"nDeviceDestroy", "(J)V", (void*)nDeviceDestroy }, {"nDeviceSetConfig", "(JII)V", (void*)nDeviceSetConfig }, {"nContextGetUserMessage", "(J[I)I", (void*)nContextGetUserMessage }, {"nContextGetErrorMessage", "(J)Ljava/lang/String;", (void*)nContextGetErrorMessage }, {"nContextPeekMessage", "(J[I)I", (void*)nContextPeekMessage }, {"nContextInitToClient", "(J)V", (void*)nContextInitToClient }, {"nContextDeinitToClient", "(J)V", (void*)nContextDeinitToClient }, // All methods below are thread protected in java. {"rsnContextCreate", "(JIIILjava/lang/String;)J", (void*)nContextCreate }, {"rsnContextFinish", "(J)V", (void*)nContextFinish }, {"rsnContextSetPriority", "(JI)V", (void*)nContextSetPriority }, {"rsnContextDestroy", "(J)V", (void*)nContextDestroy }, {"rsnContextDump", "(JI)V", (void*)nContextDump }, {"rsnContextSendMessage", "(JI[I)V", (void*)nContextSendMessage }, {"rsnClosureCreate", "(JJJ[J[J[I[J[J)J", (void*)nClosureCreate }, {"rsnInvokeClosureCreate", "(JJ[B[J[J[I)J", (void*)nInvokeClosureCreate }, {"rsnClosureSetArg", "(JJIJI)V", (void*)nClosureSetArg }, {"rsnClosureSetGlobal", "(JJJJI)V", (void*)nClosureSetGlobal }, {"rsnObjDestroy", "(JJ)V", (void*)nObjDestroy }, {"rsnElementCreate", "(JJIZI)J", (void*)nElementCreate }, {"rsnElementCreate2", "(J[J[Ljava/lang/String;[I)J", (void*)nElementCreate2 }, {"rsnElementGetSubElements", "(JJ[J[Ljava/lang/String;[I)V", (void*)nElementGetSubElements }, {"rsnTypeCreate", "(JJIIIZZI)J", (void*)nTypeCreate }, {"rsnAllocationCreateTyped", "(JJIIJ)J", (void*)nAllocationCreateTyped }, {"rsnAllocationCreateFromBitmap", "(JJILandroid/graphics/Bitmap;I)J", (void*)nAllocationCreateFromBitmap }, {"rsnAllocationCreateBitmapBackedAllocation", "(JJILandroid/graphics/Bitmap;I)J", (void*)nAllocationCreateBitmapBackedAllocation }, {"rsnAllocationCubeCreateFromBitmap","(JJILandroid/graphics/Bitmap;I)J", (void*)nAllocationCubeCreateFromBitmap }, {"rsnAllocationCopyFromBitmap", "(JJLandroid/graphics/Bitmap;)V", (void*)nAllocationCopyFromBitmap }, {"rsnAllocationCopyToBitmap", "(JJLandroid/graphics/Bitmap;)V", (void*)nAllocationCopyToBitmap }, {"rsnAllocationSyncAll", "(JJI)V", (void*)nAllocationSyncAll }, {"rsnAllocationSetSurface", "(JJLandroid/view/Surface;)V", (void*)nAllocationSetSurface }, {"rsnAllocationIoSend", "(JJ)V", (void*)nAllocationIoSend }, {"rsnAllocationData1D", "(JJIIILjava/lang/Object;IIIZ)V", (void*)nAllocationData1D }, {"rsnAllocationElementData1D", "(JJIII[BI)V", (void*)nAllocationElementData1D }, //{"rsnAllocationElementData", "(JJIIIII[BI)V", (void*)nAllocationElementData }, {"rsnAllocationData2D", "(JJIIIIIILjava/lang/Object;IIIZ)V", (void*)nAllocationData2D }, {"rsnAllocationData2D", "(JJIIIIIIJIIII)V", (void*)nAllocationData2D_alloc }, {"rsnAllocationData3D", "(JJIIIIIIILjava/lang/Object;IIIZ)V", (void*)nAllocationData3D }, {"rsnAllocationData3D", "(JJIIIIIIIJIIII)V", (void*)nAllocationData3D_alloc }, {"rsnAllocationRead", "(JJLjava/lang/Object;IIZ)V", (void*)nAllocationRead }, {"rsnAllocationRead1D", "(JJIIILjava/lang/Object;IIIZ)V", (void*)nAllocationRead1D }, //{"rsnAllocationElementRead", "(JJIIIII[BI)V", (void*)nAllocationElementRead }, {"rsnAllocationRead2D", "(JJIIIIIILjava/lang/Object;IIIZ)V", (void*)nAllocationRead2D }, //{"rsnAllocationRead3D", "(JJIIIIIIILjava/lang/Object;IIIZ)V", (void*)nAllocationRead3D }, {"rsnAllocationGetType", "(JJ)J", (void*)nAllocationGetType}, {"rsnAllocationResize1D", "(JJI)V", (void*)nAllocationResize1D }, {"rsnAllocationGenerateMipmaps", "(JJ)V", (void*)nAllocationGenerateMipmaps }, {"rsnScriptBindAllocation", "(JJJIZ)V", (void*)nScriptBindAllocation }, {"rsnScriptSetTimeZone", "(JJ[BZ)V", (void*)nScriptSetTimeZone }, {"rsnScriptInvoke", "(JJIZ)V", (void*)nScriptInvoke }, {"rsnScriptInvokeV", "(JJI[BZ)V", (void*)nScriptInvokeV }, {"rsnScriptForEach", "(JJJIJJZ)V", (void*)nScriptForEach }, {"rsnScriptForEach", "(JJJIJJ[BZ)V", (void*)nScriptForEachV }, {"rsnScriptForEach", "(JJI[JJ[B[I)V", (void*)nScriptForEachMulti }, {"rsnScriptForEachClipped", "(JJJIJJIIIIIIZ)V", (void*)nScriptForEachClipped }, {"rsnScriptForEachClipped", "(JJJIJJ[BIIIIIIZ)V", (void*)nScriptForEachClippedV }, {"rsnScriptReduce", "(JJI[JJ[I)V", (void*)nScriptReduce }, {"rsnScriptSetVarI", "(JJIIZ)V", (void*)nScriptSetVarI }, {"rsnScriptSetVarJ", "(JJIJZ)V", (void*)nScriptSetVarJ }, {"rsnScriptSetVarF", "(JJIFZ)V", (void*)nScriptSetVarF }, {"rsnScriptSetVarD", "(JJIDZ)V", (void*)nScriptSetVarD }, {"rsnScriptSetVarV", "(JJI[BZ)V", (void*)nScriptSetVarV }, {"rsnScriptSetVarVE", "(JJI[BJ[IZ)V", (void*)nScriptSetVarVE }, {"rsnScriptSetVarObj", "(JJIJZ)V", (void*)nScriptSetVarObj }, {"rsnScriptCCreate", "(JLjava/lang/String;Ljava/lang/String;[BI)J", (void*)nScriptCCreate }, {"rsnScriptIntrinsicCreate", "(JIJZ)J", (void*)nScriptIntrinsicCreate }, {"rsnScriptKernelIDCreate", "(JJIIZ)J", (void*)nScriptKernelIDCreate }, {"rsnScriptInvokeIDCreate", "(JJI)J", (void*)nScriptInvokeIDCreate }, {"rsnScriptFieldIDCreate", "(JJIZ)J", (void*)nScriptFieldIDCreate }, {"rsnScriptGroupCreate", "(J[J[J[J[J[J)J", (void*)nScriptGroupCreate }, {"rsnScriptGroup2Create", "(JLjava/lang/String;Ljava/lang/String;[J)J", (void*)nScriptGroup2Create }, {"rsnScriptGroupSetInput", "(JJJJ)V", (void*)nScriptGroupSetInput }, {"rsnScriptGroupSetOutput", "(JJJJ)V", (void*)nScriptGroupSetOutput }, {"rsnScriptGroupExecute", "(JJ)V", (void*)nScriptGroupExecute }, {"rsnScriptGroup2Execute", "(JJ)V", (void*)nScriptGroup2Execute }, {"rsnScriptIntrinsicBLAS_Single", "(JJJIIIIIIIIIFJJFJIIIIZ)V", (void*)nScriptIntrinsicBLAS_Single }, {"rsnScriptIntrinsicBLAS_Double", "(JJJIIIIIIIIIDJJDJIIIIZ)V", (void*)nScriptIntrinsicBLAS_Double }, {"rsnScriptIntrinsicBLAS_Complex", "(JJJIIIIIIIIIFFJJFFJIIIIZ)V", (void*)nScriptIntrinsicBLAS_Complex }, {"rsnScriptIntrinsicBLAS_Z", "(JJJIIIIIIIIIDDJJDDJIIIIZ)V", (void*)nScriptIntrinsicBLAS_Z }, {"rsnScriptIntrinsicBLAS_BNNM", "(JJJIIIJIJIJIIZ)V", (void*)nScriptIntrinsicBLAS_BNNM }, {"rsnSamplerCreate", "(JIIIIIF)J", (void*)nSamplerCreate }, {"rsnSystemGetPointerSize", "()I", (void*)nSystemGetPointerSize }, // Entry points for Inc libRSSupport {"nIncLoadSO", "(ILjava/lang/String;)Z", (bool*)nIncLoadSO }, {"nIncDeviceCreate", "()J", (void*)nIncDeviceCreate }, {"nIncDeviceDestroy", "(J)V", (void*)nIncDeviceDestroy }, {"rsnIncContextCreate", "(JIII)J", (void*)nIncContextCreate }, {"rsnIncContextFinish", "(J)V", (void*)nIncContextFinish }, {"rsnIncContextDestroy", "(J)V", (void*)nIncContextDestroy }, {"rsnIncObjDestroy", "(JJ)V", (void*)nIncObjDestroy }, {"rsnIncElementCreate", "(JJIZI)J", (void*)nIncElementCreate }, {"rsnIncTypeCreate", "(JJIIIZZI)J", (void*)nIncTypeCreate }, {"rsnIncAllocationCreateTyped", "(JJJJI)J", (void*)nIncAllocationCreateTyped }, {"rsnAllocationGetByteBuffer", "(JJIII)Ljava/nio/ByteBuffer;", (void*)nAllocationGetByteBuffer }, {"rsnAllocationGetStride", "(JJ)J", (void*)nAllocationGetStride }, }; // --------------------------------------------------------------------------- jint JNI_OnLoad(JavaVM* vm, void* reserved) { JNIEnv* env = NULL; jclass clazz = NULL; jint result = -1; if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) { // __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, // "ERROR: GetEnv failed\n"); goto bail; } if (env == NULL) { // __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, "ERROR: env == NULL"); goto bail; } clazz = env->FindClass(classPathName); if (clazz == NULL) { goto bail; } if (env->RegisterNatives(clazz, methods, NELEM(methods)) < 0) { // __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, // "ERROR: MediaPlayer native registration failed\n"); goto bail; } /* success -- return valid version number */ result = JNI_VERSION_1_4; bail: return result; }