16#include <freerdp/config.h>
25#include <winpr/assert.h>
26#include <winpr/image.h>
28#include <freerdp/graphics.h>
29#include <freerdp/codec/rfx.h>
30#include <freerdp/gdi/gdi.h>
31#include <freerdp/gdi/gfx.h>
32#include <freerdp/client/rdpei.h>
33#include <freerdp/client/rdpgfx.h>
34#include <freerdp/client/cliprdr.h>
35#include <freerdp/codec/h264.h>
36#include <freerdp/codec/video.h>
37#include <freerdp/channels/channels.h>
38#include <freerdp/client/channels.h>
39#include <freerdp/client/cmdline.h>
40#include <freerdp/constants.h>
41#include <freerdp/locale/keyboard.h>
42#include <freerdp/primitives.h>
43#include <freerdp/version.h>
44#include <freerdp/settings.h>
45#include <freerdp/utils/signal.h>
47#include <android/bitmap.h>
49#include "android_jni_callback.h"
50#include "android_jni_utils.h"
51#include "android_cliprdr.h"
52#include "android_disp.h"
53#include "android_rail.h"
54#include "android_freerdp_jni.h"
56#if defined(WITH_GPROF)
60#define TAG CLIENT_TAG("android")
63#define FREERDP_JNI_VERSION FREERDP_VERSION_FULL
65static jclass gJavaActivityClass;
66static jmethodID gOnPointerSetMethod;
67static jmethodID gOnRailWindowUpdateMethod;
69static UINT android_UpdateWindowFromSurface(RdpgfxClientContext* context, gdiGfxSurface* surface);
71static void android_OnChannelConnectedEventHandler(
void* context,
72 const ChannelConnectedEventArgs* e)
74 rdpSettings* settings;
79 WLog_FATAL(TAG,
"(context=%p, EventArgs=%p", context, (
void*)e);
84 settings = afc->common.context.settings;
86 if (strcmp(e->name, CLIPRDR_SVC_CHANNEL_NAME) == 0)
88 android_cliprdr_init(afc, (CliprdrClientContext*)e->pInterface);
90 else if (strcmp(e->name, DISP_DVC_CHANNEL_NAME) == 0)
92 android_disp_init(afc, (DispClientContext*)e->pInterface);
94 else if (strcmp(e->name, RAIL_SVC_CHANNEL_NAME) == 0)
96 android_rail_init(afc, (RailClientContext*)e->pInterface);
98 else if (strcmp(e->name, RDPGFX_DVC_CHANNEL_NAME) == 0)
100 freerdp_client_OnChannelConnectedEventHandler(context, e);
101 RdpgfxClientContext* gfx = (RdpgfxClientContext*)e->pInterface;
103 gfx->UpdateWindowFromSurface = android_UpdateWindowFromSurface;
106 freerdp_client_OnChannelConnectedEventHandler(context, e);
109static void android_OnChannelDisconnectedEventHandler(
void* context,
110 const ChannelDisconnectedEventArgs* e)
112 rdpSettings* settings;
117 WLog_FATAL(TAG,
"(context=%p, EventArgs=%p", context, (
void*)e);
122 settings = afc->common.context.settings;
124 if (strcmp(e->name, CLIPRDR_SVC_CHANNEL_NAME) == 0)
126 android_cliprdr_uninit(afc, (CliprdrClientContext*)e->pInterface);
128 else if (strcmp(e->name, DISP_DVC_CHANNEL_NAME) == 0)
130 android_disp_uninit(afc, (DispClientContext*)e->pInterface);
132 else if (strcmp(e->name, RAIL_SVC_CHANNEL_NAME) == 0)
134 android_rail_uninit(afc, (RailClientContext*)e->pInterface);
137 freerdp_client_OnChannelDisconnectedEventHandler(context, e);
140static BOOL android_begin_paint(rdpContext* context)
145static BOOL android_end_paint(rdpContext* context)
153 rdpSettings* settings;
155 if (!ctx || !context->instance)
158 settings = context->settings;
165 if (!gdi || !gdi->primary || !gdi->primary->hdc)
168 hwnd = ctx->common.context.gdi->primary->hdc->hwnd;
173 ninvalid = hwnd->ninvalid;
178 cinvalid = hwnd->cinvalid;
185 x2 = cinvalid[0].x + cinvalid[0].w;
186 y2 = cinvalid[0].y + cinvalid[0].h;
188 for (
int i = 0; i < ninvalid; i++)
190 x1 = MIN(x1, cinvalid[i].x);
191 y1 = MIN(y1, cinvalid[i].y);
192 x2 = MAX(x2, cinvalid[i].x + cinvalid[i].w);
193 y2 = MAX(y2, cinvalid[i].y + cinvalid[i].h);
196 freerdp_callback(
"OnGraphicsUpdate",
"(JIIII)V", (jlong)context->instance, x1, y1, x2 - x1,
199 hwnd->invalid->null = TRUE;
204static BOOL android_desktop_resize(rdpContext* context)
206 WINPR_ASSERT(context);
207 WINPR_ASSERT(context->settings);
208 WINPR_ASSERT(context->instance);
213 if (context->gdi && !gdi_resize(context->gdi, width, height))
216 freerdp_callback(
"OnGraphicsResize",
"(JIII)V", (jlong)context->instance, width, height,
221static BOOL android_pre_connect(freerdp* instance)
223 WINPR_ASSERT(instance);
224 WINPR_ASSERT(instance->context);
226 rdpSettings* settings = instance->context->settings;
231 int rc = PubSub_SubscribeChannelConnected(instance->context->pubSub,
232 android_OnChannelConnectedEventHandler);
234 if (rc != CHANNEL_RC_OK)
236 WLog_ERR(TAG,
"Could not subscribe to connect event handler [%08X]", rc);
240 rc = PubSub_SubscribeChannelDisconnected(instance->context->pubSub,
241 android_OnChannelDisconnectedEventHandler);
243 if (rc != CHANNEL_RC_OK)
245 WLog_ERR(TAG,
"Could not subscribe to disconnect event handler [%08X]", rc);
249 freerdp_callback(
"OnPreConnect",
"(J)V", (jlong)instance);
260static BOOL android_Pointer_New(rdpContext* context, rdpPointer* pointer)
262 WINPR_ASSERT(context);
263 WINPR_ASSERT(pointer);
264 WINPR_ASSERT(context->gdi);
266 androidPointer* ptr = (androidPointer*)pointer;
270 ptr->size = 4ULL * pointer->width * pointer->height;
271 ptr->data = winpr_aligned_malloc(ptr->size, 16);
275 if (!freerdp_image_copy_from_pointer_data(
276 ptr->data, PIXEL_FORMAT_BGRA32, 0, 0, 0, pointer->width, pointer->height,
277 pointer->xorMaskData, pointer->lengthXorMask, pointer->andMaskData,
278 pointer->lengthAndMask, pointer->xorBpp, &context->gdi->palette))
280 winpr_aligned_free(ptr->data);
288static void android_Pointer_Free(rdpContext* context, rdpPointer* pointer)
290 WINPR_UNUSED(context);
291 androidPointer* ptr = (androidPointer*)pointer;
295 winpr_aligned_free(ptr->data);
300static BOOL android_Pointer_Set(rdpContext* context, rdpPointer* pointer)
302 WINPR_ASSERT(context);
303 WINPR_ASSERT(pointer);
305 androidPointer* ptr = (androidPointer*)pointer;
309 const jsize nPixels = (jsize)(pointer->width * pointer->height);
312 jboolean attached = jni_attach_thread(&env);
314 if (!gJavaActivityClass || !gOnPointerSetMethod)
317 jintArray pixels = (*env)->NewIntArray(env, nPixels);
321 (*env)->SetIntArrayRegion(env, pixels, 0, nPixels, (
const jint*)ptr->data);
322 (*env)->CallStaticVoidMethod(env, gJavaActivityClass, gOnPointerSetMethod,
323 (jlong)context->instance, pixels, (jint)pointer->width,
324 (jint)pointer->height, (jint)pointer->xPos, (jint)pointer->yPos);
325 (*env)->DeleteLocalRef(env, pixels);
333static BOOL android_Pointer_SetPosition(rdpContext* context, UINT32 x, UINT32 y)
335 WINPR_ASSERT(context);
340static BOOL android_Pointer_SetNull(rdpContext* context)
342 WINPR_ASSERT(context);
344 freerdp_callback(
"OnPointerSetNull",
"(J)V", (jlong)context->instance);
348static BOOL android_Pointer_SetDefault(rdpContext* context)
350 WINPR_ASSERT(context);
352 freerdp_callback(
"OnPointerSetDefault",
"(J)V", (jlong)context->instance);
356static UINT android_UpdateWindowFromSurface(RdpgfxClientContext* context, gdiGfxSurface* surface)
358 if (!context || !surface)
359 return CHANNEL_RC_OK;
361 rdpGdi* gdi = (rdpGdi*)context->custom;
362 if (!gdi || !gdi->context)
363 return CHANNEL_RC_OK;
365 const UINT32 width = surface->mappedWidth ? surface->mappedWidth : surface->width;
366 const UINT32 height = surface->mappedHeight ? surface->mappedHeight : surface->height;
367 if (width == 0 || height == 0)
368 return CHANNEL_RC_OK;
371 jboolean attached = jni_attach_thread(&env);
372 if (!gJavaActivityClass || !gOnRailWindowUpdateMethod)
375 const jsize nPixels = (jsize)(width * height);
376 jintArray pixels = (*env)->NewIntArray(env, nPixels);
380 jint* dst = (*env)->GetIntArrayElements(env, pixels, NULL);
383 freerdp_image_copy((BYTE*)dst, surface->format, width * 4, 0, 0, width, height,
384 surface->data, surface->format, surface->scanline, 0, 0, NULL,
389 const size_t total = (size_t)width * height;
390 for (
size_t i = 0; i < total; i++)
392 const UINT32 px = (UINT32)dst[i];
393 if ((px & 0x00FFFFFFu) != 0)
394 dst[i] = (jint)(px | 0xFF000000u);
396 (*env)->ReleaseIntArrayElements(env, pixels, dst, 0);
399 freerdp* inst = gdi->context->instance;
400 (*env)->CallStaticVoidMethod(env, gJavaActivityClass, gOnRailWindowUpdateMethod, (jlong)inst,
401 (jlong)surface->windowId, (jint)width, (jint)height, pixels);
402 (*env)->DeleteLocalRef(env, pixels);
406 return CHANNEL_RC_OK;
409static BOOL android_register_pointer(rdpGraphics* graphics)
411 rdpPointer pointer = WINPR_C_ARRAY_INIT;
416 pointer.size =
sizeof(androidPointer);
417 pointer.New = android_Pointer_New;
418 pointer.Free = android_Pointer_Free;
419 pointer.Set = android_Pointer_Set;
420 pointer.SetNull = android_Pointer_SetNull;
421 pointer.SetDefault = android_Pointer_SetDefault;
422 pointer.SetPosition = android_Pointer_SetPosition;
423 graphics_register_pointer(graphics, &pointer);
428#define ANDROID_EXPERIMENTAL_REMOTEAPP 0
429#define ANDROID_EXPERIMENTAL_CAMERA 1
431static BOOL android_post_connect(freerdp* instance)
433 rdpSettings* settings;
436 WINPR_ASSERT(instance);
437 WINPR_ASSERT(instance->context);
439 update = instance->context->update;
440 WINPR_ASSERT(update);
442 settings = instance->context->settings;
443 WINPR_ASSERT(settings);
446 !freerdp_callback_bool_result(
"OnExperimentalFeature",
"(JI)Z", (jlong)instance,
447 ANDROID_EXPERIMENTAL_REMOTEAPP))
450 if (freerdp_dynamic_channel_collection_find(settings,
"rdpecam") &&
451 !freerdp_callback_bool_result(
"OnExperimentalFeature",
"(JI)Z", (jlong)instance,
452 ANDROID_EXPERIMENTAL_CAMERA))
455 if (!gdi_init(instance, PIXEL_FORMAT_RGBX32))
458 if (!android_register_pointer(instance->context->graphics))
461 update->BeginPaint = android_begin_paint;
462 update->EndPaint = android_end_paint;
463 update->DesktopResize = android_desktop_resize;
464 freerdp_callback(
"OnSettingsChanged",
"(JIII)V", (jlong)instance,
468 freerdp_callback(
"OnConnectionSuccess",
"(J)V", (jlong)instance);
472static void android_post_disconnect(freerdp* instance)
474 freerdp_callback(
"OnDisconnecting",
"(J)V", (jlong)instance);
478static BOOL android_authenticate_int(freerdp* instance,
char** username,
char** password,
479 char** domain,
const char* cb_name)
482 jboolean attached = jni_attach_thread(&env);
483 jobject jstr1 = create_string_builder(env, *username);
484 jobject jstr2 = create_string_builder(env, *domain);
485 jobject jstr3 = create_string_builder(env, *password);
487 res = freerdp_callback_bool_result(cb_name,
488 "(JLjava/lang/StringBuilder;"
489 "Ljava/lang/StringBuilder;"
490 "Ljava/lang/StringBuilder;)Z",
491 (jlong)instance, jstr1, jstr2, jstr3);
497 *username = get_string_from_string_builder(env, jstr1);
499 *domain = get_string_from_string_builder(env, jstr2);
501 *password = get_string_from_string_builder(env, jstr3);
504 if (attached == JNI_TRUE)
507 return ((res == JNI_TRUE) ? TRUE : FALSE);
510static BOOL android_authenticate_ex(freerdp* instance,
char** username,
char** password,
511 char** domain, rdp_auth_reason reason)
518 return android_authenticate_int(instance, username, password, domain,
"OnAuthenticate");
522 return android_authenticate_int(instance, username, password, domain,
523 "OnGatewayAuthenticate");
529static DWORD android_verify_certificate_ex(freerdp* instance,
const char* host, UINT16 port,
530 const char* common_name,
const char* subject,
531 const char* issuer,
const char* fingerprint, DWORD flags)
533 WLog_DBG(TAG,
"Certificate details [%s:%" PRIu16
":", host, port);
534 WLog_DBG(TAG,
"\tSubject: %s", subject);
535 WLog_DBG(TAG,
"\tIssuer: %s", issuer);
536 WLog_DBG(TAG,
"\tThumbprint: %s", fingerprint);
538 "The above X.509 certificate could not be verified, possibly because you do not have "
539 "the CA certificate in your certificate store, or the certificate has expired."
540 "Please look at the OpenSSL documentation on how to add a private CA to the store.\n");
542 jboolean attached = jni_attach_thread(&env);
543 jstring jstr0 = (*env)->NewStringUTF(env, host);
544 jstring jstr1 = (*env)->NewStringUTF(env, common_name);
545 jstring jstr2 = (*env)->NewStringUTF(env, subject);
546 jstring jstr3 = (*env)->NewStringUTF(env, issuer);
547 jstring jstr4 = (*env)->NewStringUTF(env, fingerprint);
548 jint res = freerdp_callback_int_result(
"OnVerifyCertificateEx",
549 "(JLjava/lang/String;JLjava/lang/String;Ljava/lang/"
550 "String;Ljava/lang/String;Ljava/lang/String;J)I",
551 (jlong)instance, jstr0, (jlong)port, jstr1, jstr2, jstr3,
552 jstr4, (jlong)flags);
554 if (attached == JNI_TRUE)
560static DWORD android_verify_changed_certificate_ex(freerdp* instance,
const char* host, UINT16 port,
561 const char* common_name,
const char* subject,
562 const char* issuer,
const char* new_fingerprint,
563 const char* old_subject,
const char* old_issuer,
564 const char* old_fingerprint, DWORD flags)
567 jboolean attached = jni_attach_thread(&env);
568 jstring jhost = (*env)->NewStringUTF(env, host);
569 jstring jstr0 = (*env)->NewStringUTF(env, common_name);
570 jstring jstr1 = (*env)->NewStringUTF(env, subject);
571 jstring jstr2 = (*env)->NewStringUTF(env, issuer);
572 jstring jstr3 = (*env)->NewStringUTF(env, new_fingerprint);
573 jstring jstr4 = (*env)->NewStringUTF(env, old_subject);
574 jstring jstr5 = (*env)->NewStringUTF(env, old_issuer);
575 jstring jstr6 = (*env)->NewStringUTF(env, old_fingerprint);
577 freerdp_callback_int_result(
"OnVerifyChangedCertificateEx",
578 "(JLjava/lang/String;JLjava/lang/String;Ljava/lang/"
579 "String;Ljava/lang/String;Ljava/lang/String;"
580 "Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;J)I",
581 (jlong)instance, jhost, (jlong)port, jstr0, jstr1, jstr2, jstr3,
582 jstr4, jstr5, jstr6, (jlong)flags);
584 if (attached == JNI_TRUE)
590static int android_freerdp_run(freerdp* instance)
593 DWORD status = WAIT_FAILED;
594 HANDLE handles[MAXIMUM_WAIT_OBJECTS];
595 HANDLE inputEvent =
nullptr;
596 const rdpSettings* settings = instance->context->settings;
597 rdpContext* context = instance->context;
599 inputEvent = android_get_handle(instance);
601 while (!freerdp_shall_disconnect_context(instance->context))
606 handles[count++] = inputEvent;
608 tmp = freerdp_get_event_handles(context, &handles[count], 64 - count);
612 WLog_ERR(TAG,
"freerdp_get_event_handles failed");
617 status = WaitForMultipleObjects(count, handles, FALSE, INFINITE);
619 if (status == WAIT_FAILED)
621 WLog_ERR(TAG,
"WaitForMultipleObjects failed with %u [%08X]", status,
622 (
unsigned)GetLastError());
626 if (!freerdp_check_event_handles(context))
632 WLog_ERR(TAG,
"Failed to check FreeRDP file descriptor");
633 status = GetLastError();
637 if (freerdp_shall_disconnect_context(instance->context))
640 if (android_check_handle(instance) != TRUE)
642 WLog_ERR(TAG,
"Failed to check android file descriptor");
643 status = GetLastError();
649 WLog_INFO(TAG,
"Prepare shutdown...");
654static DWORD WINAPI android_thread_func(LPVOID param)
656 DWORD status = ERROR_BAD_ARGUMENTS;
657 freerdp* instance = param;
658 WLog_DBG(TAG,
"Start...");
660 WINPR_ASSERT(instance);
661 WINPR_ASSERT(instance->context);
663 if (freerdp_client_start(instance->context) != CHANNEL_RC_OK)
666 WLog_DBG(TAG,
"Connect...");
668 if (!freerdp_connect(instance))
669 status = GetLastError();
672 status = android_freerdp_run(instance);
673 WLog_DBG(TAG,
"Disconnect...");
675 if (!freerdp_disconnect(instance))
676 status = GetLastError();
679 WLog_DBG(TAG,
"Stop...");
681 if (freerdp_client_stop(instance->context) != CHANNEL_RC_OK)
685 WLog_DBG(TAG,
"Session ended with %08" PRIX32
"", status);
687 if (status == CHANNEL_RC_OK)
688 freerdp_callback(
"OnDisconnected",
"(J)V", (jlong)instance);
690 freerdp_callback(
"OnConnectionFailure",
"(J)V", (jlong)instance);
692 WLog_DBG(TAG,
"Quit.");
697static BOOL android_client_new(freerdp* instance, rdpContext* context)
699 WINPR_ASSERT(instance);
700 WINPR_ASSERT(context);
702 if (!android_event_queue_init(instance))
705 instance->PreConnect = android_pre_connect;
706 instance->PostConnect = android_post_connect;
707 instance->PostDisconnect = android_post_disconnect;
708 instance->AuthenticateEx = android_authenticate_ex;
709 instance->VerifyCertificateEx = android_verify_certificate_ex;
710 instance->VerifyChangedCertificateEx = android_verify_changed_certificate_ex;
711 instance->LogonErrorInfo =
nullptr;
715static void android_client_free(freerdp* instance, rdpContext* context)
720 android_event_queue_uninit(instance);
723static int RdpClientEntry(RDP_CLIENT_ENTRY_POINTS* pEntryPoints)
725 WINPR_ASSERT(pEntryPoints);
727 ZeroMemory(pEntryPoints,
sizeof(RDP_CLIENT_ENTRY_POINTS));
729 pEntryPoints->Version = RDP_CLIENT_INTERFACE_VERSION;
730 pEntryPoints->Size =
sizeof(RDP_CLIENT_ENTRY_POINTS_V1);
731 pEntryPoints->GlobalInit =
nullptr;
732 pEntryPoints->GlobalUninit =
nullptr;
734 pEntryPoints->ClientNew = android_client_new;
735 pEntryPoints->ClientFree = android_client_free;
736 pEntryPoints->ClientStart =
nullptr;
737 pEntryPoints->ClientStop =
nullptr;
741JNIEXPORT jlong JNICALL Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1new(
742 JNIEnv* env, jclass cls, jobject context)
747 jmethodID getFilesDirID;
748 jmethodID getAbsolutePathID;
752 RDP_CLIENT_ENTRY_POINTS clientEntryPoints;
754#if defined(WITH_GPROF)
755 setenv(
"CPUPROFILE_FREQUENCY",
"200", 1);
756 monstartup(
"libfreerdp-android.so");
758 contextClass = (*env)->FindClass(env, JAVA_CONTEXT_CLASS);
759 fileClass = (*env)->FindClass(env, JAVA_FILE_CLASS);
761 if (!contextClass || !fileClass)
763 WLog_FATAL(TAG,
"Failed to load class references %s=%p, %s=%p", JAVA_CONTEXT_CLASS,
764 (
void*)contextClass, JAVA_FILE_CLASS, (
void*)fileClass);
765 return (jlong)
nullptr;
769 (*env)->GetMethodID(env, contextClass,
"getFilesDir",
"()L" JAVA_FILE_CLASS
";");
773 WLog_FATAL(TAG,
"Failed to find method ID getFilesDir ()L" JAVA_FILE_CLASS
";");
774 return (jlong)
nullptr;
778 (*env)->GetMethodID(env, fileClass,
"getAbsolutePath",
"()Ljava/lang/String;");
780 if (!getAbsolutePathID)
782 WLog_FATAL(TAG,
"Failed to find method ID getAbsolutePath ()Ljava/lang/String;");
783 return (jlong)
nullptr;
786 filesDirObj = (*env)->CallObjectMethod(env, context, getFilesDirID);
790 WLog_FATAL(TAG,
"Failed to call getFilesDir");
791 return (jlong)
nullptr;
794 path = (*env)->CallObjectMethod(env, filesDirObj, getAbsolutePathID);
798 WLog_FATAL(TAG,
"Failed to call getAbsolutePath");
799 return (jlong)
nullptr;
802 raw = (*env)->GetStringUTFChars(env, path, 0);
806 WLog_FATAL(TAG,
"Failed to get C string from java string");
807 return (jlong)
nullptr;
810 envStr = _strdup(raw);
811 (*env)->ReleaseStringUTFChars(env, path, raw);
815 WLog_FATAL(TAG,
"_strdup(%s) failed", raw);
816 return (jlong)
nullptr;
819 if (setenv(
"HOME", _strdup(envStr), 1) != 0)
821 char ebuffer[256] = WINPR_C_ARRAY_INIT;
822 WLog_FATAL(TAG,
"Failed to set environment HOME=%s %s [%d]", envStr,
823 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
824 return (jlong)
nullptr;
827 RdpClientEntry(&clientEntryPoints);
828 ctx = freerdp_client_context_new(&clientEntryPoints);
831 return (jlong)
nullptr;
833 return (jlong)ctx->instance;
836JNIEXPORT
void JNICALL Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1free(
837 JNIEnv* env, jclass cls, jlong instance)
839 freerdp* inst = (freerdp*)instance;
842 freerdp_client_context_free(inst->context);
844#if defined(WITH_GPROF)
849JNIEXPORT jstring JNICALL
850Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1get_1last_1error_1string(JNIEnv* env,
854 freerdp* inst = (freerdp*)instance;
856 if (!inst || !inst->context)
857 return (*env)->NewStringUTF(env,
"");
859 return (*env)->NewStringUTF(
860 env, freerdp_get_last_error_string(freerdp_get_last_error(inst->context)));
863JNIEXPORT jboolean JNICALL
864Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1parse_1arguments(JNIEnv* env, jclass cls,
866 jobjectArray arguments)
868 freerdp* inst = (freerdp*)instance;
873 if (!inst || !inst->context)
876 count = (*env)->GetArrayLength(env, arguments);
877 argv = calloc(count,
sizeof(
char*));
882 for (
int i = 0; i < count; i++)
884 jstring str = (jstring)(*env)->GetObjectArrayElement(env, arguments, i);
885 const char* raw = (*env)->GetStringUTFChars(env, str, 0);
886 argv[i] = _strdup(raw);
887 (*env)->ReleaseStringUTFChars(env, str, raw);
891 freerdp_client_settings_parse_command_line(inst->context->settings, count, argv, FALSE);
893 for (
int i = 0; i < count; i++)
897 return (status == 0) ? JNI_TRUE : JNI_FALSE;
900JNIEXPORT jboolean JNICALL Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1connect(
901 JNIEnv* env, jclass cls, jlong instance)
903 freerdp* inst = (freerdp*)instance;
905 if (!inst || !inst->context)
907 WLog_FATAL(TAG,
"(env=%p, cls=%p, instance=%" PRId64, (
void*)env, (
void*)cls,
914 if (!(ctx->thread = CreateThread(
nullptr, 0, android_thread_func, inst, 0,
nullptr)))
922JNIEXPORT jboolean JNICALL Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1disconnect(
923 JNIEnv* env, jclass cls, jlong instance)
925 freerdp* inst = (freerdp*)instance;
927 if (!inst || !inst->context || !cls || !env)
929 WLog_FATAL(TAG,
"(env=%p, cls=%p, instance=%" PRId64, (
void*)env, (
void*)cls,
940 if (!android_push_event(inst, event))
946 if (!freerdp_abort_connect_context(inst->context))
952JNIEXPORT jboolean JNICALL
953Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1update_1graphics(JNIEnv* env, jclass cls,
955 jobject bitmap, jint x,
963 AndroidBitmapInfo info;
964 freerdp* inst = (freerdp*)instance;
967 if (!env || !cls || !inst)
969 WLog_FATAL(TAG,
"(env=%p, cls=%p, instance=%" PRId64, (
void*)env, (
void*)cls,
974 gdi = inst->context->gdi;
976 if ((ret = AndroidBitmap_getInfo(env, bitmap, &info)) < 0)
978 WLog_FATAL(TAG,
"AndroidBitmap_getInfo() failed ! error=%d", ret);
982 if ((ret = AndroidBitmap_lockPixels(env, bitmap, &pixels)) < 0)
984 WLog_FATAL(TAG,
"AndroidBitmap_lockPixels() failed ! error=%d", ret);
992 case ANDROID_BITMAP_FORMAT_RGBA_8888:
993 DstFormat = PIXEL_FORMAT_RGBX32;
996 case ANDROID_BITMAP_FORMAT_RGB_565:
997 DstFormat = PIXEL_FORMAT_RGB16;
1000 case ANDROID_BITMAP_FORMAT_RGBA_4444:
1001 case ANDROID_BITMAP_FORMAT_A_8:
1002 case ANDROID_BITMAP_FORMAT_NONE:
1010 rc = freerdp_image_copy(pixels, DstFormat, info.stride, x, y, width, height,
1011 gdi->primary_buffer, gdi->dstFormat, gdi->stride, x, y,
1012 &gdi->palette, FREERDP_FLIP_NONE);
1015 if ((ret = AndroidBitmap_unlockPixels(env, bitmap)) < 0)
1017 WLog_FATAL(TAG,
"AndroidBitmap_unlockPixels() failed ! error=%d", ret);
1024JNIEXPORT jboolean JNICALL
1025Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1send_1key_1event(JNIEnv* env, jclass cls,
1032 freerdp* inst = (freerdp*)instance;
1033 scancode = GetVirtualScanCodeFromVirtualKeyCode(keycode, 4);
1034 int flags = (down == JNI_TRUE) ? KBD_FLAGS_DOWN : KBD_FLAGS_RELEASE;
1035 flags |= (scancode & KBDEXT) ? KBD_FLAGS_EXTENDED : 0;
1036 event = (
ANDROID_EVENT*)android_event_key_new(flags, scancode & 0xFF);
1041 if (!android_push_event(inst, event))
1043 android_event_free(event);
1047 WLog_DBG(TAG,
"send_key_event: %" PRIu32
", %d", scancode, flags);
1051JNIEXPORT jboolean JNICALL
1052Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1send_1unicodekey_1event(
1053 JNIEnv* env, jclass cls, jlong instance, jint keycode, jboolean down)
1056 freerdp* inst = (freerdp*)instance;
1057 UINT16 flags = (down == JNI_TRUE) ? 0 : KBD_FLAGS_RELEASE;
1058 event = (
ANDROID_EVENT*)android_event_unicodekey_new(flags, keycode);
1063 if (!android_push_event(inst, event))
1065 android_event_free(event);
1069 WLog_DBG(TAG,
"send_unicodekey_event: %d", keycode);
1073JNIEXPORT jboolean JNICALL
1074Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1is_1unicode_1input_1supported(
1075 JNIEnv* env, jclass cls, jlong instance)
1077 freerdp* inst = (freerdp*)instance;
1079 if (!inst || !inst->context || !inst->context->settings)
1088JNIEXPORT jboolean JNICALL
1089Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1send_1cursor_1event(
1090 JNIEnv* env, jclass cls, jlong instance, jint x, jint y, jint flags)
1093 freerdp* inst = (freerdp*)instance;
1094 event = (
ANDROID_EVENT*)android_event_cursor_new(flags, x, y);
1099 if (!android_push_event(inst, event))
1101 android_event_free(event);
1105 WLog_DBG(TAG,
"send_cursor_event: (%d, %d), %d", x, y, flags);
1109static jboolean android_push_clipboard_event(freerdp* inst,
const void* data,
size_t data_length,
1110 const char* mimeType)
1115 if (!android_push_event(inst, event))
1117 android_event_free(event);
1123JNIEXPORT jboolean JNICALL
1124Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1send_1clipboard_1data(JNIEnv* env,
1130 freerdp* inst = (freerdp*)instance;
1131 const char* data = jdata !=
nullptr ? (*env)->GetStringUTFChars(env, jdata,
nullptr) :
nullptr;
1132 const size_t data_length = data ? (*env)->GetStringUTFLength(env, jdata) : 0;
1133 jboolean ret = android_push_clipboard_event(inst, data, data_length,
"text/plain");
1134 WLog_DBG(TAG,
"send_clipboard_data: (%s)", data);
1137 (*env)->ReleaseStringUTFChars(env, jdata, data);
1142static BOOL android_is_image_mime_supported(
const char* mimeType)
1146 if (strcmp(mimeType,
"image/png") == 0)
1147 return winpr_image_format_is_supported(WINPR_IMAGE_PNG);
1148 if (strcmp(mimeType,
"image/jpeg") == 0 || strcmp(mimeType,
"image/jpg") == 0)
1149 return winpr_image_format_is_supported(WINPR_IMAGE_JPEG);
1150 if (strcmp(mimeType,
"image/webp") == 0)
1151 return winpr_image_format_is_supported(WINPR_IMAGE_WEBP);
1155JNIEXPORT jboolean JNICALL
1156Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1send_1clipboard_1image_1data(
1157 JNIEnv* env, jclass cls, jlong instance, jbyteArray jdata, jstring jmimeType)
1160 freerdp* inst = (freerdp*)instance;
1161 jsize data_length = (*env)->GetArrayLength(env, jdata);
1162 jbyte* data = (*env)->GetByteArrayElements(env, jdata,
nullptr);
1163 const char* mimeType = jmimeType ? (*env)->GetStringUTFChars(env, jmimeType,
nullptr) :
nullptr;
1164 jboolean ret = JNI_FALSE;
1165 if (android_is_image_mime_supported(mimeType))
1166 ret = android_push_clipboard_event(inst, data, (
size_t)data_length, mimeType);
1169 (*env)->ReleaseStringUTFChars(env, jmimeType, mimeType);
1170 (*env)->ReleaseByteArrayElements(env, jdata, data, JNI_ABORT);
1174JNIEXPORT jboolean JNICALL
1175Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1send_1monitor_1layout(
1176 JNIEnv* env, jclass cls, jlong instance, jint width, jint height)
1180 freerdp* inst = (freerdp*)instance;
1182 if (!inst || !inst->context)
1186 return android_disp_send_monitor_layout(afc, (UINT32)width, (UINT32)height) ? JNI_TRUE
1190JNIEXPORT jstring JNICALL
1191Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1get_1jni_1version(JNIEnv* env, jclass cls)
1193 return (*env)->NewStringUTF(env, FREERDP_JNI_VERSION);
1196JNIEXPORT jboolean JNICALL
1197Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1has_1h264(JNIEnv* env, jclass cls)
1199 H264_CONTEXT* ctx = h264_context_new(FALSE);
1202 h264_context_free(ctx);
1206JNIEXPORT jboolean JNICALL
1207Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1has_1camera_1redirection(JNIEnv* env,
1210 return freerdp_video_conversion_supported(FREERDP_VIDEO_FORMAT_NV12,
1211 FREERDP_VIDEO_FORMAT_YUV420P)
1216JNIEXPORT jstring JNICALL
1217Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1get_1version(JNIEnv* env, jclass cls)
1219 return (*env)->NewStringUTF(env, freerdp_get_version_string());
1222JNIEXPORT jstring JNICALL
1223Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1get_1build_1revision(JNIEnv* env,
1226 return (*env)->NewStringUTF(env, freerdp_get_build_revision());
1229JNIEXPORT jstring JNICALL
1230Java_com_freerdp_freerdpcore_services_LibFreeRDP_freerdp_1get_1build_1config(JNIEnv* env,
1233 return (*env)->NewStringUTF(env, freerdp_get_build_config());
1236jint JNI_OnLoad(JavaVM* vm,
void* reserved)
1239 setlocale(LC_ALL,
"");
1240 WLog_DBG(TAG,
"Setting up JNI environment...");
1249 if ((*vm)->GetEnv(vm, (
void**)&env, JNI_VERSION_1_6) != JNI_OK)
1251 WLog_FATAL(TAG,
"Failed to get the environment");
1256 jclass activityClass = (*env)->FindClass(env, JAVA_LIBFREERDP_CLASS);
1260 WLog_FATAL(TAG,
"failed to get %s class reference", JAVA_LIBFREERDP_CLASS);
1265 gJavaActivityClass = (*env)->NewGlobalRef(env, activityClass);
1266 gOnPointerSetMethod =
1267 (*env)->GetStaticMethodID(env, gJavaActivityClass,
"OnPointerSet",
"(J[IIIII)V");
1268 gOnRailWindowUpdateMethod =
1269 (*env)->GetStaticMethodID(env, gJavaActivityClass,
"OnRailWindowUpdate",
"(JJII[I)V");
1270 if (!gOnRailWindowUpdateMethod)
1272 (*env)->ExceptionClear(env);
1273 WLog_WARN(TAG,
"OnRailWindowUpdate method not found, RAIL window display disabled");
1276 return init_callback_environment(vm, env);
1279void JNICALL JNI_OnUnload(JavaVM* vm,
void* reserved)
1282 WLog_DBG(TAG,
"Tearing down JNI environment...");
1284 if ((*vm)->GetEnv(vm, (
void**)&env, JNI_VERSION_1_6) != JNI_OK)
1286 WLog_FATAL(TAG,
"Failed to get the environment");
1290 if (gJavaActivityClass)
1291 (*env)->DeleteGlobalRef(env, gJavaActivityClass);
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.