scummvm/backends/platform/android/android.cpp

615 lines
15 KiB
C++
Raw Normal View History

/* 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 2
* 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* $URL$
* $Id$
*
*/
#if defined(__ANDROID__)
#include <sys/time.h>
#include <time.h>
#include <unistd.h>
#include "common/util.h"
#include "common/rect.h"
#include "common/queue.h"
#include "common/mutex.h"
#include "common/events.h"
#include "common/config-manager.h"
#include "backends/keymapper/keymapper.h"
#include "backends/saves/default/default-saves.h"
#include "backends/timer/default/default-timer.h"
#include "backends/platform/android/android.h"
#include "backends/platform/android/asset-archive.h"
const char *android_log_tag = "ScummVM";
// This replaces the bionic libc assert functions with something that
// actually prints the assertion failure before aborting.
extern "C" {
void __assert(const char *file, int line, const char *expr) {
__android_log_assert(expr, android_log_tag,
2011-02-19 16:18:15 +01:00
"Assertion failure: '%s' in %s:%d",
expr, file, line);
}
void __assert2(const char *file, int line, const char *func, const char *expr) {
__android_log_assert(expr, android_log_tag,
2011-02-19 16:18:15 +01:00
"Assertion failure: '%s' in %s:%d (%s)",
expr, file, line, func);
}
}
#ifdef ANDROID_DEBUG_GL
static const char *getGlErrStr(GLenum error) {
switch (error) {
case GL_INVALID_ENUM:
return "GL_INVALID_ENUM";
case GL_INVALID_VALUE:
return "GL_INVALID_VALUE";
case GL_INVALID_OPERATION:
return "GL_INVALID_OPERATION";
case GL_STACK_OVERFLOW:
return "GL_STACK_OVERFLOW";
case GL_STACK_UNDERFLOW:
return "GL_STACK_UNDERFLOW";
case GL_OUT_OF_MEMORY:
return "GL_OUT_OF_MEMORY";
}
static char buf[40];
snprintf(buf, sizeof(buf), "(Unknown GL error code 0x%x)", error);
return buf;
}
void checkGlError(const char *expr, const char *file, int line) {
GLenum error = glGetError();
if (error != GL_NO_ERROR)
LOGE("GL ERROR: %s on %s (%s:%d)", getGlErrStr(error), expr, file, line);
}
#endif
2011-02-19 16:18:15 +01:00
// floating point. use sparingly
template <class T>
static inline T scalef(T in, float numerator, float denominator) {
return static_cast<float>(in) * numerator / denominator;
}
OSystem_Android *g_sys = 0;
2011-02-19 16:18:15 +01:00
OSystem_Android::OSystem_Android(jobject am) :
_screen_changeid(0),
_force_redraw(false),
_game_texture(0),
_overlay_texture(0),
_mouse_texture(0),
_use_mouse_palette(false),
_show_mouse(false),
_show_overlay(false),
_enable_zoning(false),
_savefile(0),
_mixer(0),
_timer(0),
_fsFactory(new POSIXFilesystemFactory()),
_asset_archive(new AndroidAssetArchive(am)),
_shake_offset(0),
_event_queue_lock(createMutex()) {
}
OSystem_Android::~OSystem_Android() {
ENTER();
2011-02-19 16:18:15 +01:00
delete _game_texture;
delete _overlay_texture;
delete _mouse_texture;
2011-02-19 16:18:15 +01:00
destroyScummVMSurface();
2011-02-19 16:18:15 +01:00
delete _savefile;
delete _mixer;
delete _timer;
delete _fsFactory;
delete _asset_archive;
2011-02-19 16:18:15 +01:00
deleteMutex(_event_queue_lock);
}
bool OSystem_Android::initJavaHooks(JNIEnv *env) {
jclass cls = env->GetObjectClass(back_ptr);
#define FIND_METHOD(name, signature) do { \
MID_ ## name = env->GetMethodID(cls, #name, signature); \
2011-02-19 16:18:15 +01:00
if (MID_ ## name == 0) \
return false; \
} while (0)
FIND_METHOD(setWindowCaption, "(Ljava/lang/String;)V");
FIND_METHOD(displayMessageOnOSD, "(Ljava/lang/String;)V");
FIND_METHOD(initBackend, "()V");
FIND_METHOD(audioSampleRate, "()I");
FIND_METHOD(showVirtualKeyboard, "(Z)V");
FIND_METHOD(getSysArchives, "()[Ljava/lang/String;");
FIND_METHOD(getPluginDirectories, "()[Ljava/lang/String;");
FIND_METHOD(setupScummVMSurface, "()V");
FIND_METHOD(destroyScummVMSurface, "()V");
FIND_METHOD(swapBuffers, "()Z");
#undef FIND_METHOD
return true;
}
2011-02-19 16:18:15 +01:00
void *OSystem_Android::timerThreadFunc(void *arg) {
OSystem_Android *system = (OSystem_Android *)arg;
DefaultTimerManager *timer = (DefaultTimerManager *)(system->_timer);
JNU_AttachThread();
struct timespec tv;
tv.tv_sec = 0;
tv.tv_nsec = 100 * 1000 * 1000; // 100ms
while (!system->_timer_thread_exit) {
timer->handler();
2011-02-19 16:18:15 +01:00
nanosleep(&tv, 0);
}
JNU_DetachThread();
2011-02-19 16:18:15 +01:00
return 0;
}
void OSystem_Android::initBackend() {
ENTER();
2011-02-19 16:18:15 +01:00
JNIEnv *env = JNU_GetEnv();
ConfMan.setInt("autosave_period", 0);
ConfMan.setInt("FM_medium_quality", true);
// must happen before creating TimerManager to avoid race in
// creating EventManager
setupKeymapper();
// BUG: "transient" ConfMan settings get nuked by the options
2011-02-19 16:18:15 +01:00
// screen. Passing the savepath in this way makes it stick
// (via ConfMan.registerDefault)
_savefile = new DefaultSaveFileManager(ConfMan.get("savepath"));
_timer = new DefaultTimerManager();
2011-02-19 16:18:15 +01:00
gettimeofday(&_startTime, 0);
jint sample_rate = env->CallIntMethod(back_ptr, MID_audioSampleRate);
if (env->ExceptionCheck()) {
warning("Error finding audio sample rate - assuming 11025HZ");
2011-02-19 16:18:15 +01:00
env->ExceptionDescribe();
env->ExceptionClear();
2011-02-19 16:18:15 +01:00
sample_rate = 11025;
}
2011-02-19 16:18:15 +01:00
_mixer = new Audio::MixerImpl(this, sample_rate);
_mixer->setReady(true);
env->CallVoidMethod(back_ptr, MID_initBackend);
2011-02-19 16:18:15 +01:00
if (env->ExceptionCheck()) {
error("Error in Java initBackend");
2011-02-19 16:18:15 +01:00
env->ExceptionDescribe();
env->ExceptionClear();
}
_timer_thread_exit = false;
2011-02-19 16:18:15 +01:00
pthread_create(&_timer_thread, 0, timerThreadFunc, this);
OSystem::initBackend();
setupScummVMSurface();
}
void OSystem_Android::addPluginDirectories(Common::FSList &dirs) const {
ENTER();
2011-02-19 16:18:15 +01:00
JNIEnv *env = JNU_GetEnv();
jobjectArray array =
(jobjectArray)env->CallObjectMethod(back_ptr, MID_getPluginDirectories);
if (env->ExceptionCheck()) {
warning("Error finding plugin directories");
2011-02-19 16:18:15 +01:00
env->ExceptionDescribe();
env->ExceptionClear();
2011-02-19 16:18:15 +01:00
return;
}
jsize size = env->GetArrayLength(array);
for (jsize i = 0; i < size; ++i) {
jstring path_obj = (jstring)env->GetObjectArrayElement(array, i);
2011-02-19 16:18:15 +01:00
if (path_obj == 0)
continue;
2011-02-19 16:18:15 +01:00
const char *path = env->GetStringUTFChars(path_obj, 0);
if (path == 0) {
warning("Error getting string characters from plugin directory");
2011-02-19 16:18:15 +01:00
env->ExceptionClear();
env->DeleteLocalRef(path_obj);
2011-02-19 16:18:15 +01:00
continue;
}
2011-02-19 16:18:15 +01:00
dirs.push_back(Common::FSNode(path));
2011-02-19 16:18:15 +01:00
env->ReleaseStringUTFChars(path_obj, path);
env->DeleteLocalRef(path_obj);
}
}
bool OSystem_Android::hasFeature(Feature f) {
return (f == kFeatureCursorHasPalette ||
2011-02-19 16:18:15 +01:00
f == kFeatureVirtualKeyboard ||
f == kFeatureOverlaySupportsAlpha);
}
void OSystem_Android::setFeatureState(Feature f, bool enable) {
ENTER("%d, %d", f, enable);
2011-02-19 16:18:15 +01:00
switch (f) {
case kFeatureVirtualKeyboard:
_virtkeybd_on = enable;
showVirtualKeyboard(enable);
break;
default:
break;
}
}
bool OSystem_Android::getFeatureState(Feature f) {
switch (f) {
case kFeatureVirtualKeyboard:
return _virtkeybd_on;
default:
return false;
}
}
void OSystem_Android::setupKeymapper() {
#ifdef ENABLE_KEYMAPPER
using namespace Common;
Keymapper *mapper = getEventManager()->getKeymapper();
HardwareKeySet *keySet = new HardwareKeySet();
2011-02-19 16:18:15 +01:00
keySet->addHardwareKey(
new HardwareKey("n", KeyState(KEYCODE_n), "n (vk)",
kTriggerLeftKeyType,
kVirtualKeyboardActionType));
2011-02-19 16:18:15 +01:00
mapper->registerHardwareKeySet(keySet);
Keymap *globalMap = new Keymap("global");
Action *act;
act = new Action(globalMap, "VIRT", "Display keyboard",
2011-02-19 16:18:15 +01:00
kVirtualKeyboardActionType);
act->addKeyEvent(KeyState(KEYCODE_F7, ASCII_F7, 0));
mapper->addGlobalKeymap(globalMap);
mapper->pushKeymap("global");
#endif
}
bool OSystem_Android::pollEvent(Common::Event &event) {
//ENTER();
2011-02-19 16:18:15 +01:00
lockMutex(_event_queue_lock);
2011-02-19 16:18:15 +01:00
if (_event_queue.empty()) {
unlockMutex(_event_queue_lock);
return false;
}
2011-02-19 16:18:15 +01:00
event = _event_queue.pop();
unlockMutex(_event_queue_lock);
switch (event.type) {
case Common::EVENT_MOUSEMOVE:
// TODO: only dirty/redraw move bounds
_force_redraw = true;
// fallthrough
case Common::EVENT_LBUTTONDOWN:
case Common::EVENT_LBUTTONUP:
case Common::EVENT_RBUTTONDOWN:
case Common::EVENT_RBUTTONUP:
case Common::EVENT_WHEELUP:
case Common::EVENT_WHEELDOWN:
case Common::EVENT_MBUTTONDOWN:
case Common::EVENT_MBUTTONUP: {
2011-02-19 16:18:15 +01:00
// relative mouse hack
if (event.kbd.flags == 1) {
// Relative (trackball) mouse hack.
const Common::Point& mouse_pos =
getEventManager()->getMousePos();
event.mouse.x += mouse_pos.x;
event.mouse.y += mouse_pos.y;
event.mouse.x = CLIP(event.mouse.x, (int16)0, _show_overlay ?
2011-02-19 16:18:15 +01:00
getOverlayWidth() : getWidth());
event.mouse.y = CLIP(event.mouse.y, (int16)0, _show_overlay ?
2011-02-19 16:18:15 +01:00
getOverlayHeight() : getHeight());
} else {
// Touchscreen events need to be converted
// from device to game coords first.
2011-02-19 16:18:15 +01:00
const GLESTexture *tex = _show_overlay
? static_cast<GLESTexture *>(_overlay_texture)
: static_cast<GLESTexture *>(_game_texture);
event.mouse.x = scalef(event.mouse.x, tex->width(),
2011-02-19 16:18:15 +01:00
_egl_surface_width);
event.mouse.y = scalef(event.mouse.y, tex->height(),
2011-02-19 16:18:15 +01:00
_egl_surface_height);
event.mouse.x -= _shake_offset;
}
break;
}
case Common::EVENT_SCREEN_CHANGED:
debug("EVENT_SCREEN_CHANGED");
_screen_changeid++;
destroyScummVMSurface();
setupScummVMSurface();
break;
default:
break;
}
return true;
}
void OSystem_Android::pushEvent(const Common::Event& event) {
lockMutex(_event_queue_lock);
// Try to combine multiple queued mouse move events
if (event.type == Common::EVENT_MOUSEMOVE &&
2011-02-19 16:18:15 +01:00
!_event_queue.empty() &&
_event_queue.back().type == Common::EVENT_MOUSEMOVE) {
Common::Event tail = _event_queue.back();
if (event.kbd.flags) {
// relative movement hack
tail.mouse.x += event.mouse.x;
tail.mouse.y += event.mouse.y;
} else {
// absolute position, clear relative flag
tail.kbd.flags = 0;
tail.mouse.x = event.mouse.x;
tail.mouse.y = event.mouse.y;
}
} else {
_event_queue.push(event);
2011-02-19 16:18:15 +01:00
}
unlockMutex(_event_queue_lock);
}
uint32 OSystem_Android::getMillis() {
timeval curTime;
2011-02-19 16:18:15 +01:00
gettimeofday(&curTime, 0);
return (uint32)(((curTime.tv_sec - _startTime.tv_sec) * 1000) + \
((curTime.tv_usec - _startTime.tv_usec) / 1000));
}
void OSystem_Android::delayMillis(uint msecs) {
usleep(msecs * 1000);
}
OSystem::MutexRef OSystem_Android::createMutex() {
pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
pthread_mutex_t *mutex = new pthread_mutex_t;
2011-02-19 16:18:15 +01:00
if (pthread_mutex_init(mutex, &attr) != 0) {
warning("pthread_mutex_init() failed");
2011-02-19 16:18:15 +01:00
delete mutex;
2011-02-19 16:18:15 +01:00
return 0;
}
2011-02-19 16:18:15 +01:00
return (MutexRef)mutex;
}
void OSystem_Android::lockMutex(MutexRef mutex) {
2011-02-19 16:18:15 +01:00
if (pthread_mutex_lock((pthread_mutex_t *)mutex) != 0)
warning("pthread_mutex_lock() failed");
}
void OSystem_Android::unlockMutex(MutexRef mutex) {
2011-02-19 16:18:15 +01:00
if (pthread_mutex_unlock((pthread_mutex_t *)mutex) != 0)
warning("pthread_mutex_unlock() failed");
}
void OSystem_Android::deleteMutex(MutexRef mutex) {
2011-02-19 16:18:15 +01:00
pthread_mutex_t *m = (pthread_mutex_t *)mutex;
if (pthread_mutex_destroy(m) != 0)
warning("pthread_mutex_destroy() failed");
else
delete m;
}
void OSystem_Android::quit() {
ENTER();
_timer_thread_exit = true;
2011-02-19 16:18:15 +01:00
pthread_join(_timer_thread, 0);
}
void OSystem_Android::setWindowCaption(const char *caption) {
ENTER("%s", caption);
2011-02-19 16:18:15 +01:00
JNIEnv *env = JNU_GetEnv();
jstring java_caption = env->NewStringUTF(caption);
env->CallVoidMethod(back_ptr, MID_setWindowCaption, java_caption);
2011-02-19 16:18:15 +01:00
if (env->ExceptionCheck()) {
warning("Failed to set window caption");
2011-02-19 16:18:15 +01:00
env->ExceptionDescribe();
env->ExceptionClear();
}
2011-02-19 16:18:15 +01:00
env->DeleteLocalRef(java_caption);
}
void OSystem_Android::displayMessageOnOSD(const char *msg) {
ENTER("%s", msg);
2011-02-19 16:18:15 +01:00
JNIEnv *env = JNU_GetEnv();
jstring java_msg = env->NewStringUTF(msg);
2011-02-19 16:18:15 +01:00
env->CallVoidMethod(back_ptr, MID_displayMessageOnOSD, java_msg);
2011-02-19 16:18:15 +01:00
if (env->ExceptionCheck()) {
warning("Failed to display OSD message");
2011-02-19 16:18:15 +01:00
env->ExceptionDescribe();
env->ExceptionClear();
}
2011-02-19 16:18:15 +01:00
env->DeleteLocalRef(java_msg);
}
void OSystem_Android::showVirtualKeyboard(bool enable) {
ENTER("%d", enable);
2011-02-19 16:18:15 +01:00
JNIEnv *env = JNU_GetEnv();
env->CallVoidMethod(back_ptr, MID_showVirtualKeyboard, enable);
2011-02-19 16:18:15 +01:00
if (env->ExceptionCheck()) {
error("Error trying to show virtual keyboard");
2011-02-19 16:18:15 +01:00
env->ExceptionDescribe();
env->ExceptionClear();
}
}
Common::SaveFileManager *OSystem_Android::getSavefileManager() {
assert(_savefile);
return _savefile;
}
Audio::Mixer *OSystem_Android::getMixer() {
assert(_mixer);
return _mixer;
}
Common::TimerManager *OSystem_Android::getTimerManager() {
assert(_timer);
return _timer;
}
void OSystem_Android::getTimeAndDate(TimeDate &td) const {
struct tm tm;
2011-02-19 16:18:15 +01:00
const time_t curTime = time(0);
localtime_r(&curTime, &tm);
td.tm_sec = tm.tm_sec;
td.tm_min = tm.tm_min;
td.tm_hour = tm.tm_hour;
td.tm_mday = tm.tm_mday;
td.tm_mon = tm.tm_mon;
td.tm_year = tm.tm_year;
}
FilesystemFactory *OSystem_Android::getFilesystemFactory() {
return _fsFactory;
}
void OSystem_Android::addSysArchivesToSearchSet(Common::SearchSet &s,
2011-02-19 16:18:15 +01:00
int priority) {
s.add("ASSET", _asset_archive, priority, false);
2011-02-19 16:18:15 +01:00
JNIEnv *env = JNU_GetEnv();
jobjectArray array =
(jobjectArray)env->CallObjectMethod(back_ptr, MID_getSysArchives);
2011-02-19 16:18:15 +01:00
if (env->ExceptionCheck()) {
warning("Error finding system archive path");
2011-02-19 16:18:15 +01:00
env->ExceptionDescribe();
env->ExceptionClear();
2011-02-19 16:18:15 +01:00
return;
}
jsize size = env->GetArrayLength(array);
for (jsize i = 0; i < size; ++i) {
jstring path_obj = (jstring)env->GetObjectArrayElement(array, i);
2011-02-19 16:18:15 +01:00
const char *path = env->GetStringUTFChars(path_obj, 0);
if (path != 0) {
s.addDirectory(path, path, priority);
env->ReleaseStringUTFChars(path_obj, path);
}
2011-02-19 16:18:15 +01:00
env->DeleteLocalRef(path_obj);
}
}
void OSystem_Android::logMessage(LogMessageType::Type type, const char *message) {
switch (type) {
case LogMessageType::kDebug:
__android_log_write(ANDROID_LOG_DEBUG, android_log_tag, message);
break;
case LogMessageType::kWarning:
__android_log_write(ANDROID_LOG_WARN, android_log_tag, message);
break;
case LogMessageType::kError:
__android_log_write(ANDROID_LOG_ERROR, android_log_tag, message);
break;
}
}
#ifdef DYNAMIC_MODULES
void AndroidPluginProvider::addCustomDirectories(Common::FSList &dirs) const {
g_sys->addPluginDirectories(dirs);
}
#endif
#endif