mirror of
https://github.com/NekoX-Dev/NekoX.git
synced 2025-01-06 01:36:28 +01:00
736 lines
40 KiB
C++
736 lines
40 KiB
C++
#include <jni.h>
|
|
#include "tgnet/ApiScheme.h"
|
|
#include "tgnet/BuffersStorage.h"
|
|
#include "tgnet/NativeByteBuffer.h"
|
|
#include "tgnet/ConnectionsManager.h"
|
|
#include "tgnet/MTProtoScheme.h"
|
|
#include "tgnet/ConnectionSocket.h"
|
|
|
|
JavaVM *java;
|
|
jclass jclass_RequestDelegateInternal;
|
|
jmethodID jclass_RequestDelegateInternal_run;
|
|
|
|
jclass jclass_RequestTimeDelegate;
|
|
jmethodID jclass_RequestTimeDelegate_run;
|
|
|
|
jclass jclass_QuickAckDelegate;
|
|
jmethodID jclass_QuickAckDelegate_run;
|
|
|
|
jclass jclass_WriteToSocketDelegate;
|
|
jmethodID jclass_WriteToSocketDelegate_run;
|
|
|
|
jclass jclass_ConnectionsManager;
|
|
jmethodID jclass_ConnectionsManager_onUnparsedMessageReceived;
|
|
jmethodID jclass_ConnectionsManager_onUpdate;
|
|
jmethodID jclass_ConnectionsManager_onSessionCreated;
|
|
jmethodID jclass_ConnectionsManager_onLogout;
|
|
jmethodID jclass_ConnectionsManager_onConnectionStateChanged;
|
|
jmethodID jclass_ConnectionsManager_onInternalPushReceived;
|
|
jmethodID jclass_ConnectionsManager_onUpdateConfig;
|
|
jmethodID jclass_ConnectionsManager_onBytesSent;
|
|
jmethodID jclass_ConnectionsManager_onBytesReceived;
|
|
jmethodID jclass_ConnectionsManager_onRequestNewServerIpAndPort;
|
|
jmethodID jclass_ConnectionsManager_onProxyError;
|
|
jmethodID jclass_ConnectionsManager_getHostByName;
|
|
jmethodID jclass_ConnectionsManager_getInitFlags;
|
|
|
|
bool check_utf8(const char *data, size_t len);
|
|
|
|
jlong getFreeBuffer(JNIEnv *env, jclass c, jint length) {
|
|
return (jlong) (intptr_t) BuffersStorage::getInstance().getFreeBuffer((uint32_t) length);
|
|
}
|
|
|
|
jint limit(JNIEnv *env, jclass c, jlong address) {
|
|
NativeByteBuffer *buffer = (NativeByteBuffer *) (intptr_t) address;
|
|
return buffer->limit();
|
|
}
|
|
|
|
jint position(JNIEnv *env, jclass c, jlong address) {
|
|
NativeByteBuffer *buffer = (NativeByteBuffer *) (intptr_t) address;
|
|
return buffer->position();
|
|
}
|
|
|
|
void reuse(JNIEnv *env, jclass c, jlong address) {
|
|
NativeByteBuffer *buffer = (NativeByteBuffer *) (intptr_t) address;
|
|
buffer->reuse();
|
|
}
|
|
|
|
jobject getJavaByteBuffer(JNIEnv *env, jclass c, jlong address) {
|
|
NativeByteBuffer *buffer = (NativeByteBuffer *) (intptr_t) address;
|
|
if (buffer == nullptr) {
|
|
return nullptr;
|
|
}
|
|
return buffer->getJavaByteBuffer();
|
|
}
|
|
|
|
static const char *NativeByteBufferClassPathName = "org/telegram/tgnet/NativeByteBuffer";
|
|
static JNINativeMethod NativeByteBufferMethods[] = {
|
|
{"native_getFreeBuffer", "(I)J", (void *) getFreeBuffer},
|
|
{"native_limit", "(J)I", (void *) limit},
|
|
{"native_position", "(J)I", (void *) position},
|
|
{"native_reuse", "(J)V", (void *) reuse},
|
|
{"native_getJavaByteBuffer", "(J)Ljava/nio/ByteBuffer;", (void *) getJavaByteBuffer}
|
|
};
|
|
|
|
jlong getCurrentTimeMillis(JNIEnv *env, jclass c, jint instanceNum) {
|
|
return ConnectionsManager::getInstance(instanceNum).getCurrentTimeMillis() +
|
|
((jlong) ConnectionsManager::getInstance(instanceNum).getTimeDifference()) * 1000;
|
|
}
|
|
|
|
jint getCurrentTime(JNIEnv *env, jclass c, jint instanceNum) {
|
|
return ConnectionsManager::getInstance(instanceNum).getCurrentTime();
|
|
}
|
|
|
|
jint getCurrentDatacenterId(JNIEnv *env, jclass c, jint instanceNum) {
|
|
return ConnectionsManager::getInstance(instanceNum).getCurrentDatacenterId();
|
|
}
|
|
|
|
jint isTestBackend(JNIEnv *env, jclass c, jint instanceNum) {
|
|
return ConnectionsManager::getInstance(instanceNum).isTestBackend() ? 1 : 0;
|
|
}
|
|
|
|
jint getTimeDifference(JNIEnv *env, jclass c, jint instanceNum) {
|
|
return ConnectionsManager::getInstance(instanceNum).getTimeDifference();
|
|
}
|
|
|
|
void sendRequest(JNIEnv *env, jclass c, jint instanceNum, jlong object, jobject onComplete,
|
|
jobject onQuickAck, jobject onWriteToSocket, jint flags, jint datacenterId,
|
|
jint connetionType, jboolean immediate, jint token) {
|
|
TL_api_request *request = new TL_api_request();
|
|
request->request = (NativeByteBuffer *) (intptr_t) object;
|
|
if (onComplete != nullptr) {
|
|
onComplete = env->NewGlobalRef(onComplete);
|
|
}
|
|
if (onQuickAck != nullptr) {
|
|
onQuickAck = env->NewGlobalRef(onQuickAck);
|
|
}
|
|
if (onWriteToSocket != nullptr) {
|
|
onWriteToSocket = env->NewGlobalRef(onWriteToSocket);
|
|
}
|
|
ConnectionsManager::getInstance(instanceNum).sendRequest(request, ([onComplete, instanceNum](
|
|
TLObject *response, TL_error *error, int32_t networkType, int64_t responseTime) {
|
|
TL_api_response *resp = (TL_api_response *) response;
|
|
jlong ptr = 0;
|
|
jint errorCode = 0;
|
|
jstring errorText = nullptr;
|
|
if (resp != nullptr) {
|
|
ptr = (jlong) resp->response.get();
|
|
} else if (error != nullptr) {
|
|
errorCode = error->code;
|
|
const char *text = error->text.c_str();
|
|
size_t size = error->text.size();
|
|
if (check_utf8(text, size)) {
|
|
errorText = jniEnv[instanceNum]->NewStringUTF(text);
|
|
} else {
|
|
errorText = jniEnv[instanceNum]->NewStringUTF("UTF-8 ERROR");
|
|
}
|
|
}
|
|
if (onComplete != nullptr) {
|
|
jniEnv[instanceNum]->CallVoidMethod(onComplete, jclass_RequestDelegateInternal_run, ptr,
|
|
errorCode, errorText, networkType, responseTime);
|
|
}
|
|
if (errorText != nullptr) {
|
|
jniEnv[instanceNum]->DeleteLocalRef(errorText);
|
|
}
|
|
}), ([onQuickAck, instanceNum] {
|
|
if (onQuickAck != nullptr) {
|
|
jniEnv[instanceNum]->CallVoidMethod(onQuickAck, jclass_QuickAckDelegate_run);
|
|
}
|
|
}), ([onWriteToSocket, instanceNum] {
|
|
if (onWriteToSocket != nullptr) {
|
|
jniEnv[instanceNum]->CallVoidMethod(onWriteToSocket, jclass_WriteToSocketDelegate_run);
|
|
}
|
|
}), (uint32_t) flags, (uint32_t) datacenterId, (ConnectionType) connetionType, immediate, token,
|
|
onComplete, onQuickAck,
|
|
onWriteToSocket);
|
|
}
|
|
|
|
void cancelRequest(JNIEnv *env, jclass c, jint instanceNum, jint token, jboolean notifyServer) {
|
|
return ConnectionsManager::getInstance(instanceNum).cancelRequest(token, notifyServer);
|
|
}
|
|
|
|
void cleanUp(JNIEnv *env, jclass c, jint instanceNum, jboolean resetKeys) {
|
|
return ConnectionsManager::getInstance(instanceNum).cleanUp(resetKeys, -1);
|
|
}
|
|
|
|
void cancelRequestsForGuid(JNIEnv *env, jclass c, jint instanceNum, jint guid) {
|
|
return ConnectionsManager::getInstance(instanceNum).cancelRequestsForGuid(guid);
|
|
}
|
|
|
|
void bindRequestToGuid(JNIEnv *env, jclass c, jint instanceNum, jint requestToken, jint guid) {
|
|
return ConnectionsManager::getInstance(instanceNum).bindRequestToGuid(requestToken, guid);
|
|
}
|
|
|
|
void applyDatacenterAddress(JNIEnv *env, jclass c, jint instanceNum, jint datacenterId,
|
|
jstring ipAddress, jint port) {
|
|
const char *valueStr = env->GetStringUTFChars(ipAddress, 0);
|
|
|
|
ConnectionsManager::getInstance(instanceNum).applyDatacenterAddress((uint32_t) datacenterId,
|
|
std::string(valueStr),
|
|
(uint32_t) port);
|
|
|
|
if (valueStr != 0) {
|
|
env->ReleaseStringUTFChars(ipAddress, valueStr);
|
|
}
|
|
}
|
|
|
|
void moveToDatacenter(JNIEnv *env, jclass c, jint instanceNum, jint datacenterId) {
|
|
|
|
ConnectionsManager::getInstance(instanceNum).moveToDatacenter((uint32_t) datacenterId);
|
|
|
|
}
|
|
|
|
void setProxySettings(JNIEnv *env, jclass c, jint instanceNum, jstring address, jint port,
|
|
jstring username, jstring password, jstring secret) {
|
|
const char *addressStr = env->GetStringUTFChars(address, 0);
|
|
const char *usernameStr = env->GetStringUTFChars(username, 0);
|
|
const char *passwordStr = env->GetStringUTFChars(password, 0);
|
|
const char *secretStr = env->GetStringUTFChars(secret, 0);
|
|
|
|
ConnectionsManager::getInstance(instanceNum).setProxySettings(addressStr, (uint16_t) port,
|
|
usernameStr, passwordStr,
|
|
secretStr);
|
|
|
|
if (addressStr != 0) {
|
|
env->ReleaseStringUTFChars(address, addressStr);
|
|
}
|
|
if (usernameStr != 0) {
|
|
env->ReleaseStringUTFChars(username, usernameStr);
|
|
}
|
|
if (passwordStr != 0) {
|
|
env->ReleaseStringUTFChars(password, passwordStr);
|
|
}
|
|
if (secretStr != 0) {
|
|
env->ReleaseStringUTFChars(secret, secretStr);
|
|
}
|
|
}
|
|
|
|
jint getConnectionState(JNIEnv *env, jclass c, jint instanceNum) {
|
|
return ConnectionsManager::getInstance(instanceNum).getConnectionState();
|
|
}
|
|
|
|
void setUserId(JNIEnv *env, jclass c, jint instanceNum, int32_t id) {
|
|
ConnectionsManager::getInstance(instanceNum).setUserId(id);
|
|
}
|
|
|
|
void switchBackend(JNIEnv *env, jclass c, jint instanceNum) {
|
|
ConnectionsManager::getInstance(instanceNum).switchBackend();
|
|
}
|
|
|
|
void pauseNetwork(JNIEnv *env, jclass c, jint instanceNum) {
|
|
ConnectionsManager::getInstance(instanceNum).pauseNetwork();
|
|
}
|
|
|
|
void resumeNetwork(JNIEnv *env, jclass c, jint instanceNum, jboolean partial) {
|
|
ConnectionsManager::getInstance(instanceNum).resumeNetwork(partial);
|
|
}
|
|
|
|
void updateDcSettings(JNIEnv *env, jclass c, jint instanceNum) {
|
|
ConnectionsManager::getInstance(instanceNum).updateDcSettings(0, false);
|
|
}
|
|
|
|
void setIpStrategy(JNIEnv *env, jclass c, jint instanceNum, jbyte value) {
|
|
ConnectionsManager::getInstance(instanceNum).setIpStrategy((uint8_t) value);
|
|
}
|
|
|
|
void setNetworkAvailable(JNIEnv *env, jclass c, jint instanceNum, jboolean value, jint networkType,
|
|
jboolean slow) {
|
|
ConnectionsManager::getInstance(instanceNum).setNetworkAvailable(value, networkType, slow);
|
|
}
|
|
|
|
void setPushConnectionEnabled(JNIEnv *env, jclass c, jint instanceNum, jboolean value) {
|
|
ConnectionsManager::getInstance(instanceNum).setPushConnectionEnabled(value);
|
|
}
|
|
|
|
void
|
|
applyDnsConfig(JNIEnv *env, jclass c, jint instanceNum, jlong address, jstring phone, jint date) {
|
|
const char *phoneStr = env->GetStringUTFChars(phone, 0);
|
|
|
|
ConnectionsManager::getInstance(instanceNum).applyDnsConfig(
|
|
(NativeByteBuffer *) (intptr_t) address, phoneStr, date);
|
|
if (phoneStr != 0) {
|
|
env->ReleaseStringUTFChars(phone, phoneStr);
|
|
}
|
|
}
|
|
|
|
jlong
|
|
checkProxy(JNIEnv *env, jclass c, jint instanceNum, jstring address, jint port, jstring username,
|
|
jstring password, jstring secret, jobject requestTimeFunc) {
|
|
const char *addressStr = env->GetStringUTFChars(address, 0);
|
|
const char *usernameStr = env->GetStringUTFChars(username, 0);
|
|
const char *passwordStr = env->GetStringUTFChars(password, 0);
|
|
const char *secretStr = env->GetStringUTFChars(secret, 0);
|
|
|
|
if (requestTimeFunc != nullptr) {
|
|
requestTimeFunc = env->NewGlobalRef(requestTimeFunc);
|
|
}
|
|
|
|
jlong result = ConnectionsManager::getInstance(instanceNum).checkProxy(addressStr,
|
|
(uint16_t) port,
|
|
usernameStr, passwordStr,
|
|
secretStr,
|
|
[instanceNum, requestTimeFunc](
|
|
int64_t time) {
|
|
if (requestTimeFunc !=
|
|
nullptr) {
|
|
jniEnv[instanceNum]->CallVoidMethod(
|
|
requestTimeFunc,
|
|
jclass_RequestTimeDelegate_run,
|
|
time);
|
|
}
|
|
}, requestTimeFunc);
|
|
|
|
if (addressStr != 0) {
|
|
env->ReleaseStringUTFChars(address, addressStr);
|
|
}
|
|
if (usernameStr != 0) {
|
|
env->ReleaseStringUTFChars(username, usernameStr);
|
|
}
|
|
if (passwordStr != 0) {
|
|
env->ReleaseStringUTFChars(password, passwordStr);
|
|
}
|
|
if (secretStr != 0) {
|
|
env->ReleaseStringUTFChars(secret, secretStr);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
class Delegate : public ConnectiosManagerDelegate {
|
|
|
|
void onUpdate(int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onUpdate, instanceNum);
|
|
}
|
|
|
|
void onSessionCreated(int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onSessionCreated,
|
|
instanceNum);
|
|
}
|
|
|
|
void onConnectionStateChanged(ConnectionState state, int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onConnectionStateChanged,
|
|
state, instanceNum);
|
|
}
|
|
|
|
void onUnparsedMessageReceived(int64_t reqMessageId, NativeByteBuffer *buffer,
|
|
ConnectionType connectionType, int32_t instanceNum) {
|
|
if (connectionType == ConnectionTypeGeneric) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onUnparsedMessageReceived,
|
|
(jlong) (intptr_t) buffer, instanceNum);
|
|
}
|
|
}
|
|
|
|
void onLogout(int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onLogout, instanceNum);
|
|
}
|
|
|
|
void onUpdateConfig(TL_config *config, int32_t instanceNum) {
|
|
NativeByteBuffer *buffer = BuffersStorage::getInstance().getFreeBuffer(
|
|
config->getObjectSize());
|
|
config->serializeToStream(buffer);
|
|
buffer->position(0);
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onUpdateConfig,
|
|
(jlong) (intptr_t) buffer, instanceNum);
|
|
buffer->reuse();
|
|
}
|
|
|
|
void onInternalPushReceived(int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onInternalPushReceived,
|
|
instanceNum);
|
|
}
|
|
|
|
void onBytesReceived(int32_t amount, int32_t networkType, int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onBytesReceived, amount,
|
|
networkType, instanceNum);
|
|
}
|
|
|
|
void onBytesSent(int32_t amount, int32_t networkType, int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onBytesSent, amount,
|
|
networkType, instanceNum);
|
|
}
|
|
|
|
void onRequestNewServerIpAndPort(int32_t second, int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onRequestNewServerIpAndPort,
|
|
second, instanceNum);
|
|
}
|
|
|
|
void onProxyError(int32_t instanceNum) {
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_onProxyError,
|
|
instanceNum);
|
|
}
|
|
|
|
void getHostByName(std::string domain, int32_t instanceNum, ConnectionSocket *socket) {
|
|
jstring domainName = jniEnv[instanceNum]->NewStringUTF(domain.c_str());
|
|
jniEnv[instanceNum]->CallStaticVoidMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_getHostByName,
|
|
domainName, (jlong) (intptr_t) socket);
|
|
jniEnv[instanceNum]->DeleteLocalRef(domainName);
|
|
}
|
|
|
|
int32_t getInitFlags(int32_t instanceNum) {
|
|
return (int32_t) jniEnv[instanceNum]->CallStaticIntMethod(jclass_ConnectionsManager,
|
|
jclass_ConnectionsManager_getInitFlags);
|
|
}
|
|
};
|
|
|
|
void
|
|
onHostNameResolved(JNIEnv *env, jclass c, jstring host, jlong address, jstring ip, jboolean ipv6) {
|
|
const char *ipStr = env->GetStringUTFChars(ip, 0);
|
|
const char *hostStr = env->GetStringUTFChars(host, 0);
|
|
std::string i = std::string(ipStr);
|
|
std::string h = std::string(hostStr);
|
|
if (ipStr != 0) {
|
|
env->ReleaseStringUTFChars(ip, ipStr);
|
|
}
|
|
if (hostStr != 0) {
|
|
env->ReleaseStringUTFChars(host, hostStr);
|
|
}
|
|
ConnectionSocket *socket = (ConnectionSocket *) (intptr_t) address;
|
|
socket->onHostNameResolved(h, i, ipv6);
|
|
}
|
|
|
|
void setLangCode(JNIEnv *env, jclass c, jint instanceNum, jstring langCode) {
|
|
const char *langCodeStr = env->GetStringUTFChars(langCode, 0);
|
|
|
|
ConnectionsManager::getInstance(instanceNum).setLangCode(std::string(langCodeStr));
|
|
|
|
if (langCodeStr != 0) {
|
|
env->ReleaseStringUTFChars(langCode, langCodeStr);
|
|
}
|
|
}
|
|
|
|
void setRegId(JNIEnv *env, jclass c, jint instanceNum, jstring regId) {
|
|
const char *regIdStr = env->GetStringUTFChars(regId, 0);
|
|
|
|
ConnectionsManager::getInstance(instanceNum).setRegId(std::string(regIdStr));
|
|
|
|
if (regIdStr != 0) {
|
|
env->ReleaseStringUTFChars(regId, regIdStr);
|
|
}
|
|
}
|
|
|
|
void setSystemLangCode(JNIEnv *env, jclass c, jint instanceNum, jstring langCode) {
|
|
const char *langCodeStr = env->GetStringUTFChars(langCode, 0);
|
|
|
|
ConnectionsManager::getInstance(instanceNum).setSystemLangCode(std::string(langCodeStr));
|
|
|
|
if (langCodeStr != 0) {
|
|
env->ReleaseStringUTFChars(langCode, langCodeStr);
|
|
}
|
|
}
|
|
|
|
void init(JNIEnv *env, jclass c, jint instanceNum, jint version, jint layer, jint apiId,
|
|
jstring deviceModel, jstring systemVersion, jstring appVersion, jstring langCode,
|
|
jstring systemLangCode, jstring configPath, jstring logPath, jstring regId,
|
|
jstring cFingerprint, jstring installerId, jint timezoneOffset, jint userId,
|
|
jboolean enablePushConnection, jboolean hasNetwork, jint networkType) {
|
|
const char *deviceModelStr = env->GetStringUTFChars(deviceModel, 0);
|
|
const char *systemVersionStr = env->GetStringUTFChars(systemVersion, 0);
|
|
const char *appVersionStr = env->GetStringUTFChars(appVersion, 0);
|
|
const char *langCodeStr = env->GetStringUTFChars(langCode, 0);
|
|
const char *systemLangCodeStr = env->GetStringUTFChars(systemLangCode, 0);
|
|
const char *configPathStr = env->GetStringUTFChars(configPath, 0);
|
|
const char *logPathStr = env->GetStringUTFChars(logPath, 0);
|
|
const char *regIdStr = env->GetStringUTFChars(regId, 0);
|
|
const char *cFingerprintStr = env->GetStringUTFChars(cFingerprint, 0);
|
|
const char *installerIdStr = env->GetStringUTFChars(installerId, 0);
|
|
|
|
ConnectionsManager::getInstance(instanceNum).init((uint32_t) version, layer, apiId,
|
|
std::string(deviceModelStr),
|
|
std::string(systemVersionStr),
|
|
std::string(appVersionStr),
|
|
std::string(langCodeStr),
|
|
std::string(systemLangCodeStr),
|
|
std::string(configPathStr),
|
|
std::string(logPathStr),
|
|
std::string(regIdStr),
|
|
std::string(cFingerprintStr),
|
|
std::string(installerIdStr), timezoneOffset,
|
|
userId, true, enablePushConnection,
|
|
hasNetwork, networkType);
|
|
|
|
if (deviceModelStr != 0) {
|
|
env->ReleaseStringUTFChars(deviceModel, deviceModelStr);
|
|
}
|
|
if (systemVersionStr != 0) {
|
|
env->ReleaseStringUTFChars(systemVersion, systemVersionStr);
|
|
}
|
|
if (appVersionStr != 0) {
|
|
env->ReleaseStringUTFChars(appVersion, appVersionStr);
|
|
}
|
|
if (langCodeStr != 0) {
|
|
env->ReleaseStringUTFChars(langCode, langCodeStr);
|
|
}
|
|
if (systemLangCodeStr != 0) {
|
|
env->ReleaseStringUTFChars(systemLangCode, systemLangCodeStr);
|
|
}
|
|
if (configPathStr != 0) {
|
|
env->ReleaseStringUTFChars(configPath, configPathStr);
|
|
}
|
|
if (logPathStr != 0) {
|
|
env->ReleaseStringUTFChars(logPath, logPathStr);
|
|
}
|
|
if (regIdStr != 0) {
|
|
env->ReleaseStringUTFChars(regId, regIdStr);
|
|
}
|
|
if (cFingerprintStr != 0) {
|
|
env->ReleaseStringUTFChars(cFingerprint, cFingerprintStr);
|
|
}
|
|
if (installerIdStr != 0) {
|
|
env->ReleaseStringUTFChars(installerId, installerIdStr);
|
|
}
|
|
}
|
|
|
|
void setJava(JNIEnv *env, jclass c, jboolean useJavaByteBuffers) {
|
|
ConnectionsManager::useJavaVM(java, useJavaByteBuffers);
|
|
}
|
|
|
|
void setJava1(JNIEnv *env, jclass c, jint instanceNum) {
|
|
if (instanceNum >= jniEnv.capacity()) {
|
|
jniEnv.resize(instanceNum + 10, nullptr);
|
|
}
|
|
ConnectionsManager::getInstance(instanceNum).setDelegate(new Delegate());
|
|
}
|
|
|
|
static const char *ConnectionsManagerClassPathName = "org/telegram/tgnet/ConnectionsManager";
|
|
static JNINativeMethod ConnectionsManagerMethods[] = {
|
|
{"native_getCurrentTimeMillis", "(I)J", (void *) getCurrentTimeMillis},
|
|
{"native_getCurrentTime", "(I)I", (void *) getCurrentTime},
|
|
{"native_getCurrentDatacenterId", "(I)I", (void *) getCurrentDatacenterId},
|
|
{"native_isTestBackend", "(I)I", (void *) isTestBackend},
|
|
{"native_getTimeDifference", "(I)I", (void *) getTimeDifference},
|
|
{"native_sendRequest", "(IJLorg/telegram/tgnet/RequestDelegateInternal;Lorg/telegram/tgnet/QuickAckDelegate;Lorg/telegram/tgnet/WriteToSocketDelegate;IIIZI)V", (void *) sendRequest},
|
|
{"native_cancelRequest", "(IIZ)V", (void *) cancelRequest},
|
|
{"native_cleanUp", "(IZ)V", (void *) cleanUp},
|
|
{"native_cancelRequestsForGuid", "(II)V", (void *) cancelRequestsForGuid},
|
|
{"native_bindRequestToGuid", "(III)V", (void *) bindRequestToGuid},
|
|
{"native_applyDatacenterAddress", "(IILjava/lang/String;I)V", (void *) applyDatacenterAddress},
|
|
{"native_moveToDatacenter", "(II)V", (void *) moveToDatacenter},
|
|
{"native_setProxySettings", "(ILjava/lang/String;ILjava/lang/String;Ljava/lang/String;Ljava/lang/String;)V", (void *) setProxySettings},
|
|
{"native_getConnectionState", "(I)I", (void *) getConnectionState},
|
|
{"native_setUserId", "(II)V", (void *) setUserId},
|
|
{"native_init", "(IIIILjava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;IIZZI)V", (void *) init},
|
|
{"native_setLangCode", "(ILjava/lang/String;)V", (void *) setLangCode},
|
|
{"native_setRegId", "(ILjava/lang/String;)V", (void *) setRegId},
|
|
{"native_setSystemLangCode", "(ILjava/lang/String;)V", (void *) setSystemLangCode},
|
|
{"native_switchBackend", "(I)V", (void *) switchBackend},
|
|
{"native_pauseNetwork", "(I)V", (void *) pauseNetwork},
|
|
{"native_resumeNetwork", "(IZ)V", (void *) resumeNetwork},
|
|
{"native_updateDcSettings", "(I)V", (void *) updateDcSettings},
|
|
{"native_setIpStrategy", "(IB)V", (void *) setIpStrategy},
|
|
{"native_setNetworkAvailable", "(IZIZ)V", (void *) setNetworkAvailable},
|
|
{"native_setPushConnectionEnabled", "(IZ)V", (void *) setPushConnectionEnabled},
|
|
{"native_setJava", "(Z)V", (void *) setJava},
|
|
{"native_setJava", "(I)V", (void *) setJava1},
|
|
{"native_applyDnsConfig", "(IJLjava/lang/String;I)V", (void *) applyDnsConfig},
|
|
{"native_checkProxy", "(ILjava/lang/String;ILjava/lang/String;Ljava/lang/String;Ljava/lang/String;Lorg/telegram/tgnet/RequestTimeDelegate;)J", (void *) checkProxy},
|
|
{"native_onHostNameResolved", "(Ljava/lang/String;JLjava/lang/String;Z)V", (void *) onHostNameResolved}
|
|
};
|
|
|
|
inline int registerNativeMethods(JNIEnv *env, const char *className, JNINativeMethod *methods,
|
|
int methodsCount) {
|
|
jclass clazz;
|
|
clazz = env->FindClass(className);
|
|
if (clazz == NULL) {
|
|
return JNI_FALSE;
|
|
}
|
|
if (env->RegisterNatives(clazz, methods, methodsCount) < 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
return JNI_TRUE;
|
|
}
|
|
|
|
extern "C" int registerNativeTgNetFunctions(JavaVM *vm, JNIEnv *env) {
|
|
java = vm;
|
|
|
|
if (!registerNativeMethods(env, NativeByteBufferClassPathName, NativeByteBufferMethods,
|
|
sizeof(NativeByteBufferMethods) /
|
|
sizeof(NativeByteBufferMethods[0]))) {
|
|
return JNI_FALSE;
|
|
}
|
|
|
|
if (!registerNativeMethods(env, ConnectionsManagerClassPathName, ConnectionsManagerMethods,
|
|
sizeof(ConnectionsManagerMethods) /
|
|
sizeof(ConnectionsManagerMethods[0]))) {
|
|
return JNI_FALSE;
|
|
}
|
|
|
|
jclass_RequestDelegateInternal = (jclass) env->NewGlobalRef(
|
|
env->FindClass("org/telegram/tgnet/RequestDelegateInternal"));
|
|
if (jclass_RequestDelegateInternal == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_RequestDelegateInternal_run = env->GetMethodID(jclass_RequestDelegateInternal, "run",
|
|
"(JILjava/lang/String;IJ)V");
|
|
if (jclass_RequestDelegateInternal_run == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
|
|
jclass_RequestTimeDelegate = (jclass) env->NewGlobalRef(
|
|
env->FindClass("org/telegram/tgnet/RequestTimeDelegate"));
|
|
if (jclass_RequestTimeDelegate == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_RequestTimeDelegate_run = env->GetMethodID(jclass_RequestTimeDelegate, "run", "(J)V");
|
|
if (jclass_RequestTimeDelegate_run == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
|
|
jclass_QuickAckDelegate = (jclass) env->NewGlobalRef(
|
|
env->FindClass("org/telegram/tgnet/QuickAckDelegate"));
|
|
if (jclass_RequestDelegateInternal == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_QuickAckDelegate_run = env->GetMethodID(jclass_QuickAckDelegate, "run", "()V");
|
|
if (jclass_QuickAckDelegate_run == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
|
|
jclass_WriteToSocketDelegate = (jclass) env->NewGlobalRef(
|
|
env->FindClass("org/telegram/tgnet/WriteToSocketDelegate"));
|
|
if (jclass_WriteToSocketDelegate == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_WriteToSocketDelegate_run = env->GetMethodID(jclass_WriteToSocketDelegate, "run", "()V");
|
|
if (jclass_WriteToSocketDelegate_run == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager = (jclass) env->NewGlobalRef(
|
|
env->FindClass("org/telegram/tgnet/ConnectionsManager"));
|
|
if (jclass_ConnectionsManager == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onUnparsedMessageReceived = env->GetStaticMethodID(
|
|
jclass_ConnectionsManager, "onUnparsedMessageReceived", "(JI)V");
|
|
if (jclass_ConnectionsManager_onUnparsedMessageReceived == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onUpdate = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"onUpdate", "(I)V");
|
|
if (jclass_ConnectionsManager_onUpdate == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onSessionCreated = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"onSessionCreated", "(I)V");
|
|
if (jclass_ConnectionsManager_onSessionCreated == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onLogout = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"onLogout", "(I)V");
|
|
if (jclass_ConnectionsManager_onLogout == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onConnectionStateChanged = env->GetStaticMethodID(
|
|
jclass_ConnectionsManager, "onConnectionStateChanged", "(II)V");
|
|
if (jclass_ConnectionsManager_onConnectionStateChanged == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onInternalPushReceived = env->GetStaticMethodID(
|
|
jclass_ConnectionsManager, "onInternalPushReceived", "(I)V");
|
|
if (jclass_ConnectionsManager_onInternalPushReceived == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onUpdateConfig = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"onUpdateConfig", "(JI)V");
|
|
if (jclass_ConnectionsManager_onUpdateConfig == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onBytesSent = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"onBytesSent", "(III)V");
|
|
if (jclass_ConnectionsManager_onBytesSent == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onBytesReceived = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"onBytesReceived", "(III)V");
|
|
if (jclass_ConnectionsManager_onBytesReceived == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onRequestNewServerIpAndPort = env->GetStaticMethodID(
|
|
jclass_ConnectionsManager, "onRequestNewServerIpAndPort", "(II)V");
|
|
if (jclass_ConnectionsManager_onRequestNewServerIpAndPort == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_onProxyError = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"onProxyError", "(I)V");
|
|
if (jclass_ConnectionsManager_onProxyError == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_getHostByName = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"getHostByName",
|
|
"(Ljava/lang/String;J)V");
|
|
if (jclass_ConnectionsManager_getHostByName == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
jclass_ConnectionsManager_getInitFlags = env->GetStaticMethodID(jclass_ConnectionsManager,
|
|
"getInitFlags", "()I");
|
|
if (jclass_ConnectionsManager_getInitFlags == 0) {
|
|
return JNI_FALSE;
|
|
}
|
|
|
|
return JNI_TRUE;
|
|
}
|
|
|
|
//
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018
|
|
//
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
//
|
|
|
|
bool check_utf8(const char *data, size_t len) {
|
|
const char *data_end = data + len;
|
|
do {
|
|
unsigned int a = (unsigned char) (*data++);
|
|
if ((a & 0x80) == 0) {
|
|
if (data == data_end + 1) {
|
|
return true;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
#define ENSURE(condition) \
|
|
if (!(condition)) { \
|
|
return false; \
|
|
}
|
|
|
|
ENSURE((a & 0x40) != 0);
|
|
|
|
unsigned int b = (unsigned char) (*data++);
|
|
ENSURE((b & 0xc0) == 0x80);
|
|
if ((a & 0x20) == 0) {
|
|
ENSURE((a & 0x1e) > 0);
|
|
continue;
|
|
}
|
|
|
|
unsigned int c = (unsigned char) (*data++);
|
|
ENSURE((c & 0xc0) == 0x80);
|
|
if ((a & 0x10) == 0) {
|
|
int x = (((a & 0x0f) << 6) | (b & 0x20));
|
|
ENSURE(x != 0 && x != 0x360);
|
|
continue;
|
|
}
|
|
|
|
unsigned int d = (unsigned char) (*data++);
|
|
ENSURE((d & 0xc0) == 0x80);
|
|
if ((a & 0x08) == 0) {
|
|
int t = (((a & 0x07) << 6) | (b & 0x30));
|
|
ENSURE(0 < t && t < 0x110);
|
|
continue;
|
|
}
|
|
|
|
return false;
|
|
#undef ENSURE
|
|
} while (1);
|
|
}
|