1142 lines
28 KiB
C++
1142 lines
28 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#if defined(__ANDROID__)
|
|
|
|
// Allow use of stuff in <time.h> and abort()
|
|
#define FORBIDDEN_SYMBOL_EXCEPTION_time_h
|
|
#define FORBIDDEN_SYMBOL_EXCEPTION_abort
|
|
|
|
// Disable printf override in common/forbidden.h to avoid
|
|
// clashes with log.h from the Android SDK.
|
|
// That header file uses
|
|
// __attribute__ ((format(printf, 3, 4)))
|
|
// which gets messed up by our override mechanism; this could
|
|
// be avoided by either changing the Android SDK to use the equally
|
|
// legal and valid
|
|
// __attribute__ ((format(__printf__, 3, 4)))
|
|
// or by refining our printf override to use a varadic macro
|
|
// (which then wouldn't be portable, though).
|
|
// Anyway, for now we just disable the printf override globally
|
|
// for the Android port
|
|
#define FORBIDDEN_SYMBOL_EXCEPTION_printf
|
|
|
|
#include <android/bitmap.h>
|
|
|
|
#include "backends/platform/android/android.h"
|
|
#include "backends/platform/android/jni-android.h"
|
|
#include "backends/platform/android/asset-archive.h"
|
|
|
|
#include "base/main.h"
|
|
#include "base/version.h"
|
|
#include "common/config-manager.h"
|
|
#include "common/error.h"
|
|
#include "common/textconsole.h"
|
|
#include "engines/engine.h"
|
|
#include "graphics/surface.h"
|
|
|
|
__attribute__ ((visibility("default")))
|
|
jint JNICALL JNI_OnLoad(JavaVM *vm, void *) {
|
|
return JNI::onLoad(vm);
|
|
}
|
|
|
|
pthread_key_t JNI::_env_tls;
|
|
|
|
JavaVM *JNI::_vm = 0;
|
|
jobject JNI::_jobj = 0;
|
|
jobject JNI::_jobj_audio_track = 0;
|
|
jobject JNI::_jobj_egl = 0;
|
|
jobject JNI::_jobj_egl_display = 0;
|
|
jobject JNI::_jobj_egl_surface = 0;
|
|
int JNI::_egl_version = 0;
|
|
|
|
Common::Archive *JNI::_asset_archive = 0;
|
|
OSystem_Android *JNI::_system = 0;
|
|
|
|
bool JNI::pause = false;
|
|
sem_t JNI::pause_sem;
|
|
|
|
int JNI::surface_changeid = 0;
|
|
int JNI::egl_surface_width = 0;
|
|
int JNI::egl_surface_height = 0;
|
|
int JNI::egl_bits_per_pixel = 0;
|
|
bool JNI::_ready_for_events = 0;
|
|
|
|
jmethodID JNI::_MID_getDPI = 0;
|
|
jmethodID JNI::_MID_displayMessageOnOSD = 0;
|
|
jmethodID JNI::_MID_openUrl = 0;
|
|
jmethodID JNI::_MID_hasTextInClipboard = 0;
|
|
jmethodID JNI::_MID_getTextFromClipboard = 0;
|
|
jmethodID JNI::_MID_setTextInClipboard = 0;
|
|
jmethodID JNI::_MID_isConnectionLimited = 0;
|
|
jmethodID JNI::_MID_setWindowCaption = 0;
|
|
jmethodID JNI::_MID_showVirtualKeyboard = 0;
|
|
jmethodID JNI::_MID_showKeyboardControl = 0;
|
|
jmethodID JNI::_MID_getBitmapResource = 0;
|
|
jmethodID JNI::_MID_setTouchMode = 0;
|
|
jmethodID JNI::_MID_getTouchMode = 0;
|
|
jmethodID JNI::_MID_setOrientation = 0;
|
|
jmethodID JNI::_MID_getScummVMBasePath;
|
|
jmethodID JNI::_MID_getScummVMConfigPath;
|
|
jmethodID JNI::_MID_getScummVMLogPath;
|
|
jmethodID JNI::_MID_getSysArchives = 0;
|
|
jmethodID JNI::_MID_getAllStorageLocations = 0;
|
|
jmethodID JNI::_MID_initSurface = 0;
|
|
jmethodID JNI::_MID_deinitSurface = 0;
|
|
jmethodID JNI::_MID_eglVersion = 0;
|
|
jmethodID JNI::_MID_getNewSAFTree = 0;
|
|
jmethodID JNI::_MID_getSAFTrees = 0;
|
|
jmethodID JNI::_MID_findSAFTree = 0;
|
|
|
|
jmethodID JNI::_MID_EGL10_eglSwapBuffers = 0;
|
|
|
|
jmethodID JNI::_MID_AudioTrack_flush = 0;
|
|
jmethodID JNI::_MID_AudioTrack_pause = 0;
|
|
jmethodID JNI::_MID_AudioTrack_play = 0;
|
|
jmethodID JNI::_MID_AudioTrack_stop = 0;
|
|
jmethodID JNI::_MID_AudioTrack_write = 0;
|
|
|
|
PauseToken JNI::_pauseToken;
|
|
|
|
const JNINativeMethod JNI::_natives[] = {
|
|
{ "create", "(Landroid/content/res/AssetManager;"
|
|
"Ljavax/microedition/khronos/egl/EGL10;"
|
|
"Ljavax/microedition/khronos/egl/EGLDisplay;"
|
|
"Landroid/media/AudioTrack;II)V",
|
|
(void *)JNI::create },
|
|
{ "destroy", "()V",
|
|
(void *)JNI::destroy },
|
|
{ "setSurface", "(III)V",
|
|
(void *)JNI::setSurface },
|
|
{ "main", "([Ljava/lang/String;)I",
|
|
(void *)JNI::main },
|
|
{ "pushEvent", "(IIIIIII)V",
|
|
(void *)JNI::pushEvent },
|
|
{ "updateTouch", "(IIII)V",
|
|
(void *)JNI::updateTouch },
|
|
{ "setupTouchMode", "(II)V",
|
|
(void *)JNI::setupTouchMode },
|
|
{ "setPause", "(Z)V",
|
|
(void *)JNI::setPause },
|
|
{ "getNativeVersionInfo", "()Ljava/lang/String;",
|
|
(void *)JNI::getNativeVersionInfo }
|
|
};
|
|
|
|
JNI::JNI() {
|
|
}
|
|
|
|
JNI::~JNI() {
|
|
}
|
|
|
|
jint JNI::onLoad(JavaVM *vm) {
|
|
if (pthread_key_create(&_env_tls, NULL)) {
|
|
return JNI_ERR;
|
|
}
|
|
|
|
_vm = vm;
|
|
|
|
JNIEnv *env;
|
|
|
|
if (_vm->GetEnv((void **)&env, JNI_VERSION_1_2))
|
|
return JNI_ERR;
|
|
|
|
if (pthread_setspecific(_env_tls, env)) {
|
|
return JNI_ERR;
|
|
}
|
|
|
|
jclass cls = env->FindClass("org/scummvm/scummvm/ScummVM");
|
|
if (cls == 0)
|
|
return JNI_ERR;
|
|
|
|
if (env->RegisterNatives(cls, _natives, ARRAYSIZE(_natives)) < 0)
|
|
return JNI_ERR;
|
|
|
|
env->DeleteLocalRef(cls);
|
|
return JNI_VERSION_1_2;
|
|
}
|
|
|
|
JNIEnv *JNI::fetchEnv() {
|
|
JNIEnv *env;
|
|
|
|
jint res = _vm->GetEnv((void **)&env, JNI_VERSION_1_2);
|
|
|
|
if (res != JNI_OK) {
|
|
LOGE("GetEnv() failed: %d", res);
|
|
abort();
|
|
}
|
|
|
|
pthread_setspecific(_env_tls, env);
|
|
|
|
return env;
|
|
}
|
|
|
|
void JNI::attachThread() {
|
|
JNIEnv *env = 0;
|
|
|
|
jint res = _vm->AttachCurrentThread(&env, 0);
|
|
|
|
if (res != JNI_OK) {
|
|
LOGE("AttachCurrentThread() failed: %d", res);
|
|
abort();
|
|
}
|
|
|
|
if (pthread_setspecific(_env_tls, env)) {
|
|
LOGE("pthread_setspecific() failed");
|
|
abort();
|
|
}
|
|
}
|
|
|
|
void JNI::detachThread() {
|
|
pthread_setspecific(_env_tls, NULL);
|
|
|
|
jint res = _vm->DetachCurrentThread();
|
|
|
|
if (res != JNI_OK) {
|
|
LOGE("DetachCurrentThread() failed: %d", res);
|
|
abort();
|
|
}
|
|
}
|
|
|
|
void JNI::setReadyForEvents(bool ready) {
|
|
_ready_for_events = ready;
|
|
}
|
|
|
|
void JNI::throwByName(JNIEnv *env, const char *name, const char *msg) {
|
|
jclass cls = env->FindClass(name);
|
|
|
|
// if cls is 0, an exception has already been thrown
|
|
if (cls != 0)
|
|
env->ThrowNew(cls, msg);
|
|
|
|
env->DeleteLocalRef(cls);
|
|
}
|
|
|
|
void JNI::throwRuntimeException(JNIEnv *env, const char *msg) {
|
|
throwByName(env, "java/lang/RuntimeException", msg);
|
|
}
|
|
|
|
// calls to the dark side
|
|
|
|
void JNI::getDPI(float *values) {
|
|
values[0] = 0.0;
|
|
values[1] = 0.0;
|
|
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jfloatArray array = env->NewFloatArray(2);
|
|
|
|
env->CallVoidMethod(_jobj, _MID_getDPI, array);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to get DPIs");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
} else {
|
|
jfloat *res = env->GetFloatArrayElements(array, 0);
|
|
|
|
if (res) {
|
|
values[0] = res[0];
|
|
values[1] = res[1];
|
|
|
|
env->ReleaseFloatArrayElements(array, res, 0);
|
|
}
|
|
}
|
|
LOGD("JNI::getDPI() xdpi: %f, ydpi: %f", values[0], values[1]);
|
|
env->DeleteLocalRef(array);
|
|
}
|
|
|
|
void JNI::displayMessageOnOSD(const Common::U32String &msg) {
|
|
// called from common/osd_message_queue, method: OSDMessageQueue::pollEvent()
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jstring java_msg = convertToJString(env, msg);
|
|
if (java_msg == nullptr) {
|
|
// Show a placeholder indicative of the translation error instead of silent failing
|
|
java_msg = env->NewStringUTF("?");
|
|
LOGE("Failed to convert message to UTF-8 for OSD!");
|
|
}
|
|
|
|
env->CallVoidMethod(_jobj, _MID_displayMessageOnOSD, java_msg);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to display OSD message");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
|
|
env->DeleteLocalRef(java_msg);
|
|
}
|
|
|
|
bool JNI::openUrl(const Common::String &url) {
|
|
bool success = true;
|
|
JNIEnv *env = JNI::getEnv();
|
|
jstring javaUrl = env->NewStringUTF(url.c_str());
|
|
|
|
env->CallVoidMethod(_jobj, _MID_openUrl, javaUrl);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to open URL");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
success = false;
|
|
}
|
|
|
|
env->DeleteLocalRef(javaUrl);
|
|
return success;
|
|
}
|
|
|
|
bool JNI::hasTextInClipboard() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
bool hasText = env->CallBooleanMethod(_jobj, _MID_hasTextInClipboard);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to check the contents of the clipboard");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
hasText = true;
|
|
}
|
|
|
|
return hasText;
|
|
}
|
|
|
|
Common::U32String JNI::getTextFromClipboard() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jstring javaText = (jstring)env->CallObjectMethod(_jobj, _MID_getTextFromClipboard);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to retrieve text from the clipboard");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return Common::U32String();
|
|
}
|
|
|
|
Common::U32String text = convertFromJString(env, javaText);
|
|
env->DeleteLocalRef(javaText);
|
|
|
|
return text;
|
|
}
|
|
|
|
bool JNI::setTextInClipboard(const Common::U32String &text) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
jstring javaText = convertToJString(env, text);
|
|
|
|
bool success = env->CallBooleanMethod(_jobj, _MID_setTextInClipboard, javaText);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to add text to the clipboard");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
success = false;
|
|
}
|
|
|
|
env->DeleteLocalRef(javaText);
|
|
return success;
|
|
}
|
|
|
|
bool JNI::isConnectionLimited() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
bool limited = env->CallBooleanMethod(_jobj, _MID_isConnectionLimited);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to check whether connection's limited");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
limited = true;
|
|
}
|
|
|
|
return limited;
|
|
}
|
|
|
|
void JNI::setWindowCaption(const Common::U32String &caption) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
jstring java_caption = convertToJString(env, caption);
|
|
|
|
env->CallVoidMethod(_jobj, _MID_setWindowCaption, java_caption);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to set window caption");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
|
|
env->DeleteLocalRef(java_caption);
|
|
}
|
|
|
|
void JNI::showVirtualKeyboard(bool enable) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->CallVoidMethod(_jobj, _MID_showVirtualKeyboard, enable);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error trying to show virtual keyboard");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
void JNI::showKeyboardControl(bool enable) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->CallVoidMethod(_jobj, _MID_showKeyboardControl, enable);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error trying to show virtual keyboard control");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
Graphics::Surface *JNI::getBitmapResource(BitmapResources resource) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jobject bitmap = env->CallObjectMethod(_jobj, _MID_getBitmapResource, (int) resource);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Can't get bitmap resource");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
if (bitmap == nullptr) {
|
|
LOGE("Bitmap resource was not found");
|
|
return nullptr;
|
|
}
|
|
|
|
AndroidBitmapInfo bitmap_info;
|
|
if (AndroidBitmap_getInfo(env, bitmap, &bitmap_info) != ANDROID_BITMAP_RESULT_SUCCESS) {
|
|
LOGE("Error reading bitmap info");
|
|
env->DeleteLocalRef(bitmap);
|
|
return nullptr;
|
|
}
|
|
|
|
Graphics::PixelFormat fmt;
|
|
switch(bitmap_info.format) {
|
|
case ANDROID_BITMAP_FORMAT_RGBA_8888:
|
|
#ifdef SCUMM_BIG_ENDIAN
|
|
fmt = Graphics::PixelFormat(4, 8, 8, 8, 8, 24, 16, 8, 0);
|
|
#else
|
|
fmt = Graphics::PixelFormat(4, 8, 8, 8, 8, 0, 8, 16, 24);
|
|
#endif
|
|
break;
|
|
case ANDROID_BITMAP_FORMAT_RGBA_4444:
|
|
fmt = Graphics::PixelFormat(2, 4, 4, 4, 4, 12, 8, 4, 0);
|
|
break;
|
|
case ANDROID_BITMAP_FORMAT_RGB_565:
|
|
fmt = Graphics::PixelFormat(2, 5, 6, 5, 0, 11, 5, 0, 0);
|
|
break;
|
|
default:
|
|
LOGE("Bitmap has unsupported format");
|
|
env->DeleteLocalRef(bitmap);
|
|
return nullptr;
|
|
}
|
|
|
|
void *src_pixels = nullptr;
|
|
if (AndroidBitmap_lockPixels(env, bitmap, &src_pixels) != ANDROID_BITMAP_RESULT_SUCCESS) {
|
|
LOGE("Error locking bitmap pixels");
|
|
env->DeleteLocalRef(bitmap);
|
|
return nullptr;
|
|
}
|
|
|
|
Graphics::Surface *ret = new Graphics::Surface();
|
|
ret->create(bitmap_info.width, bitmap_info.height, fmt);
|
|
ret->copyRectToSurface(src_pixels, bitmap_info.stride,
|
|
0, 0, bitmap_info.width, bitmap_info.height);
|
|
|
|
AndroidBitmap_unlockPixels(env, bitmap);
|
|
env->DeleteLocalRef(bitmap);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void JNI::setTouchMode(int touchMode) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->CallVoidMethod(_jobj, _MID_setTouchMode, touchMode);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error trying to set touch controls mode");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
int JNI::getTouchMode() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
int mode = env->CallIntMethod(_jobj, _MID_getTouchMode);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error trying to get touch controls status");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
|
|
return mode;
|
|
}
|
|
|
|
void JNI::setOrientation(int orientation) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->CallVoidMethod(_jobj, _MID_setOrientation, orientation);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error trying to set orientation");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
Common::String JNI::getScummVMBasePath() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jstring pathObj = (jstring)env->CallObjectMethod(_jobj, _MID_getScummVMBasePath);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to get ScummVM base folder path");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return Common::String();
|
|
}
|
|
|
|
Common::String path;
|
|
const char *pathP = env->GetStringUTFChars(pathObj, 0);
|
|
if (pathP != 0) {
|
|
path = Common::String(pathP);
|
|
env->ReleaseStringUTFChars(pathObj, pathP);
|
|
}
|
|
env->DeleteLocalRef(pathObj);
|
|
|
|
return path;
|
|
}
|
|
|
|
Common::String JNI::getScummVMConfigPath() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jstring pathObj = (jstring)env->CallObjectMethod(_jobj, _MID_getScummVMConfigPath);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to get ScummVM config file path");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return Common::String();
|
|
}
|
|
|
|
Common::String path;
|
|
const char *pathP = env->GetStringUTFChars(pathObj, 0);
|
|
if (pathP != 0) {
|
|
path = Common::String(pathP);
|
|
env->ReleaseStringUTFChars(pathObj, pathP);
|
|
}
|
|
env->DeleteLocalRef(pathObj);
|
|
|
|
return path;
|
|
}
|
|
|
|
Common::String JNI::getScummVMLogPath() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jstring pathObj = (jstring)env->CallObjectMethod(_jobj, _MID_getScummVMLogPath);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Failed to get ScummVM log file path");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return Common::String();
|
|
}
|
|
|
|
Common::String path;
|
|
const char *pathP = env->GetStringUTFChars(pathObj, 0);
|
|
if (pathP != 0) {
|
|
path = Common::String(pathP);
|
|
env->ReleaseStringUTFChars(pathObj, pathP);
|
|
}
|
|
env->DeleteLocalRef(pathObj);
|
|
|
|
return path;
|
|
}
|
|
|
|
// The following adds assets folder to search set.
|
|
// However searching and retrieving from "assets" on Android this is slow
|
|
// so we also make sure to add the base directory, with a higher priority
|
|
// This is done via a call to ScummVMActivity's (java) getSysArchives
|
|
void JNI::addSysArchivesToSearchSet(Common::SearchSet &s, int priority) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
// get any additional specified paths (from ScummVMActivity code)
|
|
// Insert them with "priority" priority.
|
|
jobjectArray array =
|
|
(jobjectArray)env->CallObjectMethod(_jobj, _MID_getSysArchives);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error finding system archive path");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return;
|
|
}
|
|
|
|
jsize size = env->GetArrayLength(array);
|
|
for (jsize i = 0; i < size; ++i) {
|
|
jstring path_obj = (jstring)env->GetObjectArrayElement(array, i);
|
|
const char *path = env->GetStringUTFChars(path_obj, 0);
|
|
|
|
if (path != 0) {
|
|
s.addDirectory(path, path, priority);
|
|
env->ReleaseStringUTFChars(path_obj, path);
|
|
}
|
|
|
|
env->DeleteLocalRef(path_obj);
|
|
}
|
|
env->DeleteLocalRef(array);
|
|
|
|
// add the internal asset (android's structure) with a lower priority,
|
|
// since:
|
|
// 1. It is very slow in accessing large files (eg our growing fonts.dat)
|
|
// 2. we extract the asset contents anyway to the internal app path
|
|
// 3. we pass the internal app path in the process above (via _MID_getSysArchives)
|
|
// However, we keep android APK's "assets" as a fall back, in case something went wrong with the extraction process
|
|
// and since we had the code anyway
|
|
s.add("ASSET", _asset_archive, priority - 1, false);
|
|
}
|
|
|
|
bool JNI::initSurface() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jobject obj = env->CallObjectMethod(_jobj, _MID_initSurface);
|
|
|
|
if (!obj || env->ExceptionCheck()) {
|
|
LOGE("initSurface failed");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return false;
|
|
}
|
|
|
|
_jobj_egl_surface = env->NewGlobalRef(obj);
|
|
env->DeleteLocalRef(obj);
|
|
|
|
return true;
|
|
}
|
|
|
|
void JNI::deinitSurface() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->DeleteGlobalRef(_jobj_egl_surface);
|
|
_jobj_egl_surface = 0;
|
|
|
|
env->CallVoidMethod(_jobj, _MID_deinitSurface);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("deinitSurface failed");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
int JNI::fetchEGLVersion() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
_egl_version = env->CallIntMethod(_jobj, _MID_eglVersion);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("eglVersion failed");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
_egl_version = 0;
|
|
}
|
|
|
|
return _egl_version;
|
|
}
|
|
|
|
void JNI::setAudioPause() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->CallVoidMethod(_jobj_audio_track, _MID_AudioTrack_flush);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error flushing AudioTrack");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
|
|
env->CallVoidMethod(_jobj_audio_track, _MID_AudioTrack_pause);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error setting AudioTrack: pause");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
void JNI::setAudioPlay() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->CallVoidMethod(_jobj_audio_track, _MID_AudioTrack_play);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error setting AudioTrack: play");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
void JNI::setAudioStop() {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
env->CallVoidMethod(_jobj_audio_track, _MID_AudioTrack_stop);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error setting AudioTrack: stop");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
}
|
|
}
|
|
|
|
// natives for the dark side
|
|
|
|
void JNI::create(JNIEnv *env, jobject self, jobject asset_manager,
|
|
jobject egl, jobject egl_display,
|
|
jobject at, jint audio_sample_rate, jint audio_buffer_size) {
|
|
LOGI("Native version: %s", gScummVMFullVersion);
|
|
|
|
assert(!_system);
|
|
|
|
// Resolve every JNI method before anything else in case we need it
|
|
|
|
// weak global ref to allow class to be unloaded
|
|
// ... except dalvik implements NewWeakGlobalRef only on froyo
|
|
//_jobj = env->NewWeakGlobalRef(self);
|
|
|
|
_jobj = env->NewGlobalRef(self);
|
|
|
|
jclass cls = env->GetObjectClass(_jobj);
|
|
|
|
#define FIND_METHOD(prefix, name, signature) do { \
|
|
_MID_ ## prefix ## name = env->GetMethodID(cls, #name, signature); \
|
|
if (_MID_ ## prefix ## name == 0) { \
|
|
LOGE("Can't find function %s", #name); \
|
|
abort(); \
|
|
} \
|
|
} while (0)
|
|
|
|
FIND_METHOD(, setWindowCaption, "(Ljava/lang/String;)V");
|
|
FIND_METHOD(, getDPI, "([F)V");
|
|
FIND_METHOD(, displayMessageOnOSD, "(Ljava/lang/String;)V");
|
|
FIND_METHOD(, openUrl, "(Ljava/lang/String;)V");
|
|
FIND_METHOD(, hasTextInClipboard, "()Z");
|
|
FIND_METHOD(, getTextFromClipboard, "()Ljava/lang/String;");
|
|
FIND_METHOD(, setTextInClipboard, "(Ljava/lang/String;)Z");
|
|
FIND_METHOD(, isConnectionLimited, "()Z");
|
|
FIND_METHOD(, showVirtualKeyboard, "(Z)V");
|
|
FIND_METHOD(, showKeyboardControl, "(Z)V");
|
|
FIND_METHOD(, getBitmapResource, "(I)Landroid/graphics/Bitmap;");
|
|
FIND_METHOD(, setTouchMode, "(I)V");
|
|
FIND_METHOD(, getTouchMode, "()I");
|
|
FIND_METHOD(, setOrientation, "(I)V");
|
|
FIND_METHOD(, getScummVMBasePath, "()Ljava/lang/String;");
|
|
FIND_METHOD(, getScummVMConfigPath, "()Ljava/lang/String;");
|
|
FIND_METHOD(, getScummVMLogPath, "()Ljava/lang/String;");
|
|
FIND_METHOD(, getSysArchives, "()[Ljava/lang/String;");
|
|
FIND_METHOD(, getAllStorageLocations, "()[Ljava/lang/String;");
|
|
FIND_METHOD(, initSurface, "()Ljavax/microedition/khronos/egl/EGLSurface;");
|
|
FIND_METHOD(, deinitSurface, "()V");
|
|
FIND_METHOD(, eglVersion, "()I");
|
|
FIND_METHOD(, getNewSAFTree,
|
|
"(ZZLjava/lang/String;Ljava/lang/String;)Lorg/scummvm/scummvm/SAFFSTree;");
|
|
FIND_METHOD(, getSAFTrees, "()[Lorg/scummvm/scummvm/SAFFSTree;");
|
|
FIND_METHOD(, findSAFTree, "(Ljava/lang/String;)Lorg/scummvm/scummvm/SAFFSTree;");
|
|
|
|
_jobj_egl = env->NewGlobalRef(egl);
|
|
_jobj_egl_display = env->NewGlobalRef(egl_display);
|
|
_egl_version = 0;
|
|
|
|
env->DeleteLocalRef(cls);
|
|
|
|
cls = env->GetObjectClass(_jobj_egl);
|
|
|
|
FIND_METHOD(EGL10_, eglSwapBuffers,
|
|
"(Ljavax/microedition/khronos/egl/EGLDisplay;"
|
|
"Ljavax/microedition/khronos/egl/EGLSurface;)Z");
|
|
|
|
_jobj_audio_track = env->NewGlobalRef(at);
|
|
|
|
env->DeleteLocalRef(cls);
|
|
|
|
cls = env->GetObjectClass(_jobj_audio_track);
|
|
|
|
FIND_METHOD(AudioTrack_, flush, "()V");
|
|
FIND_METHOD(AudioTrack_, pause, "()V");
|
|
FIND_METHOD(AudioTrack_, play, "()V");
|
|
FIND_METHOD(AudioTrack_, stop, "()V");
|
|
FIND_METHOD(AudioTrack_, write, "([BII)I");
|
|
|
|
env->DeleteLocalRef(cls);
|
|
#undef FIND_METHOD
|
|
|
|
pause = false;
|
|
// initial value of zero!
|
|
sem_init(&pause_sem, 0, 0);
|
|
|
|
_asset_archive = new AndroidAssetArchive(asset_manager);
|
|
assert(_asset_archive);
|
|
|
|
_system = new OSystem_Android(audio_sample_rate, audio_buffer_size);
|
|
assert(_system);
|
|
|
|
g_system = _system;
|
|
}
|
|
|
|
void JNI::destroy(JNIEnv *env, jobject self) {
|
|
delete _asset_archive;
|
|
_asset_archive = 0;
|
|
|
|
// _system is a pointer of OSystem_Android <--- ModularBackend <--- BaseBacked <--- Common::OSystem
|
|
// It's better to call destroy() rather than just delete here
|
|
// to avoid mutex issues if a Common::String is used after this point
|
|
_system->destroy();
|
|
|
|
g_system = 0;
|
|
_system = 0;
|
|
|
|
sem_destroy(&pause_sem);
|
|
|
|
// see above
|
|
//JNI::getEnv()->DeleteWeakGlobalRef(_jobj);
|
|
|
|
JNI::getEnv()->DeleteGlobalRef(_jobj_egl_display);
|
|
JNI::getEnv()->DeleteGlobalRef(_jobj_egl);
|
|
JNI::getEnv()->DeleteGlobalRef(_jobj_audio_track);
|
|
JNI::getEnv()->DeleteGlobalRef(_jobj);
|
|
}
|
|
|
|
void JNI::setSurface(JNIEnv *env, jobject self, jint width, jint height, jint bpp) {
|
|
egl_surface_width = width;
|
|
egl_surface_height = height;
|
|
egl_bits_per_pixel = bpp;
|
|
surface_changeid++;
|
|
}
|
|
|
|
jint JNI::main(JNIEnv *env, jobject self, jobjectArray args) {
|
|
assert(_system);
|
|
|
|
const int MAX_NARGS = 32;
|
|
int res = -1;
|
|
|
|
int argc = env->GetArrayLength(args);
|
|
if (argc > MAX_NARGS) {
|
|
throwByName(env, "java/lang/IllegalArgumentException",
|
|
"too many arguments");
|
|
return 0;
|
|
}
|
|
|
|
char *argv[MAX_NARGS];
|
|
|
|
// note use in cleanup loop below
|
|
int nargs;
|
|
|
|
for (nargs = 0; nargs < argc; ++nargs) {
|
|
jstring arg = (jstring)env->GetObjectArrayElement(args, nargs);
|
|
|
|
if (arg == 0) {
|
|
argv[nargs] = 0;
|
|
} else {
|
|
const char *cstr = env->GetStringUTFChars(arg, 0);
|
|
|
|
argv[nargs] = const_cast<char *>(cstr);
|
|
|
|
// exception already thrown?
|
|
if (cstr == 0)
|
|
goto cleanup;
|
|
}
|
|
|
|
env->DeleteLocalRef(arg);
|
|
}
|
|
|
|
LOGI("Entering scummvm_main with %d args", argc);
|
|
|
|
res = scummvm_main(argc, argv);
|
|
|
|
LOGI("scummvm_main exited with code %d", res);
|
|
|
|
_system->quit();
|
|
|
|
cleanup:
|
|
nargs--;
|
|
|
|
for (int i = 0; i < nargs; ++i) {
|
|
if (argv[i] == 0)
|
|
continue;
|
|
|
|
jstring arg = (jstring)env->GetObjectArrayElement(args, nargs);
|
|
|
|
// Exception already thrown?
|
|
if (arg == 0)
|
|
return res;
|
|
|
|
env->ReleaseStringUTFChars(arg, argv[i]);
|
|
env->DeleteLocalRef(arg);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
void JNI::pushEvent(JNIEnv *env, jobject self, int type, int arg1, int arg2,
|
|
int arg3, int arg4, int arg5, int arg6) {
|
|
// drop events until we're ready and after we quit
|
|
if (!_ready_for_events) {
|
|
LOGW("dropping event");
|
|
return;
|
|
}
|
|
|
|
assert(_system);
|
|
|
|
_system->pushEvent(type, arg1, arg2, arg3, arg4, arg5, arg6);
|
|
}
|
|
|
|
void JNI::updateTouch(JNIEnv *env, jobject self, int action, int ptr, int x, int y) {
|
|
// drop events until we're ready and after we quit
|
|
if (!_ready_for_events) {
|
|
LOGW("dropping event");
|
|
return;
|
|
}
|
|
|
|
assert(_system);
|
|
|
|
_system->getTouchControls().update((TouchControls::Action) action, ptr, x, y);
|
|
}
|
|
|
|
void JNI::setupTouchMode(JNIEnv *env, jobject self, jint oldValue, jint newValue) {
|
|
if (!_system)
|
|
return;
|
|
|
|
_system->setupTouchMode(oldValue, newValue);
|
|
}
|
|
|
|
void JNI::setPause(JNIEnv *env, jobject self, jboolean value) {
|
|
if (!_system)
|
|
return;
|
|
|
|
if (g_engine) {
|
|
LOGD("pauseEngine: %d", value);
|
|
|
|
if (value)
|
|
JNI::_pauseToken = g_engine->pauseEngine();
|
|
else if (JNI::_pauseToken.isActive())
|
|
JNI::_pauseToken.clear();
|
|
}
|
|
|
|
if (pause != value) {
|
|
pause = value;
|
|
|
|
if (!pause) {
|
|
// wake up all threads
|
|
for (uint i = 0; i < 3; ++i)
|
|
sem_post(&pause_sem);
|
|
}
|
|
}
|
|
}
|
|
|
|
jstring JNI::getNativeVersionInfo(JNIEnv *env, jobject self) {
|
|
return convertToJString(env, Common::U32String(gScummVMVersion));
|
|
}
|
|
|
|
jint JNI::getAndroidSDKVersionId() {
|
|
// based on: https://stackoverflow.com/a/10511880
|
|
JNIEnv *env = JNI::getEnv();
|
|
// VERSION is a nested class within android.os.Build (hence "$" rather than "/")
|
|
jclass versionClass = env->FindClass("android/os/Build$VERSION");
|
|
if (!versionClass) {
|
|
return 0;
|
|
}
|
|
|
|
jfieldID sdkIntFieldID = NULL;
|
|
sdkIntFieldID = env->GetStaticFieldID(versionClass, "SDK_INT", "I");
|
|
if (!sdkIntFieldID) {
|
|
return 0;
|
|
}
|
|
|
|
jint sdkInt = env->GetStaticIntField(versionClass, sdkIntFieldID);
|
|
//LOGD("sdkInt = %d", sdkInt);
|
|
|
|
env->DeleteLocalRef(versionClass);
|
|
return sdkInt;
|
|
}
|
|
|
|
jstring JNI::convertToJString(JNIEnv *env, const Common::U32String &str) {
|
|
uint len = 0;
|
|
uint16 *u16str = str.encodeUTF16Native(&len);
|
|
jstring jstr = env->NewString(u16str, len);
|
|
delete[] u16str;
|
|
return jstr;
|
|
}
|
|
|
|
Common::U32String JNI::convertFromJString(JNIEnv *env, const jstring &jstr) {
|
|
const uint16 *utf16Str = env->GetStringChars(jstr, 0);
|
|
uint jcount = env->GetStringLength(jstr);
|
|
if (!utf16Str)
|
|
return Common::U32String();
|
|
Common::U32String str = Common::U32String::decodeUTF16Native(utf16Str, jcount);
|
|
env->ReleaseStringChars(jstr, utf16Str);
|
|
|
|
return str;
|
|
}
|
|
|
|
// TODO should this be a U32String array?
|
|
Common::Array<Common::String> JNI::getAllStorageLocations() {
|
|
Common::Array<Common::String> res;
|
|
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jobjectArray array =
|
|
(jobjectArray)env->CallObjectMethod(_jobj, _MID_getAllStorageLocations);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("Error finding system archive path");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return res;
|
|
}
|
|
|
|
jsize size = env->GetArrayLength(array);
|
|
for (jsize i = 0; i < size; ++i) {
|
|
jstring path_obj = (jstring)env->GetObjectArrayElement(array, i);
|
|
const char *path = env->GetStringUTFChars(path_obj, 0);
|
|
|
|
if (path != 0) {
|
|
res.push_back(path);
|
|
env->ReleaseStringUTFChars(path_obj, path);
|
|
}
|
|
|
|
env->DeleteLocalRef(path_obj);
|
|
}
|
|
|
|
env->DeleteLocalRef(array);
|
|
return res;
|
|
}
|
|
|
|
jobject JNI::getNewSAFTree(bool folder, bool writable, const Common::String &initURI,
|
|
const Common::String &prompt) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
jstring javaInitURI = env->NewStringUTF(initURI.c_str());
|
|
jstring javaPrompt = env->NewStringUTF(prompt.c_str());
|
|
|
|
jobject tree = env->CallObjectMethod(_jobj, _MID_getNewSAFTree,
|
|
folder, writable, javaInitURI, javaPrompt);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("getNewSAFTree: error");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
env->DeleteLocalRef(javaInitURI);
|
|
env->DeleteLocalRef(javaPrompt);
|
|
|
|
return tree;
|
|
}
|
|
|
|
Common::Array<jobject> JNI::getSAFTrees() {
|
|
Common::Array<jobject> res;
|
|
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jobjectArray array =
|
|
(jobjectArray)env->CallObjectMethod(_jobj, _MID_getSAFTrees);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("getSAFTrees: error");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return res;
|
|
}
|
|
|
|
jsize size = env->GetArrayLength(array);
|
|
for (jsize i = 0; i < size; ++i) {
|
|
jobject tree = env->GetObjectArrayElement(array, i);
|
|
res.push_back(tree);
|
|
}
|
|
env->DeleteLocalRef(array);
|
|
|
|
return res;
|
|
}
|
|
|
|
jobject JNI::findSAFTree(const Common::String &name) {
|
|
JNIEnv *env = JNI::getEnv();
|
|
|
|
jstring nameObj = env->NewStringUTF(name.c_str());
|
|
|
|
jobject tree = env->CallObjectMethod(_jobj, _MID_findSAFTree, nameObj);
|
|
|
|
env->DeleteLocalRef(nameObj);
|
|
|
|
if (env->ExceptionCheck()) {
|
|
LOGE("findSAFTree: error");
|
|
|
|
env->ExceptionDescribe();
|
|
env->ExceptionClear();
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
return tree;
|
|
}
|
|
|
|
#endif
|