Telegram-Android/TMessagesProj/jni/voip/org_telegram_messenger_voip_Instance.cpp

856 lines
42 KiB
C++
Raw Normal View History

2020-08-14 18:58:22 +02:00
#include "org_telegram_messenger_voip_Instance.h"
#include <jni.h>
#include <sdk/android/native_api/video/wrapper.h>
#include <VideoCapturerInterface.h>
#include <platform/android/AndroidInterface.h>
#include <platform/android/AndroidContext.h>
2020-08-15 02:01:55 +02:00
#include <rtc_base/ssl_adapter.h>
#include <modules/utility/include/jvm_android.h>
#include <sdk/android/native_api/base/init.h>
2020-10-01 03:59:32 +02:00
#include <voip/webrtc/media/base/media_constants.h>
#include <tgnet/FileLog.h>
2021-03-19 11:25:58 +01:00
#include <voip/tgcalls/group/GroupInstanceCustomImpl.h>
2020-12-23 08:48:30 +01:00
#include <memory>
2020-08-14 18:58:22 +02:00
#include "pc/video_track.h"
#include "legacy/InstanceImplLegacy.h"
#include "InstanceImpl.h"
#include "reference/InstanceImplReference.h"
#include "libtgvoip/os/android/AudioOutputOpenSLES.h"
#include "libtgvoip/os/android/AudioInputOpenSLES.h"
#include "libtgvoip/os/android/JNIUtilities.h"
#include "tgcalls/VideoCaptureInterface.h"
2021-03-19 11:25:58 +01:00
#include "rapidjson/document.h"
#include "rapidjson/stringbuffer.h"
#include "rapidjson/writer.h"
2020-08-14 18:58:22 +02:00
using namespace tgcalls;
const auto RegisterTag = Register<InstanceImpl>();
const auto RegisterTagLegacy = Register<InstanceImplLegacy>();
const auto RegisterTagReference = tgcalls::Register<InstanceImplReference>();
2021-03-19 11:25:58 +01:00
jclass TrafficStatsClass;
jclass FingerprintClass;
jclass FinalStateClass;
jclass NativeInstanceClass;
jmethodID FinalStateInitMethod;
class BroadcastPartTaskJava : public BroadcastPartTask {
public:
BroadcastPartTaskJava(std::shared_ptr<PlatformContext> platformContext,
std::function<void(BroadcastPart &&)> callback,
int64_t timestamp) :
_platformContext(platformContext),
_callback(callback),
_timestamp(timestamp) {
}
void call(int64_t ts, int64_t responseTs, BroadcastPart::Status status, uint8_t *data, int32_t len) {
if (_timestamp != ts) {
return;
}
BroadcastPart part;
part.timestampMilliseconds = _timestamp;
part.responseTimestamp = responseTs / 1000.0;
part.status = status;
if (data != nullptr) {
part.oggData = std::vector<uint8_t>(data, data + len);
}
_callback(std::move<>(part));
}
private:
void cancel() override {
tgvoip::jni::DoWithJNI([&](JNIEnv *env) {
jobject globalRef = ((AndroidContext *) _platformContext.get())->getJavaInstance();
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onCancelRequestBroadcastPart", "(J)V"), _timestamp);
});
}
private:
std::shared_ptr<PlatformContext> _platformContext;
std::function<void(BroadcastPart &&)> _callback;
int64_t _timestamp;
};
2020-08-14 18:58:22 +02:00
class JavaObject {
private:
JNIEnv *env;
jobject obj;
jclass clazz;
public:
JavaObject(JNIEnv *env, jobject obj) : JavaObject(env, obj, env->GetObjectClass(obj)) {
}
JavaObject(JNIEnv *env, jobject obj, jclass clazz) {
this->env = env;
this->obj = obj;
this->clazz = clazz;
}
jint getIntField(const char *name) {
return env->GetIntField(obj, env->GetFieldID(clazz, name, "I"));
}
jlong getLongField(const char *name) {
return env->GetLongField(obj, env->GetFieldID(clazz, name, "J"));
}
jboolean getBooleanField(const char *name) {
return env->GetBooleanField(obj, env->GetFieldID(clazz, name, "Z"));
}
jdouble getDoubleField(const char *name) {
return env->GetDoubleField(obj, env->GetFieldID(clazz, name, "D"));
}
jbyteArray getByteArrayField(const char *name) {
return (jbyteArray) env->GetObjectField(obj, env->GetFieldID(clazz, name, "[B"));
}
jstring getStringField(const char *name) {
return (jstring) env->GetObjectField(obj, env->GetFieldID(clazz, name, "Ljava/lang/String;"));
}
};
struct InstanceHolder {
std::unique_ptr<Instance> nativeInstance;
2021-03-19 11:25:58 +01:00
std::unique_ptr<GroupInstanceCustomImpl> groupNativeInstance;
2020-08-14 18:58:22 +02:00
std::shared_ptr<tgcalls::VideoCaptureInterface> _videoCapture;
2020-08-22 01:59:49 +02:00
std::shared_ptr<PlatformContext> _platformContext;
2020-08-14 18:58:22 +02:00
};
jlong getInstanceHolderId(JNIEnv *env, jobject obj) {
2020-08-22 01:59:49 +02:00
return env->GetLongField(obj, env->GetFieldID(NativeInstanceClass, "nativePtr", "J"));
2020-08-14 18:58:22 +02:00
}
InstanceHolder *getInstanceHolder(JNIEnv *env, jobject obj) {
return reinterpret_cast<InstanceHolder *>(getInstanceHolderId(env, obj));
}
jint throwNewJavaException(JNIEnv *env, const char *className, const char *message) {
return env->ThrowNew(env->FindClass(className), message);
}
jint throwNewJavaIllegalArgumentException(JNIEnv *env, const char *message) {
return throwNewJavaException(env, "java/lang/IllegalStateException", message);
}
jbyteArray copyVectorToJavaByteArray(JNIEnv *env, const std::vector<uint8_t> &bytes) {
unsigned int size = bytes.size();
jbyteArray bytesArray = env->NewByteArray(size);
env->SetByteArrayRegion(bytesArray, 0, size, (jbyte *) bytes.data());
return bytesArray;
}
void readPersistentState(const char *filePath, PersistentState &persistentState) {
FILE *persistentStateFile = fopen(filePath, "r");
if (persistentStateFile) {
fseek(persistentStateFile, 0, SEEK_END);
auto len = static_cast<size_t>(ftell(persistentStateFile));
fseek(persistentStateFile, 0, SEEK_SET);
if (len < 1024 * 512 && len > 0) {
auto *buffer = static_cast<uint8_t *>(malloc(len));
fread(buffer, 1, len, persistentStateFile);
persistentState.value = std::vector<uint8_t>(buffer, buffer + len);
free(buffer);
}
fclose(persistentStateFile);
}
}
void savePersistentState(const char *filePath, const PersistentState &persistentState) {
FILE *persistentStateFile = fopen(filePath, "w");
if (persistentStateFile) {
fwrite(persistentState.value.data(), 1, persistentState.value.size(), persistentStateFile);
fclose(persistentStateFile);
}
}
NetworkType parseNetworkType(jint networkType) {
switch (networkType) {
case org_telegram_messenger_voip_Instance_NET_TYPE_GPRS:
return NetworkType::Gprs;
case org_telegram_messenger_voip_Instance_NET_TYPE_EDGE:
return NetworkType::Edge;
case org_telegram_messenger_voip_Instance_NET_TYPE_3G:
return NetworkType::ThirdGeneration;
case org_telegram_messenger_voip_Instance_NET_TYPE_HSPA:
return NetworkType::Hspa;
case org_telegram_messenger_voip_Instance_NET_TYPE_LTE:
return NetworkType::Lte;
case org_telegram_messenger_voip_Instance_NET_TYPE_WIFI:
return NetworkType::WiFi;
case org_telegram_messenger_voip_Instance_NET_TYPE_ETHERNET:
return NetworkType::Ethernet;
case org_telegram_messenger_voip_Instance_NET_TYPE_OTHER_HIGH_SPEED:
return NetworkType::OtherHighSpeed;
case org_telegram_messenger_voip_Instance_NET_TYPE_OTHER_LOW_SPEED:
return NetworkType::OtherLowSpeed;
case org_telegram_messenger_voip_Instance_NET_TYPE_DIALUP:
return NetworkType::Dialup;
case org_telegram_messenger_voip_Instance_NET_TYPE_OTHER_MOBILE:
return NetworkType::OtherMobile;
default:
return NetworkType::Unknown;
}
}
DataSaving parseDataSaving(JNIEnv *env, jint dataSaving) {
switch (dataSaving) {
case org_telegram_messenger_voip_Instance_DATA_SAVING_NEVER:
return DataSaving::Never;
case org_telegram_messenger_voip_Instance_DATA_SAVING_MOBILE:
return DataSaving::Mobile;
case org_telegram_messenger_voip_Instance_DATA_SAVING_ALWAYS:
return DataSaving::Always;
case org_telegram_messenger_voip_Instance_DATA_SAVING_ROAMING:
throwNewJavaIllegalArgumentException(env, "DATA_SAVING_ROAMING is not supported");
return DataSaving::Never;
default:
throwNewJavaIllegalArgumentException(env, "Unknown data saving constant: " + dataSaving);
return DataSaving::Never;
}
}
EndpointType parseEndpointType(JNIEnv *env, jint endpointType) {
switch (endpointType) {
case org_telegram_messenger_voip_Instance_ENDPOINT_TYPE_INET:
return EndpointType::Inet;
case org_telegram_messenger_voip_Instance_ENDPOINT_TYPE_LAN:
return EndpointType::Lan;
case org_telegram_messenger_voip_Instance_ENDPOINT_TYPE_TCP_RELAY:
return EndpointType::TcpRelay;
case org_telegram_messenger_voip_Instance_ENDPOINT_TYPE_UDP_RELAY:
return EndpointType::UdpRelay;
default:
throwNewJavaIllegalArgumentException(env, std::string("Unknown endpoint type: ").append(std::to_string(endpointType)).c_str());
return EndpointType::UdpRelay;
}
}
jint asJavaState(const State &state) {
switch (state) {
case State::WaitInit:
return org_telegram_messenger_voip_Instance_STATE_WAIT_INIT;
case State::WaitInitAck:
return org_telegram_messenger_voip_Instance_STATE_WAIT_INIT_ACK;
case State::Established:
return org_telegram_messenger_voip_Instance_STATE_ESTABLISHED;
case State::Failed:
return org_telegram_messenger_voip_Instance_STATE_FAILED;
case State::Reconnecting:
return org_telegram_messenger_voip_Instance_STATE_RECONNECTING;
}
}
jobject asJavaTrafficStats(JNIEnv *env, const TrafficStats &trafficStats) {
2020-08-15 23:06:36 +02:00
jmethodID initMethodId = env->GetMethodID(TrafficStatsClass, "<init>", "(JJJJ)V");
return env->NewObject(TrafficStatsClass, initMethodId, (jlong) trafficStats.bytesSentWifi, (jlong) trafficStats.bytesReceivedWifi, (jlong) trafficStats.bytesSentMobile, (jlong) trafficStats.bytesReceivedMobile);
2020-08-14 18:58:22 +02:00
}
jobject asJavaFinalState(JNIEnv *env, const FinalState &finalState) {
jbyteArray persistentState = copyVectorToJavaByteArray(env, finalState.persistentState.value);
jstring debugLog = env->NewStringUTF(finalState.debugLog.c_str());
jobject trafficStats = asJavaTrafficStats(env, finalState.trafficStats);
auto isRatingSuggested = static_cast<jboolean>(finalState.isRatingSuggested);
2020-08-15 23:06:36 +02:00
return env->NewObject(FinalStateClass, FinalStateInitMethod, persistentState, debugLog, trafficStats, isRatingSuggested);
2020-08-14 18:58:22 +02:00
}
2020-12-23 08:48:30 +01:00
jobject asJavaFingerprint(JNIEnv *env, std::string hash, std::string setup, std::string fingerprint) {
jstring hashStr = env->NewStringUTF(hash.c_str());
jstring setupStr = env->NewStringUTF(setup.c_str());
jstring fingerprintStr = env->NewStringUTF(fingerprint.c_str());
jmethodID initMethodId = env->GetMethodID(FingerprintClass, "<init>", "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)V");
return env->NewObject(FingerprintClass, initMethodId, hashStr, setupStr, fingerprintStr);
}
2020-08-14 18:58:22 +02:00
extern "C" {
2020-08-15 02:01:55 +02:00
bool webrtcLoaded = false;
void initWebRTC(JNIEnv *env) {
if (webrtcLoaded) {
return;
}
JavaVM* vm;
env->GetJavaVM(&vm);
webrtc::InitAndroid(vm);
webrtc::JVM::Initialize(vm);
rtc::InitializeSSL();
webrtcLoaded = true;
2020-08-15 23:06:36 +02:00
2020-08-22 01:59:49 +02:00
NativeInstanceClass = static_cast<jclass>(env->NewGlobalRef(env->FindClass("org/telegram/messenger/voip/NativeInstance")));
2020-08-15 23:06:36 +02:00
TrafficStatsClass = static_cast<jclass>(env->NewGlobalRef(env->FindClass("org/telegram/messenger/voip/Instance$TrafficStats")));
2020-12-23 08:48:30 +01:00
FingerprintClass = static_cast<jclass>(env->NewGlobalRef(env->FindClass("org/telegram/messenger/voip/Instance$Fingerprint")));
2020-08-15 23:06:36 +02:00
FinalStateClass = static_cast<jclass>(env->NewGlobalRef(env->FindClass("org/telegram/messenger/voip/Instance$FinalState")));
FinalStateInitMethod = env->GetMethodID(FinalStateClass, "<init>", "([BLjava/lang/String;Lorg/telegram/messenger/voip/Instance$TrafficStats;Z)V");
2020-08-15 02:01:55 +02:00
}
2021-03-19 11:25:58 +01:00
JNIEXPORT jlong JNICALL Java_org_telegram_messenger_voip_NativeInstance_makeGroupNativeInstance(JNIEnv *env, jclass clazz, jobject instanceObj, jstring logFilePath, jboolean highQuality) {
2020-12-23 08:48:30 +01:00
initWebRTC(env);
2020-12-24 06:36:01 +01:00
std::shared_ptr<PlatformContext> platformContext = std::make_shared<AndroidContext>(env, instanceObj);
2020-12-23 08:48:30 +01:00
GroupInstanceDescriptor descriptor = {
2021-03-19 11:25:58 +01:00
.threads = StaticThreads::getThreads(),
.config = {
.need_log = false,
//.logPath = tgvoip::jni::JavaStringToStdString(env, logFilePath),
},
.networkStateUpdated = [platformContext](GroupNetworkState state) {
2020-12-24 06:36:01 +01:00
tgvoip::jni::DoWithJNI([platformContext, state](JNIEnv *env) {
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
2021-03-19 11:25:58 +01:00
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onNetworkStateUpdated", "(ZZ)V"), state.isConnected, state.isTransitioningFromBroadcastToRtc);
2020-12-23 08:48:30 +01:00
});
},
2020-12-24 06:36:01 +01:00
.audioLevelsUpdated = [platformContext](GroupLevelsUpdate const &update) {
tgvoip::jni::DoWithJNI([platformContext, update](JNIEnv *env) {
2020-12-23 08:48:30 +01:00
unsigned int size = update.updates.size();
jintArray intArray = env->NewIntArray(size);
jfloatArray floatArray = env->NewFloatArray(size);
jbooleanArray boolArray = env->NewBooleanArray(size);
jint intFill[size];
jfloat floatFill[size];
jboolean boolFill[size];
for (int a = 0; a < size; a++) {
intFill[a] = update.updates[a].ssrc;
floatFill[a] = update.updates[a].value.level;
boolFill[a] = update.updates[a].value.voice;
}
env->SetIntArrayRegion(intArray, 0, size, intFill);
env->SetFloatArrayRegion(floatArray, 0, size, floatFill);
env->SetBooleanArrayRegion(boolArray, 0, size, boolFill);
2020-12-24 06:36:01 +01:00
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
2020-12-23 08:48:30 +01:00
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onAudioLevelsUpdated", "([I[F[Z)V"), intArray, floatArray, boolArray);
env->DeleteLocalRef(intArray);
env->DeleteLocalRef(floatArray);
env->DeleteLocalRef(boolArray);
});
},
2021-03-19 11:25:58 +01:00
.participantDescriptionsRequired = [platformContext](std::vector<uint32_t> const &update) {
tgvoip::jni::DoWithJNI([platformContext, update](JNIEnv *env) {
unsigned int size = update.size();
jintArray intArray = env->NewIntArray(size);
jint intFill[size];
for (int a = 0; a < size; a++) {
intFill[a] = update[a];
}
env->SetIntArrayRegion(intArray, 0, size, intFill);
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onParticipantDescriptionsRequired", "([I)V"), intArray);
env->DeleteLocalRef(intArray);
});
},
.requestBroadcastPart = [](std::shared_ptr<PlatformContext> platformContext, int64_t timestamp, int64_t duration, std::function<void(BroadcastPart &&)> callback) -> std::shared_ptr<BroadcastPartTask> {
std::shared_ptr<BroadcastPartTask> task = std::make_shared<BroadcastPartTaskJava>(platformContext, callback, timestamp);
((AndroidContext *) platformContext.get())->streamTask = task;
tgvoip::jni::DoWithJNI([platformContext, timestamp, duration, task](JNIEnv *env) {
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onRequestBroadcastPart", "(JJ)V"), timestamp, duration);
});
return task;
},
2020-12-23 08:48:30 +01:00
.platformContext = platformContext
};
auto *holder = new InstanceHolder;
2021-03-19 11:25:58 +01:00
holder->groupNativeInstance = std::make_unique<GroupInstanceCustomImpl>(std::move(descriptor));
2020-12-23 08:48:30 +01:00
holder->_platformContext = platformContext;
return reinterpret_cast<jlong>(holder);
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setJoinResponsePayload(JNIEnv *env, jobject obj, jstring ufrag, jstring pwd, jobjectArray fingerprints, jobjectArray candidates) {
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->groupNativeInstance == nullptr) {
return;
}
std::vector<GroupJoinPayloadFingerprint> fingerprintsArray;
std::vector<GroupJoinResponseCandidate> candidatesArray;
2021-03-19 11:25:58 +01:00
instance->groupNativeInstance->setConnectionMode(GroupConnectionMode::GroupConnectionModeRtc, true);
2020-12-23 08:48:30 +01:00
jsize size = env->GetArrayLength(fingerprints);
for (int i = 0; i < size; i++) {
JavaObject fingerprintObject(env, env->GetObjectArrayElement(fingerprints, i));
fingerprintsArray.push_back(
{
.hash = tgvoip::jni::JavaStringToStdString(env, fingerprintObject.getStringField("hash")),
.setup = tgvoip::jni::JavaStringToStdString(env, fingerprintObject.getStringField("setup")),
.fingerprint = tgvoip::jni::JavaStringToStdString(env, fingerprintObject.getStringField("fingerprint"))
});
}
size = env->GetArrayLength(candidates);
for (int i = 0; i < size; i++) {
JavaObject candidateObject(env, env->GetObjectArrayElement(candidates, i));
candidatesArray.push_back(
{
.port = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("port")),
.protocol = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("protocol")),
.network = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("network")),
.generation = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("generation")),
.id = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("id")),
.component = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("component")),
.foundation = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("foundation")),
.priority = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("priority")),
.ip = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("ip")),
.type = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("type")),
.tcpType = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("tcpType")),
.relAddr = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("relAddr")),
.relPort = tgvoip::jni::JavaStringToStdString(env, candidateObject.getStringField("relPort")),
});
}
2021-03-19 11:25:58 +01:00
std::vector<tgcalls::GroupParticipantDescription> participants;
2020-12-23 08:48:30 +01:00
instance->groupNativeInstance->setJoinResponsePayload(
{
.ufrag = tgvoip::jni::JavaStringToStdString(env, ufrag),
.pwd = tgvoip::jni::JavaStringToStdString(env, pwd),
.fingerprints = fingerprintsArray,
.candidates = candidatesArray,
2021-03-19 11:25:58 +01:00
}, std::move(participants));
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_prepareForStream(JNIEnv *env, jobject obj) {
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->groupNativeInstance == nullptr) {
return;
}
instance->groupNativeInstance->setConnectionMode(GroupConnectionMode::GroupConnectionModeBroadcast, true);
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_resetGroupInstance(JNIEnv *env, jobject obj, jboolean disconnect) {
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->groupNativeInstance == nullptr) {
return;
}
instance->groupNativeInstance->setConnectionMode(GroupConnectionMode::GroupConnectionModeNone, !disconnect);
std::shared_ptr<PlatformContext> platformContext = instance->_platformContext;
instance->groupNativeInstance->emitJoinPayload([platformContext](const GroupJoinPayload& payload) {
JNIEnv *env = webrtc::AttachCurrentThreadIfNeeded();
jobjectArray array = env->NewObjectArray(payload.fingerprints.size(), FingerprintClass, 0);
for (int a = 0; a < payload.fingerprints.size(); a++) {
env->SetObjectArrayElement(array, a, asJavaFingerprint(env, payload.fingerprints[a].hash, payload.fingerprints[a].setup, payload.fingerprints[a].fingerprint));
}
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onEmitJoinPayload", "(Ljava/lang/String;Ljava/lang/String;[Lorg/telegram/messenger/voip/Instance$Fingerprint;I)V"), env->NewStringUTF(payload.ufrag.c_str()), env->NewStringUTF(payload.pwd.c_str()), array, (jint) payload.ssrc);
});
2020-12-23 08:48:30 +01:00
}
2021-03-19 11:25:58 +01:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_addParticipants(JNIEnv *env, jobject obj, jintArray ssrcs, jobjectArray array) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->groupNativeInstance == nullptr) {
return;
}
2021-03-19 11:25:58 +01:00
rapidjson::Document d;
jint *ssrcsArr = env->GetIntArrayElements(ssrcs, 0);
jsize size = env->GetArrayLength(array);
std::vector<tgcalls::GroupParticipantDescription> participants;
2020-12-23 08:48:30 +01:00
for (int i = 0; i < size; i++) {
2021-03-19 11:25:58 +01:00
GroupParticipantDescription participantDescription;
participantDescription.audioSsrc = ssrcsArr[i];
jstring str = (jstring) env->GetObjectArrayElement(array, i);
if (str != nullptr) {
std::string json = tgvoip::jni::JavaStringToStdString(env, str);
d.Parse(json);
participantDescription.endpointId = d["endpoint"].GetString();
for (const auto &group : d["ssrc-groups"].GetArray()) {
tgcalls::GroupJoinPayloadVideoSourceGroup groupDesc;
groupDesc.semantics = group["semantics"].GetString();
for (const auto &source : group["sources"].GetArray()) {
groupDesc.ssrcs.push_back(source.GetUint());
}
participantDescription.videoSourceGroups.push_back(std::move(groupDesc));
}
for (const auto &extDict : d["rtp-hdrexts"].GetArray()) {
participantDescription.videoExtensionMap.emplace_back(extDict["id"].GetUint(), extDict["uri"].GetString());
}
for (const auto &payload : d["payload-types"].GetArray()) {
tgcalls::GroupJoinPayloadVideoPayloadType parsedPayload;
parsedPayload.id = payload["id"].GetUint();
parsedPayload.clockrate = payload["clockrate"].GetUint();
parsedPayload.channels = payload["channels"].GetUint();
parsedPayload.name = payload["name"].GetString();
for (const auto &fb : payload["rtcp-fbs"].GetArray()) {
tgcalls::GroupJoinPayloadVideoPayloadFeedbackType parsedFeedback;
parsedFeedback.type = fb["type"].GetString();
if (fb.HasMember("subtype")) {
parsedFeedback.subtype = fb["subtype"].GetString();
}
parsedPayload.feedbackTypes.push_back(std::move(parsedFeedback));
}
for (const auto &fb : payload["parameters"].GetObject()) {
parsedPayload.parameters.emplace_back(fb.name.GetString(), fb.value.GetString());
}
participantDescription.videoPayloadTypes.push_back(std::move(parsedPayload));
}
}
participants.push_back(std::move(participantDescription));
2020-12-23 08:48:30 +01:00
}
2021-03-19 11:25:58 +01:00
env->ReleaseIntArrayElements(ssrcs, ssrcsArr, JNI_ABORT);
instance->groupNativeInstance->addParticipants(std::move(participants));
2020-12-23 08:48:30 +01:00
}
2020-08-14 18:58:22 +02:00
JNIEXPORT jlong JNICALL Java_org_telegram_messenger_voip_NativeInstance_makeNativeInstance(JNIEnv *env, jclass clazz, jstring version, jobject instanceObj, jobject config, jstring persistentStateFilePath, jobjectArray endpoints, jobject proxyClass, jint networkType, jobject encryptionKey, jobject remoteSink, jlong videoCapturer, jfloat aspectRatio) {
2020-08-15 02:01:55 +02:00
initWebRTC(env);
2020-08-14 18:58:22 +02:00
JavaObject configObject(env, config);
JavaObject encryptionKeyObject(env, encryptionKey);
std::string v = tgvoip::jni::JavaStringToStdString(env, version);
jbyteArray valueByteArray = encryptionKeyObject.getByteArrayField("value");
auto *valueBytes = (uint8_t *) env->GetByteArrayElements(valueByteArray, nullptr);
auto encryptionKeyValue = std::make_shared<std::array<uint8_t, 256>>();
memcpy(encryptionKeyValue->data(), valueBytes, 256);
env->ReleaseByteArrayElements(valueByteArray, (jbyte *) valueBytes, JNI_ABORT);
std::shared_ptr<VideoCaptureInterface> videoCapture = videoCapturer ? std::shared_ptr<VideoCaptureInterface>(reinterpret_cast<VideoCaptureInterface *>(videoCapturer)) : nullptr;
2020-12-24 06:36:01 +01:00
std::shared_ptr<PlatformContext> platformContext;
if (videoCapture) {
platformContext = videoCapture->getPlatformContext();
((AndroidContext *) platformContext.get())->setJavaInstance(env, instanceObj);
} else {
platformContext = std::make_shared<AndroidContext>(env, instanceObj);
}
2020-08-22 01:59:49 +02:00
2020-08-14 18:58:22 +02:00
Descriptor descriptor = {
.config = Config{
.initializationTimeout = configObject.getDoubleField("initializationTimeout"),
.receiveTimeout = configObject.getDoubleField("receiveTimeout"),
.dataSaving = parseDataSaving(env, configObject.getIntField("dataSaving")),
.enableP2P = configObject.getBooleanField("enableP2p") == JNI_TRUE,
2020-12-24 06:36:01 +01:00
.enableStunMarking = configObject.getBooleanField("enableSm") == JNI_TRUE,
2020-08-14 18:58:22 +02:00
.enableAEC = configObject.getBooleanField("enableAec") == JNI_TRUE,
.enableNS = configObject.getBooleanField("enableNs") == JNI_TRUE,
.enableAGC = configObject.getBooleanField("enableAgc") == JNI_TRUE,
.enableVolumeControl = true,
.logPath = tgvoip::jni::JavaStringToStdString(env, configObject.getStringField("logPath")),
2020-12-24 06:36:01 +01:00
.statsLogPath = tgvoip::jni::JavaStringToStdString(env, configObject.getStringField("statsLogPath")),
2020-08-14 18:58:22 +02:00
.maxApiLayer = configObject.getIntField("maxApiLayer"),
2020-10-01 03:59:32 +02:00
.enableHighBitrateVideo = true,
.preferredVideoCodecs = {cricket::kVp9CodecName}
2020-08-14 18:58:22 +02:00
},
.encryptionKey = EncryptionKey(
std::move(encryptionKeyValue),
encryptionKeyObject.getBooleanField("isOutgoing") == JNI_TRUE),
.videoCapture = videoCapture,
2020-12-24 06:36:01 +01:00
.stateUpdated = [platformContext](State state) {
2020-08-14 18:58:22 +02:00
jint javaState = asJavaState(state);
2020-12-24 06:36:01 +01:00
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
2020-08-14 18:58:22 +02:00
tgvoip::jni::DoWithJNI([globalRef, javaState](JNIEnv *env) {
2020-08-22 01:59:49 +02:00
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onStateUpdated", "(I)V"), javaState);
2020-08-14 18:58:22 +02:00
});
},
2020-12-24 06:36:01 +01:00
.signalBarsUpdated = [platformContext](int count) {
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
2020-08-14 18:58:22 +02:00
tgvoip::jni::DoWithJNI([globalRef, count](JNIEnv *env) {
2020-08-22 01:59:49 +02:00
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onSignalBarsUpdated", "(I)V"), count);
2020-08-14 18:58:22 +02:00
});
},
2021-01-28 15:15:51 +01:00
.audioLevelUpdated = [platformContext](float level) {
tgvoip::jni::DoWithJNI([platformContext, level](JNIEnv *env) {
jintArray intArray = nullptr;
jfloatArray floatArray = env->NewFloatArray(1);
jbooleanArray boolArray = nullptr;
jfloat floatFill[1];
floatFill[0] = level;
env->SetFloatArrayRegion(floatArray, 0, 1, floatFill);
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onAudioLevelsUpdated", "([I[F[Z)V"), intArray, floatArray, boolArray);
env->DeleteLocalRef(floatArray);
});
},
2020-12-24 06:36:01 +01:00
.remoteMediaStateUpdated = [platformContext](AudioState audioState, VideoState videoState) {
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
2020-08-14 18:58:22 +02:00
tgvoip::jni::DoWithJNI([globalRef, audioState, videoState](JNIEnv *env) {
2020-08-22 01:59:49 +02:00
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onRemoteMediaStateUpdated", "(II)V"), (jint) audioState, (jint )videoState);
2020-08-14 18:58:22 +02:00
});
},
2020-12-24 06:36:01 +01:00
.signalingDataEmitted = [platformContext](const std::vector<uint8_t> &data) {
jobject globalRef = ((AndroidContext *) platformContext.get())->getJavaInstance();
2020-08-14 18:58:22 +02:00
tgvoip::jni::DoWithJNI([globalRef, data](JNIEnv *env) {
jbyteArray arr = copyVectorToJavaByteArray(env, data);
2020-08-22 01:59:49 +02:00
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onSignalingData", "([B)V"), arr);
env->DeleteLocalRef(arr);
2020-08-14 18:58:22 +02:00
});
},
2020-12-24 06:36:01 +01:00
.platformContext = platformContext,
2020-08-14 18:58:22 +02:00
};
for (int i = 0, size = env->GetArrayLength(endpoints); i < size; i++) {
JavaObject endpointObject(env, env->GetObjectArrayElement(endpoints, i));
bool isRtc = endpointObject.getBooleanField("isRtc");
if (isRtc) {
RtcServer rtcServer;
rtcServer.host = tgvoip::jni::JavaStringToStdString(env, endpointObject.getStringField("ipv4"));
rtcServer.port = static_cast<uint16_t>(endpointObject.getIntField("port"));
rtcServer.login = tgvoip::jni::JavaStringToStdString(env, endpointObject.getStringField("username"));
rtcServer.password = tgvoip::jni::JavaStringToStdString(env, endpointObject.getStringField("password"));
rtcServer.isTurn = endpointObject.getBooleanField("turn");
descriptor.rtcServers.push_back(std::move(rtcServer));
} else {
Endpoint endpoint;
endpoint.endpointId = endpointObject.getLongField("id");
endpoint.host = EndpointHost{tgvoip::jni::JavaStringToStdString(env, endpointObject.getStringField("ipv4")), tgvoip::jni::JavaStringToStdString(env, endpointObject.getStringField("ipv6"))};
endpoint.port = static_cast<uint16_t>(endpointObject.getIntField("port"));
endpoint.type = parseEndpointType(env, endpointObject.getIntField("type"));
jbyteArray peerTag = endpointObject.getByteArrayField("peerTag");
if (peerTag && env->GetArrayLength(peerTag)) {
jbyte *peerTagBytes = env->GetByteArrayElements(peerTag, nullptr);
memcpy(endpoint.peerTag, peerTagBytes, 16);
env->ReleaseByteArrayElements(peerTag, peerTagBytes, JNI_ABORT);
}
descriptor.endpoints.push_back(std::move(endpoint));
}
}
if (!env->IsSameObject(proxyClass, nullptr)) {
JavaObject proxyObject(env, proxyClass);
descriptor.proxy = std::unique_ptr<Proxy>(new Proxy);
descriptor.proxy->host = tgvoip::jni::JavaStringToStdString(env, proxyObject.getStringField("host"));
descriptor.proxy->port = static_cast<uint16_t>(proxyObject.getIntField("port"));
descriptor.proxy->login = tgvoip::jni::JavaStringToStdString(env, proxyObject.getStringField("login"));
descriptor.proxy->password = tgvoip::jni::JavaStringToStdString(env, proxyObject.getStringField("password"));
}
readPersistentState(tgvoip::jni::JavaStringToStdString(env, persistentStateFilePath).c_str(), descriptor.persistentState);
auto *holder = new InstanceHolder;
holder->nativeInstance = tgcalls::Meta::Create(v, std::move(descriptor));
holder->_videoCapture = videoCapture;
2020-08-22 01:59:49 +02:00
holder->_platformContext = platformContext;
2020-08-14 18:58:22 +02:00
holder->nativeInstance->setIncomingVideoOutput(webrtc::JavaToNativeVideoSink(env, remoteSink));
2020-08-15 23:06:36 +02:00
holder->nativeInstance->setNetworkType(parseNetworkType(networkType));
2020-10-01 03:59:32 +02:00
holder->nativeInstance->setRequestedVideoAspect(aspectRatio);
2020-08-14 18:58:22 +02:00
return reinterpret_cast<jlong>(holder);
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setGlobalServerConfig(JNIEnv *env, jobject obj, jstring serverConfigJson) {
SetLegacyGlobalServerConfig(tgvoip::jni::JavaStringToStdString(env, serverConfigJson));
}
JNIEXPORT jstring JNICALL Java_org_telegram_messenger_voip_NativeInstance_getVersion(JNIEnv *env, jobject obj) {
return env->NewStringUTF(tgvoip::VoIPController::GetVersion());
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setBufferSize(JNIEnv *env, jobject obj, jint size) {
tgvoip::audio::AudioOutputOpenSLES::nativeBufferSize = (unsigned int) size;
tgvoip::audio::AudioInputOpenSLES::nativeBufferSize = (unsigned int) size;
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setNetworkType(JNIEnv *env, jobject obj, jint networkType) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return;
}
instance->nativeInstance->setNetworkType(parseNetworkType(networkType));
2020-08-14 18:58:22 +02:00
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setMuteMicrophone(JNIEnv *env, jobject obj, jboolean muteMicrophone) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance != nullptr) {
instance->nativeInstance->setMuteMicrophone(muteMicrophone);
} else if (instance->groupNativeInstance != nullptr) {
instance->groupNativeInstance->setIsMuted(muteMicrophone);
}
2020-08-14 18:58:22 +02:00
}
2021-01-28 15:15:51 +01:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setVolume(JNIEnv *env, jobject obj, jint ssrc, jdouble volume) {
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->groupNativeInstance != nullptr) {
instance->groupNativeInstance->setVolume(ssrc, volume);
}
}
2020-08-14 18:58:22 +02:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setAudioOutputGainControlEnabled(JNIEnv *env, jobject obj, jboolean enabled) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return;
}
instance->nativeInstance->setAudioOutputGainControlEnabled(enabled);
2020-08-14 18:58:22 +02:00
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setEchoCancellationStrength(JNIEnv *env, jobject obj, jint strength) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return;
}
instance->nativeInstance->setEchoCancellationStrength(strength);
2020-08-14 18:58:22 +02:00
}
JNIEXPORT jstring JNICALL Java_org_telegram_messenger_voip_NativeInstance_getLastError(JNIEnv *env, jobject obj) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return nullptr;
}
return env->NewStringUTF(instance->nativeInstance->getLastError().c_str());
2020-08-14 18:58:22 +02:00
}
JNIEXPORT jstring JNICALL Java_org_telegram_messenger_voip_NativeInstance_getDebugInfo(JNIEnv *env, jobject obj) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return nullptr;
}
return env->NewStringUTF(instance->nativeInstance->getDebugInfo().c_str());
2020-08-14 18:58:22 +02:00
}
JNIEXPORT jlong JNICALL Java_org_telegram_messenger_voip_NativeInstance_getPreferredRelayId(JNIEnv *env, jobject obj) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return 0;
}
return instance->nativeInstance->getPreferredRelayId();
2020-08-14 18:58:22 +02:00
}
JNIEXPORT jobject JNICALL Java_org_telegram_messenger_voip_NativeInstance_getTrafficStats(JNIEnv *env, jobject obj) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return nullptr;
}
return asJavaTrafficStats(env, instance->nativeInstance->getTrafficStats());
2020-08-14 18:58:22 +02:00
}
JNIEXPORT jbyteArray JNICALL Java_org_telegram_messenger_voip_NativeInstance_getPersistentState(JNIEnv *env, jobject obj) {
2020-12-23 08:48:30 +01:00
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->nativeInstance == nullptr) {
return nullptr;
}
return copyVectorToJavaByteArray(env, instance->nativeInstance->getPersistentState().value);
2020-08-14 18:58:22 +02:00
}
2020-08-15 23:06:36 +02:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_stopNative(JNIEnv *env, jobject obj) {
2020-08-14 18:58:22 +02:00
InstanceHolder *instance = getInstanceHolder(env, obj);
2020-12-23 08:48:30 +01:00
if (instance->nativeInstance == nullptr) {
return;
}
2020-08-15 23:06:36 +02:00
instance->nativeInstance->stop([instance](FinalState finalState) {
JNIEnv *env = webrtc::AttachCurrentThreadIfNeeded();
2020-12-24 06:36:01 +01:00
jobject globalRef = ((AndroidContext *) instance->_platformContext.get())->getJavaInstance();
const std::string &path = tgvoip::jni::JavaStringToStdString(env, JavaObject(env, globalRef).getStringField("persistentStateFilePath"));
2020-08-15 23:06:36 +02:00
savePersistentState(path.c_str(), finalState.persistentState);
2020-12-24 06:36:01 +01:00
env->CallVoidMethod(globalRef, env->GetMethodID(NativeInstanceClass, "onStop", "(Lorg/telegram/messenger/voip/Instance$FinalState;)V"), asJavaFinalState(env, finalState));
2020-08-15 23:06:36 +02:00
delete instance;
});
2020-08-14 18:58:22 +02:00
}
2020-12-23 08:48:30 +01:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_stopGroupNative(JNIEnv *env, jobject obj) {
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->groupNativeInstance == nullptr) {
return;
}
instance->groupNativeInstance->stop();
instance->groupNativeInstance.reset();
delete instance;
}
2021-03-19 11:25:58 +01:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_onStreamPartAvailable(JNIEnv *env, jobject obj, jlong ts, jobject byteBuffer, jint size, jlong responseTs) {
InstanceHolder *instance = getInstanceHolder(env, obj);
if (instance->groupNativeInstance == nullptr) {
return;
}
AndroidContext *context = (AndroidContext *) instance->_platformContext.get();
std::shared_ptr<BroadcastPartTask> streamTask = context->streamTask;
BroadcastPartTaskJava *task = (BroadcastPartTaskJava *) streamTask.get();
if (task != nullptr) {
if (byteBuffer != nullptr) {
uint8_t *buf = (uint8_t *) env->GetDirectBufferAddress(byteBuffer);
task->call(ts, responseTs, BroadcastPart::Status::Success, buf, size);
} else {
task->call(ts, responseTs, size == 0 ? BroadcastPart::Status::NotReady : BroadcastPart::Status::ResyncNeeded, nullptr, 0);
}
}
}
2020-10-01 03:59:32 +02:00
JNIEXPORT jlong JNICALL Java_org_telegram_messenger_voip_NativeInstance_createVideoCapturer(JNIEnv *env, jclass clazz, jobject localSink, jboolean front) {
2020-08-15 02:01:55 +02:00
initWebRTC(env);
2021-03-19 11:25:58 +01:00
std::unique_ptr<VideoCaptureInterface> capture = tgcalls::VideoCaptureInterface::Create(StaticThreads::getThreads(), front ? "front" : "back", std::make_shared<AndroidContext>(env, nullptr));
2020-08-14 18:58:22 +02:00
capture->setOutput(webrtc::JavaToNativeVideoSink(env, localSink));
capture->setState(VideoState::Active);
return reinterpret_cast<intptr_t>(capture.release());
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_destroyVideoCapturer(JNIEnv *env, jclass clazz, jlong videoCapturer) {
VideoCaptureInterface *capturer = reinterpret_cast<VideoCaptureInterface *>(videoCapturer);
delete capturer;
}
2020-10-01 03:59:32 +02:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_switchCameraCapturer(JNIEnv *env, jclass clazz, jlong videoCapturer, jboolean front) {
2020-08-14 18:58:22 +02:00
VideoCaptureInterface *capturer = reinterpret_cast<VideoCaptureInterface *>(videoCapturer);
2020-10-01 03:59:32 +02:00
capturer->switchToDevice(front ? "front" : "back");
2020-08-14 18:58:22 +02:00
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setVideoStateCapturer(JNIEnv *env, jclass clazz, jlong videoCapturer, jint videoState) {
VideoCaptureInterface *capturer = reinterpret_cast<VideoCaptureInterface *>(videoCapturer);
capturer->setState(static_cast<VideoState>(videoState));
}
2020-10-01 03:59:32 +02:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_switchCamera(JNIEnv *env, jobject obj, jboolean front) {
2020-08-14 18:58:22 +02:00
InstanceHolder *instance = getInstanceHolder(env, obj);
2020-12-23 08:48:30 +01:00
if (instance->nativeInstance == nullptr) {
return;
}
2020-08-14 18:58:22 +02:00
if (instance->_videoCapture == nullptr) {
return;
}
2020-10-01 03:59:32 +02:00
instance->_videoCapture->switchToDevice(front ? "front" : "back");
2020-08-14 18:58:22 +02:00
}
JNIEXPORT void Java_org_telegram_messenger_voip_NativeInstance_setVideoState(JNIEnv *env, jobject obj, jint state) {
InstanceHolder *instance = getInstanceHolder(env, obj);
2020-12-23 08:48:30 +01:00
if (instance->nativeInstance == nullptr) {
return;
}
2020-08-14 18:58:22 +02:00
if (instance->_videoCapture == nullptr) {
return;
}
instance->_videoCapture->setState(static_cast<VideoState>(state));
}
2020-10-01 03:59:32 +02:00
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_setupOutgoingVideo(JNIEnv *env, jobject obj, jobject localSink, jboolean front) {
2020-08-14 18:58:22 +02:00
InstanceHolder *instance = getInstanceHolder(env, obj);
2020-12-23 08:48:30 +01:00
if (instance->nativeInstance == nullptr) {
return;
}
2020-08-14 18:58:22 +02:00
if (instance->_videoCapture) {
return;
}
2021-03-19 11:25:58 +01:00
instance->_videoCapture = tgcalls::VideoCaptureInterface::Create(StaticThreads::getThreads(), front ? "front" : "back", instance->_platformContext);
2020-08-14 18:58:22 +02:00
instance->_videoCapture->setOutput(webrtc::JavaToNativeVideoSink(env, localSink));
instance->_videoCapture->setState(VideoState::Active);
instance->nativeInstance->setVideoCapture(instance->_videoCapture);
}
JNIEXPORT void JNICALL Java_org_telegram_messenger_voip_NativeInstance_onSignalingDataReceive(JNIEnv *env, jobject obj, jbyteArray value) {
InstanceHolder *instance = getInstanceHolder(env, obj);
2020-12-23 08:48:30 +01:00
if (instance->nativeInstance == nullptr) {
return;
}
2020-08-14 18:58:22 +02:00
auto *valueBytes = (uint8_t *) env->GetByteArrayElements(value, nullptr);
const size_t size = env->GetArrayLength(value);
auto array = std::vector<uint8_t>(size);
memcpy(&array[0], valueBytes, size);
instance->nativeInstance->receiveSignalingData(std::move(array));
env->ReleaseByteArrayElements(value, (jbyte *) valueBytes, JNI_ABORT);
}
}