2 Simple DirectMedia Layer
3 Copyright (C) 1997-2020 Sam Lantinga <slouken@libsdl.org>
5 This software is provided 'as-is', without any express or implied
6 warranty. In no event will the authors be held liable for any damages
7 arising from the use of this software.
9 Permission is granted to anyone to use this software for any purpose,
10 including commercial applications, and to alter it and redistribute it
11 freely, subject to the following restrictions:
13 1. The origin of this software must not be misrepresented; you must not
14 claim that you wrote the original software. If you use this software
15 in a product, an acknowledgment in the product documentation would be
16 appreciated but is not required.
17 2. Altered source versions must be plainly marked as such, and must not be
18 misrepresented as being the original software.
19 3. This notice may not be removed or altered from any source distribution.
21 #include "../../SDL_internal.h"
23 #include "SDL_stdinc.h"
24 #include "SDL_atomic.h"
25 #include "SDL_hints.h"
27 #include "SDL_timer.h"
31 #include "SDL_system.h"
32 #include "SDL_android.h"
34 #include "keyinfotable.h"
36 #include "../../events/SDL_events_c.h"
37 #include "../../video/android/SDL_androidkeyboard.h"
38 #include "../../video/android/SDL_androidmouse.h"
39 #include "../../video/android/SDL_androidtouch.h"
40 #include "../../video/android/SDL_androidvideo.h"
41 #include "../../video/android/SDL_androidwindow.h"
42 #include "../../joystick/android/SDL_sysjoystick_c.h"
43 #include "../../haptic/android/SDL_syshaptic_c.h"
45 #include <android/log.h>
46 #include <android/configuration.h>
47 #include <android/asset_manager_jni.h>
48 #include <sys/system_properties.h>
50 #include <sys/types.h>
54 #define SDL_JAVA_PREFIX org_libsdl_app
55 #define CONCAT1(prefix, class, function) CONCAT2(prefix, class, function)
56 #define CONCAT2(prefix, class, function) Java_ ## prefix ## _ ## class ## _ ## function
57 #define SDL_JAVA_INTERFACE(function) CONCAT1(SDL_JAVA_PREFIX, SDLActivity, function)
58 #define SDL_JAVA_AUDIO_INTERFACE(function) CONCAT1(SDL_JAVA_PREFIX, SDLAudioManager, function)
59 #define SDL_JAVA_CONTROLLER_INTERFACE(function) CONCAT1(SDL_JAVA_PREFIX, SDLControllerManager, function)
60 #define SDL_JAVA_INTERFACE_INPUT_CONNECTION(function) CONCAT1(SDL_JAVA_PREFIX, SDLInputConnection, function)
62 /* Audio encoding definitions */
63 #define ENCODING_PCM_8BIT 3
64 #define ENCODING_PCM_16BIT 2
65 #define ENCODING_PCM_FLOAT 4
67 /* Java class SDLActivity */
68 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetupJNI)(
69 JNIEnv *env, jclass cls);
71 JNIEXPORT int JNICALL SDL_JAVA_INTERFACE(nativeRunMain)(
72 JNIEnv *env, jclass cls,
73 jstring library, jstring function, jobject array);
75 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeDropFile)(
76 JNIEnv *env, jclass jcls,
79 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetScreenResolution)(
80 JNIEnv *env, jclass jcls,
81 jint surfaceWidth, jint surfaceHeight,
82 jint deviceWidth, jint deviceHeight, jint format, jfloat rate);
84 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeResize)(
85 JNIEnv *env, jclass cls);
87 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceCreated)(
88 JNIEnv *env, jclass jcls);
90 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceChanged)(
91 JNIEnv *env, jclass jcls);
93 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceDestroyed)(
94 JNIEnv *env, jclass jcls);
96 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyDown)(
97 JNIEnv *env, jclass jcls,
100 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyUp)(
101 JNIEnv *env, jclass jcls,
104 JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(onNativeSoftReturnKey)(
105 JNIEnv *env, jclass jcls);
107 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyboardFocusLost)(
108 JNIEnv *env, jclass jcls);
110 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeTouch)(
111 JNIEnv *env, jclass jcls,
112 jint touch_device_id_in, jint pointer_finger_id_in,
113 jint action, jfloat x, jfloat y, jfloat p);
115 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeMouse)(
116 JNIEnv *env, jclass jcls,
117 jint button, jint action, jfloat x, jfloat y, jboolean relative);
119 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeAccel)(
120 JNIEnv *env, jclass jcls,
121 jfloat x, jfloat y, jfloat z);
123 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeClipboardChanged)(
124 JNIEnv *env, jclass jcls);
126 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeLowMemory)(
127 JNIEnv *env, jclass cls);
129 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeLocaleChanged)(
130 JNIEnv *env, jclass cls);
132 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSendQuit)(
133 JNIEnv *env, jclass cls);
135 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeQuit)(
136 JNIEnv *env, jclass cls);
138 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePause)(
139 JNIEnv *env, jclass cls);
141 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeResume)(
142 JNIEnv *env, jclass cls);
144 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeFocusChanged)(
145 JNIEnv *env, jclass cls, jboolean hasFocus);
147 JNIEXPORT jstring JNICALL SDL_JAVA_INTERFACE(nativeGetHint)(
148 JNIEnv *env, jclass cls,
151 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetenv)(
152 JNIEnv *env, jclass cls,
153 jstring name, jstring value);
155 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeOrientationChanged)(
156 JNIEnv *env, jclass cls,
159 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeAddTouch)(
160 JNIEnv* env, jclass cls,
161 jint touchId, jstring name);
163 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePermissionResult)(
164 JNIEnv* env, jclass cls,
165 jint requestCode, jboolean result);
167 static JNINativeMethod SDLActivity_tab[] = {
168 { "nativeSetupJNI", "()I", SDL_JAVA_INTERFACE(nativeSetupJNI) },
169 { "nativeRunMain", "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/Object;)I", SDL_JAVA_INTERFACE(nativeRunMain) },
170 { "onNativeDropFile", "(Ljava/lang/String;)V", SDL_JAVA_INTERFACE(onNativeDropFile) },
171 { "nativeSetScreenResolution", "(IIIIIF)V", SDL_JAVA_INTERFACE(nativeSetScreenResolution) },
172 { "onNativeResize", "()V", SDL_JAVA_INTERFACE(onNativeResize) },
173 { "onNativeSurfaceCreated", "()V", SDL_JAVA_INTERFACE(onNativeSurfaceCreated) },
174 { "onNativeSurfaceChanged", "()V", SDL_JAVA_INTERFACE(onNativeSurfaceChanged) },
175 { "onNativeSurfaceDestroyed", "()V", SDL_JAVA_INTERFACE(onNativeSurfaceDestroyed) },
176 { "onNativeKeyDown", "(I)V", SDL_JAVA_INTERFACE(onNativeKeyDown) },
177 { "onNativeKeyUp", "(I)V", SDL_JAVA_INTERFACE(onNativeKeyUp) },
178 { "onNativeSoftReturnKey", "()Z", SDL_JAVA_INTERFACE(onNativeSoftReturnKey) },
179 { "onNativeKeyboardFocusLost", "()V", SDL_JAVA_INTERFACE(onNativeKeyboardFocusLost) },
180 { "onNativeTouch", "(IIIFFF)V", SDL_JAVA_INTERFACE(onNativeTouch) },
181 { "onNativeMouse", "(IIFFZ)V", SDL_JAVA_INTERFACE(onNativeMouse) },
182 { "onNativeAccel", "(FFF)V", SDL_JAVA_INTERFACE(onNativeAccel) },
183 { "onNativeClipboardChanged", "()V", SDL_JAVA_INTERFACE(onNativeClipboardChanged) },
184 { "nativeLowMemory", "()V", SDL_JAVA_INTERFACE(nativeLowMemory) },
185 { "onNativeLocaleChanged", "()V", SDL_JAVA_INTERFACE(onNativeLocaleChanged) },
186 { "nativeSendQuit", "()V", SDL_JAVA_INTERFACE(nativeSendQuit) },
187 { "nativeQuit", "()V", SDL_JAVA_INTERFACE(nativeQuit) },
188 { "nativePause", "()V", SDL_JAVA_INTERFACE(nativePause) },
189 { "nativeResume", "()V", SDL_JAVA_INTERFACE(nativeResume) },
190 { "nativeFocusChanged", "(Z)V", SDL_JAVA_INTERFACE(nativeFocusChanged) },
191 { "nativeGetHint", "(Ljava/lang/String;)Ljava/lang/String;", SDL_JAVA_INTERFACE(nativeGetHint) },
192 { "nativeSetenv", "(Ljava/lang/String;Ljava/lang/String;)V", SDL_JAVA_INTERFACE(nativeSetenv) },
193 { "onNativeOrientationChanged", "(I)V", SDL_JAVA_INTERFACE(onNativeOrientationChanged) },
194 { "nativeAddTouch", "(ILjava/lang/String;)V", SDL_JAVA_INTERFACE(nativeAddTouch) },
195 { "nativePermissionResult", "(IZ)V", SDL_JAVA_INTERFACE(nativePermissionResult) }
198 /* Java class SDLInputConnection */
199 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeCommitText)(
200 JNIEnv *env, jclass cls,
201 jstring text, jint newCursorPosition);
203 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeGenerateScancodeForUnichar)(
204 JNIEnv *env, jclass cls,
207 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeSetComposingText)(
208 JNIEnv *env, jclass cls,
209 jstring text, jint newCursorPosition);
211 static JNINativeMethod SDLInputConnection_tab[] = {
212 { "nativeCommitText", "(Ljava/lang/String;I)V", SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeCommitText) },
213 { "nativeGenerateScancodeForUnichar", "(C)V", SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeGenerateScancodeForUnichar) },
214 { "nativeSetComposingText", "(Ljava/lang/String;I)V", SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeSetComposingText) }
217 /* Java class SDLAudioManager */
218 JNIEXPORT void JNICALL SDL_JAVA_AUDIO_INTERFACE(nativeSetupJNI)(
219 JNIEnv *env, jclass jcls);
221 static JNINativeMethod SDLAudioManager_tab[] = {
222 { "nativeSetupJNI", "()I", SDL_JAVA_AUDIO_INTERFACE(nativeSetupJNI) }
225 /* Java class SDLControllerManager */
226 JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeSetupJNI)(
227 JNIEnv *env, jclass jcls);
229 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadDown)(
230 JNIEnv *env, jclass jcls,
231 jint device_id, jint keycode);
233 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadUp)(
234 JNIEnv *env, jclass jcls,
235 jint device_id, jint keycode);
237 JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoy)(
238 JNIEnv *env, jclass jcls,
239 jint device_id, jint axis, jfloat value);
241 JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeHat)(
242 JNIEnv *env, jclass jcls,
243 jint device_id, jint hat_id, jint x, jint y);
245 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddJoystick)(
246 JNIEnv *env, jclass jcls,
247 jint device_id, jstring device_name, jstring device_desc, jint vendor_id, jint product_id,
248 jboolean is_accelerometer, jint button_mask, jint naxes, jint nhats, jint nballs);
250 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveJoystick)(
251 JNIEnv *env, jclass jcls,
254 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddHaptic)(
255 JNIEnv *env, jclass jcls,
256 jint device_id, jstring device_name);
258 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveHaptic)(
259 JNIEnv *env, jclass jcls,
262 static JNINativeMethod SDLControllerManager_tab[] = {
263 { "nativeSetupJNI", "()I", SDL_JAVA_CONTROLLER_INTERFACE(nativeSetupJNI) },
264 { "onNativePadDown", "(II)I", SDL_JAVA_CONTROLLER_INTERFACE(onNativePadDown) },
265 { "onNativePadUp", "(II)I", SDL_JAVA_CONTROLLER_INTERFACE(onNativePadUp) },
266 { "onNativeJoy", "(IIF)V", SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoy) },
267 { "onNativeHat", "(IIII)V", SDL_JAVA_CONTROLLER_INTERFACE(onNativeHat) },
268 { "nativeAddJoystick", "(ILjava/lang/String;Ljava/lang/String;IIZIIII)I", SDL_JAVA_CONTROLLER_INTERFACE(nativeAddJoystick) },
269 { "nativeRemoveJoystick", "(I)I", SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveJoystick) },
270 { "nativeAddHaptic", "(ILjava/lang/String;)I", SDL_JAVA_CONTROLLER_INTERFACE(nativeAddHaptic) },
271 { "nativeRemoveHaptic", "(I)I", SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveHaptic) }
275 /* Uncomment this to log messages entering and exiting methods in this file */
276 /* #define DEBUG_JNI */
278 static void checkJNIReady(void);
280 /*******************************************************************************
281 This file links the Java side of Android with libsdl
282 *******************************************************************************/
286 /*******************************************************************************
288 *******************************************************************************/
289 static pthread_key_t mThreadKey;
290 static pthread_once_t key_once = PTHREAD_ONCE_INIT;
291 static JavaVM *mJavaVM = NULL;
294 static jclass mActivityClass;
296 /* method signatures */
297 static jmethodID midClipboardGetText;
298 static jmethodID midClipboardHasText;
299 static jmethodID midClipboardSetText;
300 static jmethodID midCreateCustomCursor;
301 static jmethodID midGetContext;
302 static jmethodID midGetDisplayDPI;
303 static jmethodID midGetManifestEnvironmentVariables;
304 static jmethodID midGetNativeSurface;
305 static jmethodID midInitTouch;
306 static jmethodID midIsAndroidTV;
307 static jmethodID midIsChromebook;
308 static jmethodID midIsDeXMode;
309 static jmethodID midIsScreenKeyboardShown;
310 static jmethodID midIsTablet;
311 static jmethodID midManualBackButton;
312 static jmethodID midMinimizeWindow;
313 static jmethodID midOpenURL;
314 static jmethodID midRequestPermission;
315 static jmethodID midSendMessage;
316 static jmethodID midSetActivityTitle;
317 static jmethodID midSetCustomCursor;
318 static jmethodID midSetOrientation;
319 static jmethodID midSetRelativeMouseEnabled;
320 static jmethodID midSetSurfaceViewFormat;
321 static jmethodID midSetSystemCursor;
322 static jmethodID midSetWindowStyle;
323 static jmethodID midShouldMinimizeOnFocusLoss;
324 static jmethodID midShowTextInput;
325 static jmethodID midSupportsRelativeMouse;
328 static jclass mAudioManagerClass;
330 /* method signatures */
331 static jmethodID midAudioOpen;
332 static jmethodID midAudioWriteByteBuffer;
333 static jmethodID midAudioWriteShortBuffer;
334 static jmethodID midAudioWriteFloatBuffer;
335 static jmethodID midAudioClose;
336 static jmethodID midCaptureOpen;
337 static jmethodID midCaptureReadByteBuffer;
338 static jmethodID midCaptureReadShortBuffer;
339 static jmethodID midCaptureReadFloatBuffer;
340 static jmethodID midCaptureClose;
341 static jmethodID midAudioSetThreadPriority;
343 /* controller manager */
344 static jclass mControllerManagerClass;
346 /* method signatures */
347 static jmethodID midPollInputDevices;
348 static jmethodID midPollHapticDevices;
349 static jmethodID midHapticRun;
350 static jmethodID midHapticStop;
352 /* Accelerometer data storage */
353 static SDL_DisplayOrientation displayOrientation;
354 static float fLastAccelerometer[3];
355 static SDL_bool bHasNewData;
357 static SDL_bool bHasEnvironmentVariables;
359 static SDL_atomic_t bPermissionRequestPending;
360 static SDL_bool bPermissionRequestResult;
362 /* Android AssetManager */
363 static void Internal_Android_Create_AssetManager(void);
364 static void Internal_Android_Destroy_AssetManager(void);
365 static AAssetManager *asset_manager = NULL;
366 static jobject javaAssetManagerRef = 0;
368 /*******************************************************************************
369 Functions called by JNI
370 *******************************************************************************/
372 /* From http://developer.android.com/guide/practices/jni.html
373 * All threads are Linux threads, scheduled by the kernel.
374 * They're usually started from managed code (using Thread.start), but they can also be created elsewhere and then
375 * attached to the JavaVM. For example, a thread started with pthread_create can be attached with the
376 * JNI AttachCurrentThread or AttachCurrentThreadAsDaemon functions. Until a thread is attached, it has no JNIEnv,
377 * and cannot make JNI calls.
378 * Attaching a natively-created thread causes a java.lang.Thread object to be constructed and added to the "main"
379 * ThreadGroup, making it visible to the debugger. Calling AttachCurrentThread on an already-attached thread
381 * Note: You can call this function any number of times for the same thread, there's no harm in it
384 /* From http://developer.android.com/guide/practices/jni.html
385 * Threads attached through JNI must call DetachCurrentThread before they exit. If coding this directly is awkward,
386 * in Android 2.0 (Eclair) and higher you can use pthread_key_create to define a destructor function that will be
387 * called before the thread exits, and call DetachCurrentThread from there. (Use that key with pthread_setspecific
388 * to store the JNIEnv in thread-local-storage; that way it'll be passed into your destructor as the argument.)
389 * Note: The destructor is not called unless the stored value is != NULL
390 * Note: You can call this function any number of times for the same thread, there's no harm in it
391 * (except for some lost CPU cycles)
394 /* Set local storage value */
396 Android_JNI_SetEnv(JNIEnv *env) {
397 int status = pthread_setspecific(mThreadKey, env);
399 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed pthread_setspecific() in Android_JNI_SetEnv() (err=%d)", status);
404 /* Get local storage value */
405 JNIEnv* Android_JNI_GetEnv(void)
407 /* Get JNIEnv from the Thread local storage */
408 JNIEnv *env = pthread_getspecific(mThreadKey);
410 /* If it fails, try to attach ! (e.g the thread isn't created with SDL_CreateThread() */
413 /* There should be a JVM */
414 if (mJavaVM == NULL) {
415 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed, there is no JavaVM");
419 /* Attach the current thread to the JVM and get a JNIEnv.
420 * It will be detached by pthread_create destructor 'Android_JNI_ThreadDestroyed' */
421 status = (*mJavaVM)->AttachCurrentThread(mJavaVM, &env, NULL);
423 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to attach current thread (err=%d)", status);
427 /* Save JNIEnv into the Thread local storage */
428 if (Android_JNI_SetEnv(env) < 0) {
436 /* Set up an external thread for using JNI with Android_JNI_GetEnv() */
437 int Android_JNI_SetupThread(void)
442 /* There should be a JVM */
443 if (mJavaVM == NULL) {
444 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed, there is no JavaVM");
448 /* Attach the current thread to the JVM and get a JNIEnv.
449 * It will be detached by pthread_create destructor 'Android_JNI_ThreadDestroyed' */
450 status = (*mJavaVM)->AttachCurrentThread(mJavaVM, &env, NULL);
452 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to attach current thread (err=%d)", status);
456 /* Save JNIEnv into the Thread local storage */
457 if (Android_JNI_SetEnv(env) < 0) {
464 /* Destructor called for each thread where mThreadKey is not NULL */
466 Android_JNI_ThreadDestroyed(void *value)
468 /* The thread is being destroyed, detach it from the Java VM and set the mThreadKey value to NULL as required */
469 JNIEnv *env = (JNIEnv *) value;
471 (*mJavaVM)->DetachCurrentThread(mJavaVM);
472 Android_JNI_SetEnv(NULL);
476 /* Creation of local storage mThreadKey */
478 Android_JNI_CreateKey(void)
480 int status = pthread_key_create(&mThreadKey, Android_JNI_ThreadDestroyed);
482 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Error initializing mThreadKey with pthread_key_create() (err=%d)", status);
487 Android_JNI_CreateKey_once(void)
489 int status = pthread_once(&key_once, Android_JNI_CreateKey);
491 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Error initializing mThreadKey with pthread_once() (err=%d)", status);
496 register_methods(JNIEnv *env, const char *classname, JNINativeMethod *methods, int nb)
498 jclass clazz = (*env)->FindClass(env, classname);
499 if (clazz == NULL || (*env)->RegisterNatives(env, clazz, methods, nb) < 0) {
500 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to register methods of %s", classname);
506 JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved)
511 if ((*mJavaVM)->GetEnv(mJavaVM, (void **)&env, JNI_VERSION_1_4) != JNI_OK) {
512 __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to get JNI Env");
513 return JNI_VERSION_1_4;
516 register_methods(env, "org/libsdl/app/SDLActivity", SDLActivity_tab, SDL_arraysize(SDLActivity_tab));
517 register_methods(env, "org/libsdl/app/SDLInputConnection", SDLInputConnection_tab, SDL_arraysize(SDLInputConnection_tab));
518 register_methods(env, "org/libsdl/app/SDLAudioManager", SDLAudioManager_tab, SDL_arraysize(SDLAudioManager_tab));
519 register_methods(env, "org/libsdl/app/SDLControllerManager", SDLControllerManager_tab, SDL_arraysize(SDLControllerManager_tab));
521 return JNI_VERSION_1_4;
524 void checkJNIReady(void)
526 if (!mActivityClass || !mAudioManagerClass || !mControllerManagerClass) {
527 /* We aren't fully initialized, let's just return. */
534 /* Activity initialization -- called before SDL_main() to initialize JNI bindings */
535 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetupJNI)(JNIEnv *env, jclass cls)
537 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeSetupJNI()");
540 * Create mThreadKey so we can keep track of the JNIEnv assigned to each thread
541 * Refer to http://developer.android.com/guide/practices/design/jni.html for the rationale behind this
543 Android_JNI_CreateKey_once();
545 /* Save JNIEnv of SDLActivity */
546 Android_JNI_SetEnv(env);
548 if (mJavaVM == NULL) {
549 __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to found a JavaVM");
552 /* Use a mutex to prevent concurrency issues between Java Activity and Native thread code, when using 'Android_Window'.
553 * (Eg. Java sending Touch events, while native code is destroying the main SDL_Window. )
555 if (Android_ActivityMutex == NULL) {
556 Android_ActivityMutex = SDL_CreateMutex(); /* Could this be created twice if onCreate() is called a second time ? */
559 if (Android_ActivityMutex == NULL) {
560 __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to create Android_ActivityMutex mutex");
564 Android_PauseSem = SDL_CreateSemaphore(0);
565 if (Android_PauseSem == NULL) {
566 __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to create Android_PauseSem semaphore");
569 Android_ResumeSem = SDL_CreateSemaphore(0);
570 if (Android_ResumeSem == NULL) {
571 __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to create Android_ResumeSem semaphore");
574 mActivityClass = (jclass)((*env)->NewGlobalRef(env, cls));
576 midClipboardGetText = (*env)->GetStaticMethodID(env, mActivityClass, "clipboardGetText", "()Ljava/lang/String;");
577 midClipboardHasText = (*env)->GetStaticMethodID(env, mActivityClass, "clipboardHasText", "()Z");
578 midClipboardSetText = (*env)->GetStaticMethodID(env, mActivityClass, "clipboardSetText", "(Ljava/lang/String;)V");
579 midCreateCustomCursor = (*env)->GetStaticMethodID(env, mActivityClass, "createCustomCursor", "([IIIII)I");
580 midGetContext = (*env)->GetStaticMethodID(env, mActivityClass, "getContext","()Landroid/content/Context;");
581 midGetDisplayDPI = (*env)->GetStaticMethodID(env, mActivityClass, "getDisplayDPI", "()Landroid/util/DisplayMetrics;");
582 midGetManifestEnvironmentVariables = (*env)->GetStaticMethodID(env, mActivityClass, "getManifestEnvironmentVariables", "()Z");
583 midGetNativeSurface = (*env)->GetStaticMethodID(env, mActivityClass, "getNativeSurface","()Landroid/view/Surface;");
584 midInitTouch = (*env)->GetStaticMethodID(env, mActivityClass, "initTouch", "()V");
585 midIsAndroidTV = (*env)->GetStaticMethodID(env, mActivityClass, "isAndroidTV","()Z");
586 midIsChromebook = (*env)->GetStaticMethodID(env, mActivityClass, "isChromebook", "()Z");
587 midIsDeXMode = (*env)->GetStaticMethodID(env, mActivityClass, "isDeXMode", "()Z");
588 midIsScreenKeyboardShown = (*env)->GetStaticMethodID(env, mActivityClass, "isScreenKeyboardShown","()Z");
589 midIsTablet = (*env)->GetStaticMethodID(env, mActivityClass, "isTablet", "()Z");
590 midManualBackButton = (*env)->GetStaticMethodID(env, mActivityClass, "manualBackButton", "()V");
591 midMinimizeWindow = (*env)->GetStaticMethodID(env, mActivityClass, "minimizeWindow","()V");
592 midOpenURL = (*env)->GetStaticMethodID(env, mActivityClass, "openURL", "(Ljava/lang/String;)I");
593 midRequestPermission = (*env)->GetStaticMethodID(env, mActivityClass, "requestPermission", "(Ljava/lang/String;I)V");
594 midSendMessage = (*env)->GetStaticMethodID(env, mActivityClass, "sendMessage", "(II)Z");
595 midSetActivityTitle = (*env)->GetStaticMethodID(env, mActivityClass, "setActivityTitle","(Ljava/lang/String;)Z");
596 midSetCustomCursor = (*env)->GetStaticMethodID(env, mActivityClass, "setCustomCursor", "(I)Z");
597 midSetOrientation = (*env)->GetStaticMethodID(env, mActivityClass, "setOrientation","(IIZLjava/lang/String;)V");
598 midSetRelativeMouseEnabled = (*env)->GetStaticMethodID(env, mActivityClass, "setRelativeMouseEnabled", "(Z)Z");
599 midSetSurfaceViewFormat = (*env)->GetStaticMethodID(env, mActivityClass, "setSurfaceViewFormat","(I)V");
600 midSetSystemCursor = (*env)->GetStaticMethodID(env, mActivityClass, "setSystemCursor", "(I)Z");
601 midSetWindowStyle = (*env)->GetStaticMethodID(env, mActivityClass, "setWindowStyle","(Z)V");
602 midShouldMinimizeOnFocusLoss = (*env)->GetStaticMethodID(env, mActivityClass, "shouldMinimizeOnFocusLoss","()Z");
603 midShowTextInput = (*env)->GetStaticMethodID(env, mActivityClass, "showTextInput", "(IIII)Z");
604 midSupportsRelativeMouse = (*env)->GetStaticMethodID(env, mActivityClass, "supportsRelativeMouse", "()Z");
606 if (!midClipboardGetText ||
607 !midClipboardHasText ||
608 !midClipboardSetText ||
609 !midCreateCustomCursor ||
612 !midGetManifestEnvironmentVariables ||
613 !midGetNativeSurface ||
618 !midIsScreenKeyboardShown ||
620 !midManualBackButton ||
621 !midMinimizeWindow ||
623 !midRequestPermission ||
625 !midSetActivityTitle ||
626 !midSetCustomCursor ||
627 !midSetOrientation ||
628 !midSetRelativeMouseEnabled ||
629 !midSetSurfaceViewFormat ||
630 !midSetSystemCursor ||
631 !midSetWindowStyle ||
632 !midShouldMinimizeOnFocusLoss ||
634 !midSupportsRelativeMouse) {
635 __android_log_print(ANDROID_LOG_WARN, "SDL", "Missing some Java callbacks, do you have the latest version of SDLActivity.java?");
641 /* Audio initialization -- called before SDL_main() to initialize JNI bindings */
642 JNIEXPORT void JNICALL SDL_JAVA_AUDIO_INTERFACE(nativeSetupJNI)(JNIEnv *env, jclass cls)
644 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "AUDIO nativeSetupJNI()");
646 mAudioManagerClass = (jclass)((*env)->NewGlobalRef(env, cls));
648 midAudioOpen = (*env)->GetStaticMethodID(env, mAudioManagerClass,
649 "audioOpen", "(IIII)[I");
650 midAudioWriteByteBuffer = (*env)->GetStaticMethodID(env, mAudioManagerClass,
651 "audioWriteByteBuffer", "([B)V");
652 midAudioWriteShortBuffer = (*env)->GetStaticMethodID(env, mAudioManagerClass,
653 "audioWriteShortBuffer", "([S)V");
654 midAudioWriteFloatBuffer = (*env)->GetStaticMethodID(env, mAudioManagerClass,
655 "audioWriteFloatBuffer", "([F)V");
656 midAudioClose = (*env)->GetStaticMethodID(env, mAudioManagerClass,
657 "audioClose", "()V");
658 midCaptureOpen = (*env)->GetStaticMethodID(env, mAudioManagerClass,
659 "captureOpen", "(IIII)[I");
660 midCaptureReadByteBuffer = (*env)->GetStaticMethodID(env, mAudioManagerClass,
661 "captureReadByteBuffer", "([BZ)I");
662 midCaptureReadShortBuffer = (*env)->GetStaticMethodID(env, mAudioManagerClass,
663 "captureReadShortBuffer", "([SZ)I");
664 midCaptureReadFloatBuffer = (*env)->GetStaticMethodID(env, mAudioManagerClass,
665 "captureReadFloatBuffer", "([FZ)I");
666 midCaptureClose = (*env)->GetStaticMethodID(env, mAudioManagerClass,
667 "captureClose", "()V");
668 midAudioSetThreadPriority = (*env)->GetStaticMethodID(env, mAudioManagerClass,
669 "audioSetThreadPriority", "(ZI)V");
671 if (!midAudioOpen || !midAudioWriteByteBuffer || !midAudioWriteShortBuffer || !midAudioWriteFloatBuffer || !midAudioClose ||
672 !midCaptureOpen || !midCaptureReadByteBuffer || !midCaptureReadShortBuffer || !midCaptureReadFloatBuffer || !midCaptureClose || !midAudioSetThreadPriority) {
673 __android_log_print(ANDROID_LOG_WARN, "SDL", "Missing some Java callbacks, do you have the latest version of SDLAudioManager.java?");
679 /* Controller initialization -- called before SDL_main() to initialize JNI bindings */
680 JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeSetupJNI)(JNIEnv *env, jclass cls)
682 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "CONTROLLER nativeSetupJNI()");
684 mControllerManagerClass = (jclass)((*env)->NewGlobalRef(env, cls));
686 midPollInputDevices = (*env)->GetStaticMethodID(env, mControllerManagerClass,
687 "pollInputDevices", "()V");
688 midPollHapticDevices = (*env)->GetStaticMethodID(env, mControllerManagerClass,
689 "pollHapticDevices", "()V");
690 midHapticRun = (*env)->GetStaticMethodID(env, mControllerManagerClass,
691 "hapticRun", "(IFI)V");
692 midHapticStop = (*env)->GetStaticMethodID(env, mControllerManagerClass,
693 "hapticStop", "(I)V");
695 if (!midPollInputDevices || !midPollHapticDevices || !midHapticRun || !midHapticStop) {
696 __android_log_print(ANDROID_LOG_WARN, "SDL", "Missing some Java callbacks, do you have the latest version of SDLControllerManager.java?");
702 /* SDL main function prototype */
703 typedef int (*SDL_main_func)(int argc, char *argv[]);
705 /* Start up the SDL app */
706 JNIEXPORT int JNICALL SDL_JAVA_INTERFACE(nativeRunMain)(JNIEnv *env, jclass cls, jstring library, jstring function, jobject array)
709 const char *library_file;
710 void *library_handle;
712 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeRunMain()");
714 /* Save JNIEnv of SDLThread */
715 Android_JNI_SetEnv(env);
717 library_file = (*env)->GetStringUTFChars(env, library, NULL);
718 library_handle = dlopen(library_file, RTLD_GLOBAL);
720 if (!library_handle) {
721 /* When deploying android app bundle format uncompressed native libs may not extract from apk to filesystem.
722 In this case we should use lib name without path. https://bugzilla.libsdl.org/show_bug.cgi?id=4739 */
723 const char *library_name = SDL_strrchr(library_file, '/');
724 if (library_name && *library_name) {
726 library_handle = dlopen(library_name, RTLD_GLOBAL);
730 if (library_handle) {
731 const char *function_name;
732 SDL_main_func SDL_main;
734 function_name = (*env)->GetStringUTFChars(env, function, NULL);
735 SDL_main = (SDL_main_func)dlsym(library_handle, function_name);
743 /* Prepare the arguments. */
744 len = (*env)->GetArrayLength(env, array);
745 argv = SDL_small_alloc(char *, 1 + len + 1, &isstack); /* !!! FIXME: check for NULL */
747 /* Use the name "app_process" so PHYSFS_platformCalcBaseDir() works.
748 https://bitbucket.org/MartinFelis/love-android-sdl2/issue/23/release-build-crash-on-start
750 argv[argc++] = SDL_strdup("app_process");
751 for (i = 0; i < len; ++i) {
754 jstring string = (*env)->GetObjectArrayElement(env, array, i);
756 utf = (*env)->GetStringUTFChars(env, string, 0);
758 arg = SDL_strdup(utf);
759 (*env)->ReleaseStringUTFChars(env, string, utf);
761 (*env)->DeleteLocalRef(env, string);
764 arg = SDL_strdup("");
771 /* Run the application. */
772 status = SDL_main(argc, argv);
774 /* Release the arguments. */
775 for (i = 0; i < argc; ++i) {
778 SDL_small_free(argv, isstack);
781 __android_log_print(ANDROID_LOG_ERROR, "SDL", "nativeRunMain(): Couldn't find function %s in library %s", function_name, library_file);
783 (*env)->ReleaseStringUTFChars(env, function, function_name);
785 dlclose(library_handle);
788 __android_log_print(ANDROID_LOG_ERROR, "SDL", "nativeRunMain(): Couldn't load library %s", library_file);
790 (*env)->ReleaseStringUTFChars(env, library, library_file);
792 /* This is a Java thread, it doesn't need to be Detached from the JVM.
793 * Set to mThreadKey value to NULL not to call pthread_create destructor 'Android_JNI_ThreadDestroyed' */
794 Android_JNI_SetEnv(NULL);
796 /* Do not issue an exit or the whole application will terminate instead of just the SDL thread */
803 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeDropFile)(
804 JNIEnv *env, jclass jcls,
807 const char *path = (*env)->GetStringUTFChars(env, filename, NULL);
808 SDL_SendDropFile(NULL, path);
809 (*env)->ReleaseStringUTFChars(env, filename, path);
810 SDL_SendDropComplete(NULL);
813 /* Lock / Unlock Mutex */
814 void Android_ActivityMutex_Lock() {
815 SDL_LockMutex(Android_ActivityMutex);
818 void Android_ActivityMutex_Unlock() {
819 SDL_UnlockMutex(Android_ActivityMutex);
822 /* Lock the Mutex when the Activity is in its 'Running' state */
823 void Android_ActivityMutex_Lock_Running() {
824 int pauseSignaled = 0;
825 int resumeSignaled = 0;
829 SDL_LockMutex(Android_ActivityMutex);
831 pauseSignaled = SDL_SemValue(Android_PauseSem);
832 resumeSignaled = SDL_SemValue(Android_ResumeSem);
834 if (pauseSignaled > resumeSignaled) {
835 SDL_UnlockMutex(Android_ActivityMutex);
841 /* Set screen resolution */
842 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetScreenResolution)(
843 JNIEnv *env, jclass jcls,
844 jint surfaceWidth, jint surfaceHeight,
845 jint deviceWidth, jint deviceHeight, jint format, jfloat rate)
847 SDL_LockMutex(Android_ActivityMutex);
849 Android_SetScreenResolution(surfaceWidth, surfaceHeight, deviceWidth, deviceHeight, format, rate);
851 SDL_UnlockMutex(Android_ActivityMutex);
855 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeResize)(
856 JNIEnv *env, jclass jcls)
858 SDL_LockMutex(Android_ActivityMutex);
862 Android_SendResize(Android_Window);
865 SDL_UnlockMutex(Android_ActivityMutex);
868 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeOrientationChanged)(
869 JNIEnv *env, jclass jcls,
872 SDL_LockMutex(Android_ActivityMutex);
874 displayOrientation = (SDL_DisplayOrientation)orientation;
878 SDL_VideoDisplay *display = SDL_GetDisplay(0);
879 SDL_SendDisplayEvent(display, SDL_DISPLAYEVENT_ORIENTATION, orientation);
882 SDL_UnlockMutex(Android_ActivityMutex);
885 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeAddTouch)(
886 JNIEnv* env, jclass cls,
887 jint touchId, jstring name)
889 const char *utfname = (*env)->GetStringUTFChars(env, name, NULL);
891 SDL_AddTouch((SDL_TouchID) touchId, SDL_TOUCH_DEVICE_DIRECT, utfname);
893 (*env)->ReleaseStringUTFChars(env, name, utfname);
896 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePermissionResult)(
897 JNIEnv* env, jclass cls,
898 jint requestCode, jboolean result)
900 bPermissionRequestResult = result;
901 SDL_AtomicSet(&bPermissionRequestPending, SDL_FALSE);
905 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadDown)(
906 JNIEnv *env, jclass jcls,
907 jint device_id, jint keycode)
909 return Android_OnPadDown(device_id, keycode);
913 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadUp)(
914 JNIEnv *env, jclass jcls,
915 jint device_id, jint keycode)
917 return Android_OnPadUp(device_id, keycode);
921 JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoy)(
922 JNIEnv *env, jclass jcls,
923 jint device_id, jint axis, jfloat value)
925 Android_OnJoy(device_id, axis, value);
929 JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeHat)(
930 JNIEnv *env, jclass jcls,
931 jint device_id, jint hat_id, jint x, jint y)
933 Android_OnHat(device_id, hat_id, x, y);
937 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddJoystick)(
938 JNIEnv *env, jclass jcls,
939 jint device_id, jstring device_name, jstring device_desc,
940 jint vendor_id, jint product_id, jboolean is_accelerometer,
941 jint button_mask, jint naxes, jint nhats, jint nballs)
944 const char *name = (*env)->GetStringUTFChars(env, device_name, NULL);
945 const char *desc = (*env)->GetStringUTFChars(env, device_desc, NULL);
947 retval = Android_AddJoystick(device_id, name, desc, vendor_id, product_id, is_accelerometer ? SDL_TRUE : SDL_FALSE, button_mask, naxes, nhats, nballs);
949 (*env)->ReleaseStringUTFChars(env, device_name, name);
950 (*env)->ReleaseStringUTFChars(env, device_desc, desc);
955 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveJoystick)(
956 JNIEnv *env, jclass jcls,
959 return Android_RemoveJoystick(device_id);
962 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddHaptic)(
963 JNIEnv *env, jclass jcls, jint device_id, jstring device_name)
966 const char *name = (*env)->GetStringUTFChars(env, device_name, NULL);
968 retval = Android_AddHaptic(device_id, name);
970 (*env)->ReleaseStringUTFChars(env, device_name, name);
975 JNIEXPORT jint JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveHaptic)(
976 JNIEnv *env, jclass jcls, jint device_id)
978 return Android_RemoveHaptic(device_id);
981 /* Called from surfaceCreated() */
982 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceCreated)(JNIEnv *env, jclass jcls)
984 SDL_LockMutex(Android_ActivityMutex);
988 SDL_WindowData *data = (SDL_WindowData *) Android_Window->driverdata;
990 data->native_window = Android_JNI_GetNativeWindow();
991 if (data->native_window == NULL) {
992 SDL_SetError("Could not fetch native window from UI thread");
996 SDL_UnlockMutex(Android_ActivityMutex);
999 /* Called from surfaceChanged() */
1000 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceChanged)(JNIEnv *env, jclass jcls)
1002 SDL_LockMutex(Android_ActivityMutex);
1006 SDL_VideoDevice *_this = SDL_GetVideoDevice();
1007 SDL_WindowData *data = (SDL_WindowData *) Android_Window->driverdata;
1009 /* If the surface has been previously destroyed by onNativeSurfaceDestroyed, recreate it here */
1010 if (data->egl_surface == EGL_NO_SURFACE) {
1011 data->egl_surface = SDL_EGL_CreateSurface(_this, (NativeWindowType) data->native_window);
1014 /* GL Context handling is done in the event loop because this function is run from the Java thread */
1017 SDL_UnlockMutex(Android_ActivityMutex);
1020 /* Called from surfaceDestroyed() */
1021 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceDestroyed)(JNIEnv *env, jclass jcls)
1023 int nb_attempt = 50;
1027 SDL_LockMutex(Android_ActivityMutex);
1031 SDL_VideoDevice *_this = SDL_GetVideoDevice();
1032 SDL_WindowData *data = (SDL_WindowData *) Android_Window->driverdata;
1034 /* Wait for Main thread being paused and context un-activated to release 'egl_surface' */
1035 if (! data->backup_done) {
1037 if (nb_attempt == 0) {
1038 SDL_SetError("Try to release egl_surface with context probably still active");
1040 SDL_UnlockMutex(Android_ActivityMutex);
1046 if (data->egl_surface != EGL_NO_SURFACE) {
1047 SDL_EGL_DestroySurface(_this, data->egl_surface);
1048 data->egl_surface = EGL_NO_SURFACE;
1051 if (data->native_window) {
1052 ANativeWindow_release(data->native_window);
1053 data->native_window = NULL;
1056 /* GL Context handling is done in the event loop because this function is run from the Java thread */
1059 SDL_UnlockMutex(Android_ActivityMutex);
1063 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyDown)(
1064 JNIEnv *env, jclass jcls,
1067 Android_OnKeyDown(keycode);
1071 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyUp)(
1072 JNIEnv *env, jclass jcls,
1075 Android_OnKeyUp(keycode);
1078 /* Virtual keyboard return key might stop text input */
1079 JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(onNativeSoftReturnKey)(
1080 JNIEnv *env, jclass jcls)
1082 if (SDL_GetHintBoolean(SDL_HINT_RETURN_KEY_HIDES_IME, SDL_FALSE)) {
1083 SDL_StopTextInput();
1089 /* Keyboard Focus Lost */
1090 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyboardFocusLost)(
1091 JNIEnv *env, jclass jcls)
1093 /* Calling SDL_StopTextInput will take care of hiding the keyboard and cleaning up the DummyText widget */
1094 SDL_StopTextInput();
1099 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeTouch)(
1100 JNIEnv *env, jclass jcls,
1101 jint touch_device_id_in, jint pointer_finger_id_in,
1102 jint action, jfloat x, jfloat y, jfloat p)
1104 SDL_LockMutex(Android_ActivityMutex);
1106 Android_OnTouch(Android_Window, touch_device_id_in, pointer_finger_id_in, action, x, y, p);
1108 SDL_UnlockMutex(Android_ActivityMutex);
1112 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeMouse)(
1113 JNIEnv *env, jclass jcls,
1114 jint button, jint action, jfloat x, jfloat y, jboolean relative)
1116 SDL_LockMutex(Android_ActivityMutex);
1118 Android_OnMouse(Android_Window, button, action, x, y, relative);
1120 SDL_UnlockMutex(Android_ActivityMutex);
1124 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeAccel)(
1125 JNIEnv *env, jclass jcls,
1126 jfloat x, jfloat y, jfloat z)
1128 fLastAccelerometer[0] = x;
1129 fLastAccelerometer[1] = y;
1130 fLastAccelerometer[2] = z;
1131 bHasNewData = SDL_TRUE;
1135 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeClipboardChanged)(
1136 JNIEnv *env, jclass jcls)
1138 SDL_SendClipboardUpdate();
1142 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeLowMemory)(
1143 JNIEnv *env, jclass cls)
1145 SDL_SendAppEvent(SDL_APP_LOWMEMORY);
1149 * requires android:configChanges="layoutDirection|locale" in AndroidManifest.xml */
1150 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeLocaleChanged)(
1151 JNIEnv *env, jclass cls)
1153 SDL_SendAppEvent(SDL_LOCALECHANGED);
1157 /* Send Quit event to "SDLThread" thread */
1158 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSendQuit)(
1159 JNIEnv *env, jclass cls)
1161 /* Discard previous events. The user should have handled state storage
1162 * in SDL_APP_WILLENTERBACKGROUND. After nativeSendQuit() is called, no
1163 * events other than SDL_QUIT and SDL_APP_TERMINATING should fire */
1164 SDL_FlushEvents(SDL_FIRSTEVENT, SDL_LASTEVENT);
1165 /* Inject a SDL_QUIT event */
1167 SDL_SendAppEvent(SDL_APP_TERMINATING);
1168 /* Robustness: clear any pending Pause */
1169 while (SDL_SemTryWait(Android_PauseSem) == 0) {
1172 /* Resume the event loop so that the app can catch SDL_QUIT which
1173 * should now be the top event in the event queue. */
1174 SDL_SemPost(Android_ResumeSem);
1178 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeQuit)(
1179 JNIEnv *env, jclass cls)
1183 if (Android_ActivityMutex) {
1184 SDL_DestroyMutex(Android_ActivityMutex);
1185 Android_ActivityMutex = NULL;
1188 if (Android_PauseSem) {
1189 SDL_DestroySemaphore(Android_PauseSem);
1190 Android_PauseSem = NULL;
1193 if (Android_ResumeSem) {
1194 SDL_DestroySemaphore(Android_ResumeSem);
1195 Android_ResumeSem = NULL;
1198 Internal_Android_Destroy_AssetManager();
1200 str = SDL_GetError();
1201 if (str && str[0]) {
1202 __android_log_print(ANDROID_LOG_ERROR, "SDL", "SDLActivity thread ends (error=%s)", str);
1204 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "SDLActivity thread ends");
1209 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePause)(
1210 JNIEnv *env, jclass cls)
1212 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativePause()");
1214 /* Signal the pause semaphore so the event loop knows to pause and (optionally) block itself.
1215 * Sometimes 2 pauses can be queued (eg pause/resume/pause), so it's always increased. */
1216 SDL_SemPost(Android_PauseSem);
1220 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeResume)(
1221 JNIEnv *env, jclass cls)
1223 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeResume()");
1225 /* Signal the resume semaphore so the event loop knows to resume and restore the GL Context
1226 * We can't restore the GL Context here because it needs to be done on the SDL main thread
1227 * and this function will be called from the Java thread instead.
1229 SDL_SemPost(Android_ResumeSem);
1232 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeFocusChanged)(
1233 JNIEnv *env, jclass cls, jboolean hasFocus)
1235 SDL_LockMutex(Android_ActivityMutex);
1237 if (Android_Window) {
1238 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeFocusChanged()");
1239 SDL_SendWindowEvent(Android_Window, (hasFocus ? SDL_WINDOWEVENT_FOCUS_GAINED : SDL_WINDOWEVENT_FOCUS_LOST), 0, 0);
1242 SDL_UnlockMutex(Android_ActivityMutex);
1245 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeCommitText)(
1246 JNIEnv *env, jclass cls,
1247 jstring text, jint newCursorPosition)
1249 const char *utftext = (*env)->GetStringUTFChars(env, text, NULL);
1251 SDL_SendKeyboardText(utftext);
1253 (*env)->ReleaseStringUTFChars(env, text, utftext);
1256 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeGenerateScancodeForUnichar)(
1257 JNIEnv *env, jclass cls,
1260 SDL_Scancode code = SDL_SCANCODE_UNKNOWN;
1263 /* We do not care about bigger than 127. */
1264 if (chUnicode < 127) {
1265 AndroidKeyInfo info = unicharToAndroidKeyInfoTable[chUnicode];
1270 if (mod & KMOD_SHIFT) {
1271 /* If character uses shift, press shift down */
1272 SDL_SendKeyboardKey(SDL_PRESSED, SDL_SCANCODE_LSHIFT);
1275 /* send a keydown and keyup even for the character */
1276 SDL_SendKeyboardKey(SDL_PRESSED, code);
1277 SDL_SendKeyboardKey(SDL_RELEASED, code);
1279 if (mod & KMOD_SHIFT) {
1280 /* If character uses shift, press shift back up */
1281 SDL_SendKeyboardKey(SDL_RELEASED, SDL_SCANCODE_LSHIFT);
1285 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeSetComposingText)(
1286 JNIEnv *env, jclass cls,
1287 jstring text, jint newCursorPosition)
1289 const char *utftext = (*env)->GetStringUTFChars(env, text, NULL);
1291 SDL_SendEditingText(utftext, 0, 0);
1293 (*env)->ReleaseStringUTFChars(env, text, utftext);
1296 JNIEXPORT jstring JNICALL SDL_JAVA_INTERFACE(nativeGetHint)(
1297 JNIEnv *env, jclass cls,
1300 const char *utfname = (*env)->GetStringUTFChars(env, name, NULL);
1301 const char *hint = SDL_GetHint(utfname);
1303 jstring result = (*env)->NewStringUTF(env, hint);
1304 (*env)->ReleaseStringUTFChars(env, name, utfname);
1309 JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetenv)(
1310 JNIEnv *env, jclass cls,
1311 jstring name, jstring value)
1313 const char *utfname = (*env)->GetStringUTFChars(env, name, NULL);
1314 const char *utfvalue = (*env)->GetStringUTFChars(env, value, NULL);
1316 SDL_setenv(utfname, utfvalue, 1);
1318 (*env)->ReleaseStringUTFChars(env, name, utfname);
1319 (*env)->ReleaseStringUTFChars(env, value, utfvalue);
1323 /*******************************************************************************
1324 Functions called by SDL into Java
1325 *******************************************************************************/
1327 static SDL_atomic_t s_active;
1328 struct LocalReferenceHolder
1334 static struct LocalReferenceHolder LocalReferenceHolder_Setup(const char *func)
1336 struct LocalReferenceHolder refholder;
1337 refholder.m_env = NULL;
1338 refholder.m_func = func;
1340 SDL_Log("Entering function %s", func);
1345 static SDL_bool LocalReferenceHolder_Init(struct LocalReferenceHolder *refholder, JNIEnv *env)
1347 const int capacity = 16;
1348 if ((*env)->PushLocalFrame(env, capacity) < 0) {
1349 SDL_SetError("Failed to allocate enough JVM local references");
1352 SDL_AtomicIncRef(&s_active);
1353 refholder->m_env = env;
1357 static void LocalReferenceHolder_Cleanup(struct LocalReferenceHolder *refholder)
1360 SDL_Log("Leaving function %s", refholder->m_func);
1362 if (refholder->m_env) {
1363 JNIEnv *env = refholder->m_env;
1364 (*env)->PopLocalFrame(env, NULL);
1365 SDL_AtomicDecRef(&s_active);
1369 ANativeWindow* Android_JNI_GetNativeWindow(void)
1371 ANativeWindow *anw = NULL;
1373 JNIEnv *env = Android_JNI_GetEnv();
1375 s = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetNativeSurface);
1377 anw = ANativeWindow_fromSurface(env, s);
1378 (*env)->DeleteLocalRef(env, s);
1384 void Android_JNI_SetSurfaceViewFormat(int format)
1386 JNIEnv *env = Android_JNI_GetEnv();
1389 /* Format from android/native_window.h,
1390 * convert to temporary arbitrary values,
1391 * then to java PixelFormat */
1392 if (format == WINDOW_FORMAT_RGBA_8888) {
1394 } else if (format == WINDOW_FORMAT_RGBX_8888) {
1396 } else if (format == WINDOW_FORMAT_RGB_565) {
1401 (*env)->CallStaticVoidMethod(env, mActivityClass, midSetSurfaceViewFormat, new_format);
1404 void Android_JNI_SetActivityTitle(const char *title)
1406 JNIEnv *env = Android_JNI_GetEnv();
1408 jstring jtitle = (*env)->NewStringUTF(env, title);
1409 (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetActivityTitle, jtitle);
1410 (*env)->DeleteLocalRef(env, jtitle);
1413 void Android_JNI_SetWindowStyle(SDL_bool fullscreen)
1415 JNIEnv *env = Android_JNI_GetEnv();
1416 (*env)->CallStaticVoidMethod(env, mActivityClass, midSetWindowStyle, fullscreen ? 1 : 0);
1419 void Android_JNI_SetOrientation(int w, int h, int resizable, const char *hint)
1421 JNIEnv *env = Android_JNI_GetEnv();
1423 jstring jhint = (*env)->NewStringUTF(env, (hint ? hint : ""));
1424 (*env)->CallStaticVoidMethod(env, mActivityClass, midSetOrientation, w, h, (resizable? 1 : 0), jhint);
1425 (*env)->DeleteLocalRef(env, jhint);
1428 void Android_JNI_MinizeWindow()
1430 JNIEnv *env = Android_JNI_GetEnv();
1431 (*env)->CallStaticVoidMethod(env, mActivityClass, midMinimizeWindow);
1434 SDL_bool Android_JNI_ShouldMinimizeOnFocusLoss()
1436 JNIEnv *env = Android_JNI_GetEnv();
1437 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midShouldMinimizeOnFocusLoss);
1440 SDL_bool Android_JNI_GetAccelerometerValues(float values[3])
1443 SDL_bool retval = SDL_FALSE;
1446 for (i = 0; i < 3; ++i) {
1447 values[i] = fLastAccelerometer[i];
1449 bHasNewData = SDL_FALSE;
1459 static int audioBufferFormat = 0;
1460 static jobject audioBuffer = NULL;
1461 static void *audioBufferPinned = NULL;
1462 static int captureBufferFormat = 0;
1463 static jobject captureBuffer = NULL;
1465 int Android_JNI_OpenAudioDevice(int iscapture, SDL_AudioSpec *spec)
1468 jobject jbufobj = NULL;
1470 int *resultElements;
1473 JNIEnv *env = Android_JNI_GetEnv();
1475 switch (spec->format) {
1477 audioformat = ENCODING_PCM_8BIT;
1480 audioformat = ENCODING_PCM_16BIT;
1483 audioformat = ENCODING_PCM_FLOAT;
1486 return SDL_SetError("Unsupported audio format: 0x%x", spec->format);
1490 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "SDL audio: opening device for capture");
1491 result = (*env)->CallStaticObjectMethod(env, mAudioManagerClass, midCaptureOpen, spec->freq, audioformat, spec->channels, spec->samples);
1493 __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "SDL audio: opening device for output");
1494 result = (*env)->CallStaticObjectMethod(env, mAudioManagerClass, midAudioOpen, spec->freq, audioformat, spec->channels, spec->samples);
1496 if (result == NULL) {
1497 /* Error during audio initialization, error printed from Java */
1498 return SDL_SetError("Java-side initialization failed");
1501 if ((*env)->GetArrayLength(env, (jintArray)result) != 4) {
1502 return SDL_SetError("Unexpected results from Java, expected 4, got %d", (*env)->GetArrayLength(env, (jintArray)result));
1505 resultElements = (*env)->GetIntArrayElements(env, (jintArray)result, &isCopy);
1506 spec->freq = resultElements[0];
1507 audioformat = resultElements[1];
1508 switch (audioformat) {
1509 case ENCODING_PCM_8BIT:
1510 spec->format = AUDIO_U8;
1512 case ENCODING_PCM_16BIT:
1513 spec->format = AUDIO_S16;
1515 case ENCODING_PCM_FLOAT:
1516 spec->format = AUDIO_F32;
1519 return SDL_SetError("Unexpected audio format from Java: %d\n", audioformat);
1521 spec->channels = resultElements[2];
1522 spec->samples = resultElements[3];
1523 (*env)->ReleaseIntArrayElements(env, (jintArray)result, resultElements, JNI_ABORT);
1524 (*env)->DeleteLocalRef(env, result);
1526 /* Allocating the audio buffer from the Java side and passing it as the return value for audioInit no longer works on
1527 * Android >= 4.2 due to a "stale global reference" error. So now we allocate this buffer directly from this side. */
1528 switch (audioformat) {
1529 case ENCODING_PCM_8BIT:
1531 jbyteArray audioBufferLocal = (*env)->NewByteArray(env, spec->samples * spec->channels);
1532 if (audioBufferLocal) {
1533 jbufobj = (*env)->NewGlobalRef(env, audioBufferLocal);
1534 (*env)->DeleteLocalRef(env, audioBufferLocal);
1538 case ENCODING_PCM_16BIT:
1540 jshortArray audioBufferLocal = (*env)->NewShortArray(env, spec->samples * spec->channels);
1541 if (audioBufferLocal) {
1542 jbufobj = (*env)->NewGlobalRef(env, audioBufferLocal);
1543 (*env)->DeleteLocalRef(env, audioBufferLocal);
1547 case ENCODING_PCM_FLOAT:
1549 jfloatArray audioBufferLocal = (*env)->NewFloatArray(env, spec->samples * spec->channels);
1550 if (audioBufferLocal) {
1551 jbufobj = (*env)->NewGlobalRef(env, audioBufferLocal);
1552 (*env)->DeleteLocalRef(env, audioBufferLocal);
1557 return SDL_SetError("Unexpected audio format from Java: %d\n", audioformat);
1560 if (jbufobj == NULL) {
1561 __android_log_print(ANDROID_LOG_WARN, "SDL", "SDL audio: could not allocate an audio buffer");
1562 return SDL_OutOfMemory();
1566 captureBufferFormat = audioformat;
1567 captureBuffer = jbufobj;
1569 audioBufferFormat = audioformat;
1570 audioBuffer = jbufobj;
1576 switch (audioformat) {
1577 case ENCODING_PCM_8BIT:
1578 audioBufferPinned = (*env)->GetByteArrayElements(env, (jbyteArray)audioBuffer, &isCopy);
1580 case ENCODING_PCM_16BIT:
1581 audioBufferPinned = (*env)->GetShortArrayElements(env, (jshortArray)audioBuffer, &isCopy);
1583 case ENCODING_PCM_FLOAT:
1584 audioBufferPinned = (*env)->GetFloatArrayElements(env, (jfloatArray)audioBuffer, &isCopy);
1587 return SDL_SetError("Unexpected audio format from Java: %d\n", audioformat);
1593 SDL_DisplayOrientation Android_JNI_GetDisplayOrientation(void)
1595 return displayOrientation;
1598 int Android_JNI_GetDisplayDPI(float *ddpi, float *xdpi, float *ydpi)
1600 JNIEnv *env = Android_JNI_GetEnv();
1602 jobject jDisplayObj = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetDisplayDPI);
1603 jclass jDisplayClass = (*env)->GetObjectClass(env, jDisplayObj);
1605 jfieldID fidXdpi = (*env)->GetFieldID(env, jDisplayClass, "xdpi", "F");
1606 jfieldID fidYdpi = (*env)->GetFieldID(env, jDisplayClass, "ydpi", "F");
1607 jfieldID fidDdpi = (*env)->GetFieldID(env, jDisplayClass, "densityDpi", "I");
1609 float nativeXdpi = (*env)->GetFloatField(env, jDisplayObj, fidXdpi);
1610 float nativeYdpi = (*env)->GetFloatField(env, jDisplayObj, fidYdpi);
1611 int nativeDdpi = (*env)->GetIntField(env, jDisplayObj, fidDdpi);
1614 (*env)->DeleteLocalRef(env, jDisplayObj);
1615 (*env)->DeleteLocalRef(env, jDisplayClass);
1618 *ddpi = (float)nativeDdpi;
1630 void * Android_JNI_GetAudioBuffer(void)
1632 return audioBufferPinned;
1635 void Android_JNI_WriteAudioBuffer(void)
1637 JNIEnv *env = Android_JNI_GetEnv();
1639 switch (audioBufferFormat) {
1640 case ENCODING_PCM_8BIT:
1641 (*env)->ReleaseByteArrayElements(env, (jbyteArray)audioBuffer, (jbyte *)audioBufferPinned, JNI_COMMIT);
1642 (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midAudioWriteByteBuffer, (jbyteArray)audioBuffer);
1644 case ENCODING_PCM_16BIT:
1645 (*env)->ReleaseShortArrayElements(env, (jshortArray)audioBuffer, (jshort *)audioBufferPinned, JNI_COMMIT);
1646 (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midAudioWriteShortBuffer, (jshortArray)audioBuffer);
1648 case ENCODING_PCM_FLOAT:
1649 (*env)->ReleaseFloatArrayElements(env, (jfloatArray)audioBuffer, (jfloat *)audioBufferPinned, JNI_COMMIT);
1650 (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midAudioWriteFloatBuffer, (jfloatArray)audioBuffer);
1653 __android_log_print(ANDROID_LOG_WARN, "SDL", "SDL audio: unhandled audio buffer format");
1657 /* JNI_COMMIT means the changes are committed to the VM but the buffer remains pinned */
1660 int Android_JNI_CaptureAudioBuffer(void *buffer, int buflen)
1662 JNIEnv *env = Android_JNI_GetEnv();
1663 jboolean isCopy = JNI_FALSE;
1666 switch (captureBufferFormat) {
1667 case ENCODING_PCM_8BIT:
1668 SDL_assert((*env)->GetArrayLength(env, (jshortArray)captureBuffer) == buflen);
1669 br = (*env)->CallStaticIntMethod(env, mAudioManagerClass, midCaptureReadByteBuffer, (jbyteArray)captureBuffer, JNI_TRUE);
1671 jbyte *ptr = (*env)->GetByteArrayElements(env, (jbyteArray)captureBuffer, &isCopy);
1672 SDL_memcpy(buffer, ptr, br);
1673 (*env)->ReleaseByteArrayElements(env, (jbyteArray)captureBuffer, ptr, JNI_ABORT);
1676 case ENCODING_PCM_16BIT:
1677 SDL_assert((*env)->GetArrayLength(env, (jshortArray)captureBuffer) == (buflen / sizeof(Sint16)));
1678 br = (*env)->CallStaticIntMethod(env, mAudioManagerClass, midCaptureReadShortBuffer, (jshortArray)captureBuffer, JNI_TRUE);
1680 jshort *ptr = (*env)->GetShortArrayElements(env, (jshortArray)captureBuffer, &isCopy);
1681 br *= sizeof(Sint16);
1682 SDL_memcpy(buffer, ptr, br);
1683 (*env)->ReleaseShortArrayElements(env, (jshortArray)captureBuffer, ptr, JNI_ABORT);
1686 case ENCODING_PCM_FLOAT:
1687 SDL_assert((*env)->GetArrayLength(env, (jfloatArray)captureBuffer) == (buflen / sizeof(float)));
1688 br = (*env)->CallStaticIntMethod(env, mAudioManagerClass, midCaptureReadFloatBuffer, (jfloatArray)captureBuffer, JNI_TRUE);
1690 jfloat *ptr = (*env)->GetFloatArrayElements(env, (jfloatArray)captureBuffer, &isCopy);
1691 br *= sizeof(float);
1692 SDL_memcpy(buffer, ptr, br);
1693 (*env)->ReleaseFloatArrayElements(env, (jfloatArray)captureBuffer, ptr, JNI_ABORT);
1697 __android_log_print(ANDROID_LOG_WARN, "SDL", "SDL audio: unhandled capture buffer format");
1703 void Android_JNI_FlushCapturedAudio(void)
1705 JNIEnv *env = Android_JNI_GetEnv();
1706 #if 0 /* !!! FIXME: this needs API 23, or it'll do blocking reads and never end. */
1707 switch (captureBufferFormat) {
1708 case ENCODING_PCM_8BIT:
1710 const jint len = (*env)->GetArrayLength(env, (jbyteArray)captureBuffer);
1711 while ((*env)->CallStaticIntMethod(env, mActivityClass, midCaptureReadByteBuffer, (jbyteArray)captureBuffer, JNI_FALSE) == len) { /* spin */ }
1714 case ENCODING_PCM_16BIT:
1716 const jint len = (*env)->GetArrayLength(env, (jshortArray)captureBuffer);
1717 while ((*env)->CallStaticIntMethod(env, mActivityClass, midCaptureReadShortBuffer, (jshortArray)captureBuffer, JNI_FALSE) == len) { /* spin */ }
1720 case ENCODING_PCM_FLOAT:
1722 const jint len = (*env)->GetArrayLength(env, (jfloatArray)captureBuffer);
1723 while ((*env)->CallStaticIntMethod(env, mActivityClass, midCaptureReadFloatBuffer, (jfloatArray)captureBuffer, JNI_FALSE) == len) { /* spin */ }
1727 __android_log_print(ANDROID_LOG_WARN, "SDL", "SDL audio: flushing unhandled capture buffer format");
1731 switch (captureBufferFormat) {
1732 case ENCODING_PCM_8BIT:
1733 (*env)->CallStaticIntMethod(env, mAudioManagerClass, midCaptureReadByteBuffer, (jbyteArray)captureBuffer, JNI_FALSE);
1735 case ENCODING_PCM_16BIT:
1736 (*env)->CallStaticIntMethod(env, mAudioManagerClass, midCaptureReadShortBuffer, (jshortArray)captureBuffer, JNI_FALSE);
1738 case ENCODING_PCM_FLOAT:
1739 (*env)->CallStaticIntMethod(env, mAudioManagerClass, midCaptureReadFloatBuffer, (jfloatArray)captureBuffer, JNI_FALSE);
1742 __android_log_print(ANDROID_LOG_WARN, "SDL", "SDL audio: flushing unhandled capture buffer format");
1748 void Android_JNI_CloseAudioDevice(const int iscapture)
1750 JNIEnv *env = Android_JNI_GetEnv();
1753 (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midCaptureClose);
1754 if (captureBuffer) {
1755 (*env)->DeleteGlobalRef(env, captureBuffer);
1756 captureBuffer = NULL;
1759 (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midAudioClose);
1761 (*env)->DeleteGlobalRef(env, audioBuffer);
1763 audioBufferPinned = NULL;
1768 void Android_JNI_AudioSetThreadPriority(int iscapture, int device_id)
1770 JNIEnv *env = Android_JNI_GetEnv();
1771 (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midAudioSetThreadPriority, iscapture, device_id);
1774 /* Test for an exception and call SDL_SetError with its detail if one occurs */
1775 /* If the parameter silent is truthy then SDL_SetError() will not be called. */
1776 static SDL_bool Android_JNI_ExceptionOccurred(SDL_bool silent)
1778 JNIEnv *env = Android_JNI_GetEnv();
1779 jthrowable exception;
1781 /* Detect mismatch LocalReferenceHolder_Init/Cleanup */
1782 SDL_assert(SDL_AtomicGet(&s_active) > 0);
1784 exception = (*env)->ExceptionOccurred(env);
1785 if (exception != NULL) {
1788 /* Until this happens most JNI operations have undefined behaviour */
1789 (*env)->ExceptionClear(env);
1792 jclass exceptionClass = (*env)->GetObjectClass(env, exception);
1793 jclass classClass = (*env)->FindClass(env, "java/lang/Class");
1794 jstring exceptionName;
1795 const char *exceptionNameUTF8;
1796 jstring exceptionMessage;
1798 mid = (*env)->GetMethodID(env, classClass, "getName", "()Ljava/lang/String;");
1799 exceptionName = (jstring)(*env)->CallObjectMethod(env, exceptionClass, mid);
1800 exceptionNameUTF8 = (*env)->GetStringUTFChars(env, exceptionName, 0);
1802 mid = (*env)->GetMethodID(env, exceptionClass, "getMessage", "()Ljava/lang/String;");
1803 exceptionMessage = (jstring)(*env)->CallObjectMethod(env, exception, mid);
1805 if (exceptionMessage != NULL) {
1806 const char *exceptionMessageUTF8 = (*env)->GetStringUTFChars(env, exceptionMessage, 0);
1807 SDL_SetError("%s: %s", exceptionNameUTF8, exceptionMessageUTF8);
1808 (*env)->ReleaseStringUTFChars(env, exceptionMessage, exceptionMessageUTF8);
1810 SDL_SetError("%s", exceptionNameUTF8);
1813 (*env)->ReleaseStringUTFChars(env, exceptionName, exceptionNameUTF8);
1822 static void Internal_Android_Create_AssetManager() {
1824 struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(__FUNCTION__);
1825 JNIEnv *env = Android_JNI_GetEnv();
1828 jobject javaAssetManager;
1830 if (!LocalReferenceHolder_Init(&refs, env)) {
1831 LocalReferenceHolder_Cleanup(&refs);
1835 /* context = SDLActivity.getContext(); */
1836 context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
1838 /* javaAssetManager = context.getAssets(); */
1839 mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context),
1840 "getAssets", "()Landroid/content/res/AssetManager;");
1841 javaAssetManager = (*env)->CallObjectMethod(env, context, mid);
1844 * Given a Dalvik AssetManager object, obtain the corresponding native AAssetManager
1845 * object. Note that the caller is responsible for obtaining and holding a VM reference
1846 * to the jobject to prevent its being garbage collected while the native object is
1849 javaAssetManagerRef = (*env)->NewGlobalRef(env, javaAssetManager);
1850 asset_manager = AAssetManager_fromJava(env, javaAssetManagerRef);
1852 if (asset_manager == NULL) {
1853 (*env)->DeleteGlobalRef(env, javaAssetManagerRef);
1854 Android_JNI_ExceptionOccurred(SDL_TRUE);
1857 LocalReferenceHolder_Cleanup(&refs);
1860 static void Internal_Android_Destroy_AssetManager() {
1861 JNIEnv *env = Android_JNI_GetEnv();
1863 if (asset_manager) {
1864 (*env)->DeleteGlobalRef(env, javaAssetManagerRef);
1865 asset_manager = NULL;
1869 int Android_JNI_FileOpen(SDL_RWops *ctx,
1870 const char *fileName, const char *mode)
1872 AAsset *asset = NULL;
1873 ctx->hidden.androidio.asset = NULL;
1875 if (asset_manager == NULL) {
1876 Internal_Android_Create_AssetManager();
1879 if (asset_manager == NULL) {
1883 asset = AAssetManager_open(asset_manager, fileName, AASSET_MODE_UNKNOWN);
1884 if (asset == NULL) {
1889 ctx->hidden.androidio.asset = (void*) asset;
1893 size_t Android_JNI_FileRead(SDL_RWops* ctx, void* buffer,
1894 size_t size, size_t maxnum)
1897 AAsset *asset = (AAsset*) ctx->hidden.androidio.asset;
1898 result = AAsset_read(asset, buffer, size * maxnum);
1901 /* Number of chuncks */
1902 return (result / size);
1909 size_t Android_JNI_FileWrite(SDL_RWops *ctx, const void *buffer,
1910 size_t size, size_t num)
1912 SDL_SetError("Cannot write to Android package filesystem");
1916 Sint64 Android_JNI_FileSize(SDL_RWops *ctx)
1919 AAsset *asset = (AAsset*) ctx->hidden.androidio.asset;
1920 result = AAsset_getLength64(asset);
1924 Sint64 Android_JNI_FileSeek(SDL_RWops* ctx, Sint64 offset, int whence)
1927 AAsset *asset = (AAsset*) ctx->hidden.androidio.asset;
1928 result = AAsset_seek64(asset, offset, whence);
1932 int Android_JNI_FileClose(SDL_RWops *ctx)
1934 AAsset *asset = (AAsset*) ctx->hidden.androidio.asset;
1935 AAsset_close(asset);
1939 int Android_JNI_SetClipboardText(const char *text)
1941 JNIEnv *env = Android_JNI_GetEnv();
1942 jstring string = (*env)->NewStringUTF(env, text);
1943 (*env)->CallStaticVoidMethod(env, mActivityClass, midClipboardSetText, string);
1944 (*env)->DeleteLocalRef(env, string);
1948 char* Android_JNI_GetClipboardText(void)
1950 JNIEnv *env = Android_JNI_GetEnv();
1954 string = (*env)->CallStaticObjectMethod(env, mActivityClass, midClipboardGetText);
1956 const char *utf = (*env)->GetStringUTFChars(env, string, 0);
1958 text = SDL_strdup(utf);
1959 (*env)->ReleaseStringUTFChars(env, string, utf);
1961 (*env)->DeleteLocalRef(env, string);
1964 return (text == NULL) ? SDL_strdup("") : text;
1967 SDL_bool Android_JNI_HasClipboardText(void)
1969 JNIEnv *env = Android_JNI_GetEnv();
1970 jboolean retval = (*env)->CallStaticBooleanMethod(env, mActivityClass, midClipboardHasText);
1971 return (retval == JNI_TRUE) ? SDL_TRUE : SDL_FALSE;
1974 /* returns 0 on success or -1 on error (others undefined then)
1975 * returns truthy or falsy value in plugged, charged and battery
1976 * returns the value in seconds and percent or -1 if not available
1978 int Android_JNI_GetPowerInfo(int *plugged, int *charged, int *battery, int *seconds, int *percent)
1980 struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(__FUNCTION__);
1981 JNIEnv *env = Android_JNI_GetEnv();
1992 if (!LocalReferenceHolder_Init(&refs, env)) {
1993 LocalReferenceHolder_Cleanup(&refs);
1998 /* context = SDLActivity.getContext(); */
1999 context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
2001 action = (*env)->NewStringUTF(env, "android.intent.action.BATTERY_CHANGED");
2003 cls = (*env)->FindClass(env, "android/content/IntentFilter");
2005 mid = (*env)->GetMethodID(env, cls, "<init>", "(Ljava/lang/String;)V");
2006 filter = (*env)->NewObject(env, cls, mid, action);
2008 (*env)->DeleteLocalRef(env, action);
2010 mid = (*env)->GetMethodID(env, mActivityClass, "registerReceiver", "(Landroid/content/BroadcastReceiver;Landroid/content/IntentFilter;)Landroid/content/Intent;");
2011 intent = (*env)->CallObjectMethod(env, context, mid, NULL, filter);
2013 (*env)->DeleteLocalRef(env, filter);
2015 cls = (*env)->GetObjectClass(env, intent);
2017 imid = (*env)->GetMethodID(env, cls, "getIntExtra", "(Ljava/lang/String;I)I");
2019 /* Watch out for C89 scoping rules because of the macro */
2020 #define GET_INT_EXTRA(var, key) \
2022 iname = (*env)->NewStringUTF(env, key); \
2023 (var) = (*env)->CallIntMethod(env, intent, imid, iname, -1); \
2024 (*env)->DeleteLocalRef(env, iname);
2026 bmid = (*env)->GetMethodID(env, cls, "getBooleanExtra", "(Ljava/lang/String;Z)Z");
2028 /* Watch out for C89 scoping rules because of the macro */
2029 #define GET_BOOL_EXTRA(var, key) \
2031 bname = (*env)->NewStringUTF(env, key); \
2032 (var) = (*env)->CallBooleanMethod(env, intent, bmid, bname, JNI_FALSE); \
2033 (*env)->DeleteLocalRef(env, bname);
2036 /* Watch out for C89 scoping rules because of the macro */
2037 GET_INT_EXTRA(plug, "plugged") /* == BatteryManager.EXTRA_PLUGGED (API 5) */
2039 LocalReferenceHolder_Cleanup(&refs);
2042 /* 1 == BatteryManager.BATTERY_PLUGGED_AC */
2043 /* 2 == BatteryManager.BATTERY_PLUGGED_USB */
2044 *plugged = (0 < plug) ? 1 : 0;
2048 /* Watch out for C89 scoping rules because of the macro */
2049 GET_INT_EXTRA(status, "status") /* == BatteryManager.EXTRA_STATUS (API 5) */
2051 LocalReferenceHolder_Cleanup(&refs);
2054 /* 5 == BatteryManager.BATTERY_STATUS_FULL */
2055 *charged = (status == 5) ? 1 : 0;
2059 GET_BOOL_EXTRA(present, "present") /* == BatteryManager.EXTRA_PRESENT (API 5) */
2060 *battery = present ? 1 : 0;
2064 *seconds = -1; /* not possible */
2071 /* Watch out for C89 scoping rules because of the macro */
2073 GET_INT_EXTRA(level_temp, "level") /* == BatteryManager.EXTRA_LEVEL (API 5) */
2076 /* Watch out for C89 scoping rules because of the macro */
2078 GET_INT_EXTRA(scale_temp, "scale") /* == BatteryManager.EXTRA_SCALE (API 5) */
2082 if ((level == -1) || (scale == -1)) {
2083 LocalReferenceHolder_Cleanup(&refs);
2086 *percent = level * 100 / scale;
2089 (*env)->DeleteLocalRef(env, intent);
2091 LocalReferenceHolder_Cleanup(&refs);
2095 /* Add all touch devices */
2096 void Android_JNI_InitTouch() {
2097 JNIEnv *env = Android_JNI_GetEnv();
2098 (*env)->CallStaticVoidMethod(env, mActivityClass, midInitTouch);
2101 void Android_JNI_PollInputDevices(void)
2103 JNIEnv *env = Android_JNI_GetEnv();
2104 (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midPollInputDevices);
2107 void Android_JNI_PollHapticDevices(void)
2109 JNIEnv *env = Android_JNI_GetEnv();
2110 (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midPollHapticDevices);
2113 void Android_JNI_HapticRun(int device_id, float intensity, int length)
2115 JNIEnv *env = Android_JNI_GetEnv();
2116 (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midHapticRun, device_id, intensity, length);
2119 void Android_JNI_HapticStop(int device_id)
2121 JNIEnv *env = Android_JNI_GetEnv();
2122 (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midHapticStop, device_id);
2125 /* See SDLActivity.java for constants. */
2126 #define COMMAND_SET_KEEP_SCREEN_ON 5
2128 /* sends message to be handled on the UI event dispatch thread */
2129 int Android_JNI_SendMessage(int command, int param)
2131 JNIEnv *env = Android_JNI_GetEnv();
2133 success = (*env)->CallStaticBooleanMethod(env, mActivityClass, midSendMessage, command, param);
2134 return success ? 0 : -1;
2137 void Android_JNI_SuspendScreenSaver(SDL_bool suspend)
2139 Android_JNI_SendMessage(COMMAND_SET_KEEP_SCREEN_ON, (suspend == SDL_FALSE) ? 0 : 1);
2142 void Android_JNI_ShowTextInput(SDL_Rect *inputRect)
2144 JNIEnv *env = Android_JNI_GetEnv();
2145 (*env)->CallStaticBooleanMethod(env, mActivityClass, midShowTextInput,
2152 void Android_JNI_HideTextInput(void)
2154 /* has to match Activity constant */
2155 const int COMMAND_TEXTEDIT_HIDE = 3;
2156 Android_JNI_SendMessage(COMMAND_TEXTEDIT_HIDE, 0);
2159 SDL_bool Android_JNI_IsScreenKeyboardShown(void)
2161 JNIEnv *env = Android_JNI_GetEnv();
2162 jboolean is_shown = 0;
2163 is_shown = (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsScreenKeyboardShown);
2168 int Android_JNI_ShowMessageBox(const SDL_MessageBoxData *messageboxdata, int *buttonid)
2176 jintArray button_flags;
2177 jintArray button_ids;
2178 jobjectArray button_texts;
2184 env = Android_JNI_GetEnv();
2186 /* convert parameters */
2188 clazz = (*env)->FindClass(env, "java/lang/String");
2190 title = (*env)->NewStringUTF(env, messageboxdata->title);
2191 message = (*env)->NewStringUTF(env, messageboxdata->message);
2193 button_flags = (*env)->NewIntArray(env, messageboxdata->numbuttons);
2194 button_ids = (*env)->NewIntArray(env, messageboxdata->numbuttons);
2195 button_texts = (*env)->NewObjectArray(env, messageboxdata->numbuttons,
2197 for (i = 0; i < messageboxdata->numbuttons; ++i) {
2198 const SDL_MessageBoxButtonData *sdlButton;
2200 if (messageboxdata->flags & SDL_MESSAGEBOX_BUTTONS_RIGHT_TO_LEFT) {
2201 sdlButton = &messageboxdata->buttons[messageboxdata->numbuttons - 1 - i];
2203 sdlButton = &messageboxdata->buttons[i];
2206 temp = sdlButton->flags;
2207 (*env)->SetIntArrayRegion(env, button_flags, i, 1, &temp);
2208 temp = sdlButton->buttonid;
2209 (*env)->SetIntArrayRegion(env, button_ids, i, 1, &temp);
2210 text = (*env)->NewStringUTF(env, sdlButton->text);
2211 (*env)->SetObjectArrayElement(env, button_texts, i, text);
2212 (*env)->DeleteLocalRef(env, text);
2215 if (messageboxdata->colorScheme) {
2216 colors = (*env)->NewIntArray(env, SDL_MESSAGEBOX_COLOR_MAX);
2217 for (i = 0; i < SDL_MESSAGEBOX_COLOR_MAX; ++i) {
2218 temp = (0xFFU << 24) |
2219 (messageboxdata->colorScheme->colors[i].r << 16) |
2220 (messageboxdata->colorScheme->colors[i].g << 8) |
2221 (messageboxdata->colorScheme->colors[i].b << 0);
2222 (*env)->SetIntArrayRegion(env, colors, i, 1, &temp);
2228 (*env)->DeleteLocalRef(env, clazz);
2230 /* context = SDLActivity.getContext(); */
2231 context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
2233 clazz = (*env)->GetObjectClass(env, context);
2235 mid = (*env)->GetMethodID(env, clazz,
2236 "messageboxShowMessageBox", "(ILjava/lang/String;Ljava/lang/String;[I[I[Ljava/lang/String;[I)I");
2237 *buttonid = (*env)->CallIntMethod(env, context, mid,
2238 messageboxdata->flags,
2246 (*env)->DeleteLocalRef(env, context);
2247 (*env)->DeleteLocalRef(env, clazz);
2249 /* delete parameters */
2251 (*env)->DeleteLocalRef(env, title);
2252 (*env)->DeleteLocalRef(env, message);
2253 (*env)->DeleteLocalRef(env, button_flags);
2254 (*env)->DeleteLocalRef(env, button_ids);
2255 (*env)->DeleteLocalRef(env, button_texts);
2256 (*env)->DeleteLocalRef(env, colors);
2262 //////////////////////////////////////////////////////////////////////////////
2264 // Functions exposed to SDL applications in SDL_system.h
2265 //////////////////////////////////////////////////////////////////////////////
2268 void *SDL_AndroidGetJNIEnv(void)
2270 return Android_JNI_GetEnv();
2273 void *SDL_AndroidGetActivity(void)
2275 /* See SDL_system.h for caveats on using this function. */
2277 JNIEnv *env = Android_JNI_GetEnv();
2282 /* return SDLActivity.getContext(); */
2283 return (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
2286 int SDL_GetAndroidSDKVersion(void)
2288 static int sdk_version;
2290 char sdk[PROP_VALUE_MAX] = {0};
2291 if (__system_property_get("ro.build.version.sdk", sdk) != 0) {
2292 sdk_version = SDL_atoi(sdk);
2298 SDL_bool SDL_IsAndroidTablet(void)
2300 JNIEnv *env = Android_JNI_GetEnv();
2301 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsTablet);
2304 SDL_bool SDL_IsAndroidTV(void)
2306 JNIEnv *env = Android_JNI_GetEnv();
2307 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsAndroidTV);
2310 SDL_bool SDL_IsChromebook(void)
2312 JNIEnv *env = Android_JNI_GetEnv();
2313 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsChromebook);
2316 SDL_bool SDL_IsDeXMode(void)
2318 JNIEnv *env = Android_JNI_GetEnv();
2319 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsDeXMode);
2322 void SDL_AndroidBackButton(void)
2324 JNIEnv *env = Android_JNI_GetEnv();
2325 (*env)->CallStaticVoidMethod(env, mActivityClass, midManualBackButton);
2328 const char * SDL_AndroidGetInternalStoragePath(void)
2330 static char *s_AndroidInternalFilesPath = NULL;
2332 if (!s_AndroidInternalFilesPath) {
2333 struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(__FUNCTION__);
2340 JNIEnv *env = Android_JNI_GetEnv();
2341 if (!LocalReferenceHolder_Init(&refs, env)) {
2342 LocalReferenceHolder_Cleanup(&refs);
2346 /* context = SDLActivity.getContext(); */
2347 context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
2349 SDL_SetError("Couldn't get Android context!");
2350 LocalReferenceHolder_Cleanup(&refs);
2354 /* fileObj = context.getFilesDir(); */
2355 mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context),
2356 "getFilesDir", "()Ljava/io/File;");
2357 fileObject = (*env)->CallObjectMethod(env, context, mid);
2359 SDL_SetError("Couldn't get internal directory");
2360 LocalReferenceHolder_Cleanup(&refs);
2364 /* path = fileObject.getCanonicalPath(); */
2365 mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, fileObject),
2366 "getCanonicalPath", "()Ljava/lang/String;");
2367 pathString = (jstring)(*env)->CallObjectMethod(env, fileObject, mid);
2368 if (Android_JNI_ExceptionOccurred(SDL_FALSE)) {
2369 LocalReferenceHolder_Cleanup(&refs);
2373 path = (*env)->GetStringUTFChars(env, pathString, NULL);
2374 s_AndroidInternalFilesPath = SDL_strdup(path);
2375 (*env)->ReleaseStringUTFChars(env, pathString, path);
2377 LocalReferenceHolder_Cleanup(&refs);
2379 return s_AndroidInternalFilesPath;
2382 int SDL_AndroidGetExternalStorageState(void)
2384 struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(__FUNCTION__);
2387 jstring stateString;
2391 JNIEnv *env = Android_JNI_GetEnv();
2392 if (!LocalReferenceHolder_Init(&refs, env)) {
2393 LocalReferenceHolder_Cleanup(&refs);
2397 cls = (*env)->FindClass(env, "android/os/Environment");
2398 mid = (*env)->GetStaticMethodID(env, cls,
2399 "getExternalStorageState", "()Ljava/lang/String;");
2400 stateString = (jstring)(*env)->CallStaticObjectMethod(env, cls, mid);
2402 state = (*env)->GetStringUTFChars(env, stateString, NULL);
2404 /* Print an info message so people debugging know the storage state */
2405 __android_log_print(ANDROID_LOG_INFO, "SDL", "external storage state: %s", state);
2407 if (SDL_strcmp(state, "mounted") == 0) {
2408 stateFlags = SDL_ANDROID_EXTERNAL_STORAGE_READ |
2409 SDL_ANDROID_EXTERNAL_STORAGE_WRITE;
2410 } else if (SDL_strcmp(state, "mounted_ro") == 0) {
2411 stateFlags = SDL_ANDROID_EXTERNAL_STORAGE_READ;
2415 (*env)->ReleaseStringUTFChars(env, stateString, state);
2417 LocalReferenceHolder_Cleanup(&refs);
2421 const char * SDL_AndroidGetExternalStoragePath(void)
2423 static char *s_AndroidExternalFilesPath = NULL;
2425 if (!s_AndroidExternalFilesPath) {
2426 struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(__FUNCTION__);
2433 JNIEnv *env = Android_JNI_GetEnv();
2434 if (!LocalReferenceHolder_Init(&refs, env)) {
2435 LocalReferenceHolder_Cleanup(&refs);
2439 /* context = SDLActivity.getContext(); */
2440 context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
2442 /* fileObj = context.getExternalFilesDir(); */
2443 mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context),
2444 "getExternalFilesDir", "(Ljava/lang/String;)Ljava/io/File;");
2445 fileObject = (*env)->CallObjectMethod(env, context, mid, NULL);
2447 SDL_SetError("Couldn't get external directory");
2448 LocalReferenceHolder_Cleanup(&refs);
2452 /* path = fileObject.getAbsolutePath(); */
2453 mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, fileObject),
2454 "getAbsolutePath", "()Ljava/lang/String;");
2455 pathString = (jstring)(*env)->CallObjectMethod(env, fileObject, mid);
2457 path = (*env)->GetStringUTFChars(env, pathString, NULL);
2458 s_AndroidExternalFilesPath = SDL_strdup(path);
2459 (*env)->ReleaseStringUTFChars(env, pathString, path);
2461 LocalReferenceHolder_Cleanup(&refs);
2463 return s_AndroidExternalFilesPath;
2466 SDL_bool SDL_AndroidRequestPermission(const char *permission)
2468 return Android_JNI_RequestPermission(permission);
2471 void Android_JNI_GetManifestEnvironmentVariables(void)
2473 if (!mActivityClass || !midGetManifestEnvironmentVariables) {
2474 __android_log_print(ANDROID_LOG_WARN, "SDL", "Request to get environment variables before JNI is ready");
2478 if (!bHasEnvironmentVariables) {
2479 JNIEnv *env = Android_JNI_GetEnv();
2480 SDL_bool ret = (*env)->CallStaticBooleanMethod(env, mActivityClass, midGetManifestEnvironmentVariables);
2482 bHasEnvironmentVariables = SDL_TRUE;
2487 int Android_JNI_CreateCustomCursor(SDL_Surface *surface, int hot_x, int hot_y)
2489 JNIEnv *env = Android_JNI_GetEnv();
2490 int custom_cursor = 0;
2492 pixels = (*env)->NewIntArray(env, surface->w * surface->h);
2494 (*env)->SetIntArrayRegion(env, pixels, 0, surface->w * surface->h, (int *)surface->pixels);
2495 custom_cursor = (*env)->CallStaticIntMethod(env, mActivityClass, midCreateCustomCursor, pixels, surface->w, surface->h, hot_x, hot_y);
2496 (*env)->DeleteLocalRef(env, pixels);
2500 return custom_cursor;
2504 SDL_bool Android_JNI_SetCustomCursor(int cursorID)
2506 JNIEnv *env = Android_JNI_GetEnv();
2507 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetCustomCursor, cursorID);
2510 SDL_bool Android_JNI_SetSystemCursor(int cursorID)
2512 JNIEnv *env = Android_JNI_GetEnv();
2513 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetSystemCursor, cursorID);
2516 SDL_bool Android_JNI_SupportsRelativeMouse(void)
2518 JNIEnv *env = Android_JNI_GetEnv();
2519 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSupportsRelativeMouse);
2522 SDL_bool Android_JNI_SetRelativeMouseEnabled(SDL_bool enabled)
2524 JNIEnv *env = Android_JNI_GetEnv();
2525 return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetRelativeMouseEnabled, (enabled == 1));
2528 SDL_bool Android_JNI_RequestPermission(const char *permission)
2530 JNIEnv *env = Android_JNI_GetEnv();
2531 const int requestCode = 1;
2533 /* Wait for any pending request on another thread */
2534 while (SDL_AtomicGet(&bPermissionRequestPending) == SDL_TRUE) {
2537 SDL_AtomicSet(&bPermissionRequestPending, SDL_TRUE);
2539 jstring jpermission = (*env)->NewStringUTF(env, permission);
2540 (*env)->CallStaticVoidMethod(env, mActivityClass, midRequestPermission, jpermission, requestCode);
2541 (*env)->DeleteLocalRef(env, jpermission);
2543 /* Wait for the request to complete */
2544 while (SDL_AtomicGet(&bPermissionRequestPending) == SDL_TRUE) {
2547 return bPermissionRequestResult;
2550 int Android_JNI_GetLocale(char *buf, size_t buflen)
2552 AConfiguration *cfg;
2554 SDL_assert(buflen > 6);
2556 /* Need to re-create the asset manager if locale has changed (SDL_LOCALECHANGED) */
2557 Internal_Android_Destroy_AssetManager();
2559 if (asset_manager == NULL) {
2560 Internal_Android_Create_AssetManager();
2563 if (asset_manager == NULL) {
2567 cfg = AConfiguration_new();
2573 char language[2] = {};
2574 char country[2] = {};
2577 AConfiguration_fromAssetManager(cfg, asset_manager);
2578 AConfiguration_getLanguage(cfg, language);
2579 AConfiguration_getCountry(cfg, country);
2581 /* copy language (not null terminated) */
2583 buf[id++] = language[0];
2585 buf[id++] = language[1];
2591 /* copy country (not null terminated) */
2593 buf[id++] = country[0];
2595 buf[id++] = country[1];
2600 SDL_assert(id <= buflen);
2603 AConfiguration_delete(cfg);
2609 Android_JNI_OpenURL(const char *url)
2611 JNIEnv *env = Android_JNI_GetEnv();
2612 jstring jurl = (*env)->NewStringUTF(env, url);
2613 const int ret = (*env)->CallStaticIntMethod(env, mActivityClass, midOpenURL, jurl);
2614 (*env)->DeleteLocalRef(env, jurl);
2618 #endif /* __ANDROID__ */
2620 /* vi: set ts=4 sw=4 expandtab: */