| /* |
| * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include <android/log.h> |
| #include <stdio.h> |
| #include <string.h> |
| |
| #include "webrtc/video_engine/test/android/jni/org_webrtc_videoengineapp_vie_android_java_api.h" |
| |
| #include "webrtc/voice_engine/include/voe_audio_processing.h" |
| #include "webrtc/voice_engine/include/voe_base.h" |
| #include "webrtc/voice_engine/include/voe_codec.h" |
| #include "webrtc/voice_engine/include/voe_file.h" |
| #include "webrtc/voice_engine/include/voe_hardware.h" |
| #include "webrtc/voice_engine/include/voe_network.h" |
| #include "webrtc/voice_engine/include/voe_rtp_rtcp.h" |
| #include "webrtc/voice_engine/include/voe_volume_control.h" |
| |
| #include "webrtc/video_engine/include/vie_base.h" |
| #include "webrtc/video_engine/include/vie_capture.h" |
| #include "webrtc/video_engine/include/vie_codec.h" |
| #include "webrtc/video_engine/include/vie_external_codec.h" |
| #include "webrtc/video_engine/include/vie_network.h" |
| #include "webrtc/video_engine/include/vie_render.h" |
| #include "webrtc/video_engine/include/vie_rtp_rtcp.h" |
| |
| #include "webrtc/common_types.h" |
| #include "webrtc/video_engine/test/android/jni/android_media_codec_decoder.h" |
| |
| #include "webrtc/system_wrappers/interface/scoped_ptr.h" |
| #include "webrtc/test/channel_transport/include/channel_transport.h" |
| |
| #define WEBRTC_LOG_TAG "*WEBRTCN*" |
| #define VALIDATE_BASE_POINTER \ |
| if (!voeData.base) \ |
| { \ |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "Base pointer doesn't exist"); \ |
| return -1; \ |
| } |
| #define VALIDATE_CODEC_POINTER \ |
| if (!voeData.codec) \ |
| { \ |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "Codec pointer doesn't exist"); \ |
| return -1; \ |
| } |
| #define VALIDATE_FILE_POINTER \ |
| if (!voeData.file) \ |
| { \ |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "File pointer doesn't exist"); \ |
| return -1; \ |
| } |
| #define VALIDATE_APM_POINTER \ |
| if (!voeData.codec) \ |
| { \ |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "Apm pointer doesn't exist"); \ |
| return -1; \ |
| } |
| #define VALIDATE_HARDWARE_POINTER \ |
| if (!voeData.hardware) \ |
| { \ |
| __android_log_write( \ |
| ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "Hardware pointer doesn't exist"); \ |
| return -1; \ |
| } |
| #define VALIDATE_VOLUME_POINTER \ |
| if (!voeData.volume) \ |
| { \ |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "Volume pointer doesn't exist"); \ |
| return -1; \ |
| } |
| |
| #define VALIDATE_RTP_POINTER \ |
| if (!voeData.rtp) \ |
| { \ |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "rtp pointer doesn't exist"); \ |
| return -1; \ |
| } |
| |
| using namespace webrtc; |
| |
| //Forward declaration. |
| class VideoCallbackAndroid; |
| |
| // VoiceEngine data struct |
| typedef struct |
| { |
| // VoiceEngine |
| VoiceEngine* ve; |
| // Sub-APIs |
| VoEBase* base; |
| VoECodec* codec; |
| VoEFile* file; |
| VoENetwork* netw; |
| VoEAudioProcessing* apm; |
| VoEVolumeControl* volume; |
| VoEHardware* hardware; |
| VoERTP_RTCP* rtp; |
| |
| JavaVM* jvm; |
| scoped_ptr<test::VoiceChannelTransport> transport; |
| } VoiceEngineData; |
| |
| class AndroidVideoRenderCallback; |
| // VideoEngine data struct |
| typedef struct |
| { |
| VideoEngine* vie; |
| ViEBase* base; |
| ViECodec* codec; |
| ViENetwork* netw; |
| ViERTP_RTCP* rtp; |
| ViERender* render; |
| ViECapture* capture; |
| ViEExternalCodec* externalCodec; |
| |
| VideoCallbackAndroid* callback; |
| scoped_ptr<test::VideoChannelTransport> transport; |
| } VideoEngineData; |
| |
| // Global variables |
| JavaVM* webrtcGlobalVM; |
| |
| // Global variables visible in this file |
| static VoiceEngineData voeData; |
| static VideoEngineData vieData; |
| |
| // "Local" functions (i.e. not Java accessible) |
| #define WEBRTC_TRACE_MAX_MESSAGE_SIZE 1024 |
| static bool VE_GetSubApis(); |
| static bool VE_ReleaseSubApis(); |
| |
| #define CHECK_API_RETURN(ret) \ |
| if (ret!=0) \ |
| { \ |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, \ |
| "Return error %d",ret); \ |
| break; \ |
| } |
| |
| class VideoCallbackAndroid: public ViEDecoderObserver, |
| public ViEEncoderObserver |
| { |
| |
| // Implements ViEDecoderObserver |
| virtual void IncomingRate(const int videoChannel, |
| const unsigned int framerate, |
| const unsigned int bitrate) |
| { |
| // Let's print out the network statistics from this call back as well |
| unsigned short fraction_lost; |
| unsigned int dummy; |
| int intdummy; |
| _vieData.rtp->GetReceivedRTCPStatistics(videoChannel, fraction_lost, |
| dummy, dummy, dummy, intdummy); |
| unsigned short packetLossRate = 0; |
| if (fraction_lost > 0) |
| { |
| // Change from frac to % |
| packetLossRate = (fraction_lost * 100) >> 8; |
| } |
| |
| JNIEnv* threadEnv = NULL; |
| int ret = webrtcGlobalVM->AttachCurrentThread(&threadEnv, NULL); |
| // Get the JNI env for this thread |
| if ((ret < 0) || !threadEnv) |
| { |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "Could not attach thread to JVM (%d, %p)", ret, |
| threadEnv); |
| return; |
| } |
| threadEnv->CallIntMethod(_callbackObj, _callbackId, framerate, bitrate, |
| packetLossRate, _frameRateO, _bitRateO); |
| webrtcGlobalVM->DetachCurrentThread(); |
| } |
| ; |
| |
| virtual void IncomingCodecChanged(const int videoChannel, |
| const webrtc::VideoCodec& videoCodec) |
| { |
| } |
| ; |
| |
| virtual void RequestNewKeyFrame(const int videoChannel) |
| { |
| } |
| ; |
| |
| virtual void OutgoingRate(const int videoChannel, |
| const unsigned int framerate, |
| const unsigned int bitrate) |
| { |
| _frameRateO = framerate; |
| _bitRateO = bitrate; |
| //__android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| // "SendRate frameRate %d bitrate %d\n",frameRate,bitrate); |
| } |
| ; |
| |
| public: |
| VideoEngineData& _vieData; |
| JNIEnv * _env; |
| jobject _callbackObj; |
| jclass _callbackCls; |
| jmethodID _callbackId; |
| int _frameRateO, _bitRateO; |
| VideoCallbackAndroid(VideoEngineData& vieData, JNIEnv * env, |
| jobject callback) : |
| _vieData(vieData), _env(env), _callbackObj(callback), |
| _frameRateO(0), _bitRateO(0) { |
| _callbackCls = _env->GetObjectClass(_callbackObj); |
| _callbackId |
| = _env->GetMethodID(_callbackCls, "updateStats", "(IIIII)I"); |
| if (_callbackId == NULL) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to get jid"); |
| } |
| _callbackObj = _env->NewGlobalRef(_callbackObj); |
| } |
| }; |
| |
| // JNI_OnLoad |
| jint JNI_OnLoad(JavaVM* vm, void* reserved) { |
| webrtcGlobalVM = vm; |
| if (!webrtcGlobalVM) |
| { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "JNI_OnLoad did not receive a valid VM pointer"); |
| return -1; |
| } |
| |
| // Get JNI |
| JNIEnv* env; |
| if (JNI_OK != vm->GetEnv(reinterpret_cast<void**> (&env), |
| JNI_VERSION_1_4)) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "JNI_OnLoad could not get JNI env"); |
| return -1; |
| } |
| |
| // Init VoiceEngine data |
| memset(&voeData, 0, sizeof(voeData)); |
| // Store the JVM |
| voeData.jvm = vm; |
| |
| // Init VideoEngineData data |
| memset(&vieData, 0, sizeof(vieData)); |
| |
| return JNI_VERSION_1_4; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: NativeInit |
| * Signature: (Landroid/content/Context;)Z |
| */ |
| JNIEXPORT jboolean JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_NativeInit( |
| JNIEnv * env, |
| jobject, |
| jobject context) |
| { |
| return true; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: GetVideoEngine |
| * Signature: ()I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_GetVideoEngine( |
| JNIEnv *, |
| jobject context) { |
| |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "GetVideoEngine"); |
| |
| // Check if already got |
| if (vieData.vie) { |
| __android_log_write(ANDROID_LOG_INFO, WEBRTC_LOG_TAG, |
| "ViE already got"); |
| return 0; |
| } |
| |
| VideoEngine::SetAndroidObjects(webrtcGlobalVM, context); |
| |
| // Create |
| vieData.vie = VideoEngine::Create(); |
| if (!vieData.vie) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, "Get ViE failed"); |
| return -1; |
| } |
| vieData.base = ViEBase::GetInterface(vieData.vie); |
| if (!vieData.base) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get base sub-API failed"); |
| return -1; |
| } |
| |
| vieData.codec = ViECodec::GetInterface(vieData.vie); |
| if (!vieData.codec) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get codec sub-API failed"); |
| return -1; |
| } |
| |
| vieData.netw = ViENetwork::GetInterface(vieData.vie); |
| if (!vieData.netw) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get network sub-API failed"); |
| return -1; |
| } |
| |
| vieData.rtp = ViERTP_RTCP::GetInterface(vieData.vie); |
| if (!vieData.rtp) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get RTP sub-API failed"); |
| return -1; |
| } |
| |
| vieData.render = ViERender::GetInterface(vieData.vie); |
| if (!vieData.render) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get Render sub-API failed"); |
| return -1; |
| } |
| |
| vieData.capture = ViECapture::GetInterface(vieData.vie); |
| if (!vieData.capture) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get Capture sub-API failed"); |
| return -1; |
| } |
| |
| vieData.externalCodec = ViEExternalCodec::GetInterface(vieData.vie); |
| if (!vieData.capture) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get External Codec sub-API failed"); |
| return -1; |
| } |
| |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: Init |
| * Signature: (Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_Init( |
| JNIEnv *, |
| jobject, |
| jboolean enableTrace) |
| { |
| if (vieData.vie) { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "Init"); |
| |
| int ret = vieData.base->Init(); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "Init return %d", ret); |
| if (enableTrace) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "SetTraceFile"); |
| if (0 != vieData.vie->SetTraceFile(("/sdcard/trace.txt"), false)) |
| { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Video Engine could not enable trace"); |
| } |
| |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "SetTraceFilter"); |
| if (0 != vieData.vie->SetTraceFilter(webrtc::kTraceError)) |
| { |
| __android_log_write(ANDROID_LOG_WARN, WEBRTC_LOG_TAG, |
| "Could not set trace filter"); |
| } |
| } |
| else |
| { |
| if (0 != vieData.vie->SetTraceFilter(webrtc::kTraceNone)) |
| { |
| __android_log_write(ANDROID_LOG_WARN, WEBRTC_LOG_TAG, |
| "Could not set trace filter"); |
| } |
| } |
| if (voeData.ve) // VoiceEngine is enabled |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "SetVoiceEngine"); |
| if (0 != vieData.base->SetVoiceEngine(voeData.ve)) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "SetVoiceEngine failed"); |
| } |
| } |
| return ret; |
| } |
| else |
| { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: Terminate |
| * Signature: ()I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_Terminate( |
| JNIEnv *, |
| jobject) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "Terminate"); |
| |
| if (vieData.vie) { |
| if (!vieData.rtp || vieData.rtp->Release() != 0) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to release RTP sub-API"); |
| } |
| |
| if (!vieData.netw || vieData.netw->Release() != 0) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to release Network sub-API"); |
| } |
| |
| if (!vieData.codec || vieData.codec->Release() != 0) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to release Codec sub-API"); |
| } |
| |
| if (!vieData.render || vieData.render->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to release Render sub-API"); |
| } |
| |
| if (!vieData.capture || vieData.capture->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to release Capture sub-API"); |
| } |
| |
| if (!vieData.base || vieData.base->Release() != 0) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to release Base sub-API"); |
| } |
| |
| if (!vieData.externalCodec || vieData.externalCodec->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to release External Codec sub-API"); |
| } |
| |
| // Delete Vie |
| if (!VideoEngine::Delete(vieData.vie)) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to delete ViE "); |
| return -1; |
| } |
| memset(&vieData, 0, sizeof(vieData)); |
| return 0; |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StartSend |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StartSend( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StartSend"); |
| |
| if (vieData.base) { |
| int ret = vieData.base->StartSend(channel); |
| return ret; |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StopRender |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StopRender( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StopRender"); |
| |
| if (vieData.render) { |
| return vieData.render->StopRender(channel); |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StopSend |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StopSend( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StopSend"); |
| |
| if (vieData.base) { |
| return vieData.base->StopSend(channel); |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StartReceive |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StartReceive( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StartReceive"); |
| |
| if (vieData.base) { |
| return vieData.base->StartReceive(channel); |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StopReceive |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StopReceive( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StopReceive"); |
| if (vieData.base) { |
| return vieData.base->StopReceive(channel); |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: CreateChannel |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_CreateChannel( |
| JNIEnv *, |
| jobject, |
| jint voiceChannel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "CreateChannel"); |
| |
| if (vieData.vie) { |
| int channel = 0; |
| if (vieData.base->CreateChannel(channel) != 0) { |
| return -1; |
| } |
| if (voiceChannel >= 0) { |
| vieData.base->ConnectAudioChannel(channel, voiceChannel); |
| } |
| vieData.transport.reset(new test::VideoChannelTransport(vieData.netw, |
| channel)); |
| return channel; |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetLocalReceiver |
| * Signature: (II)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_SetLocalReceiver( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jint port) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "SetLocalReceiver"); |
| |
| if (vieData.transport.get()) { |
| return vieData.transport->SetLocalReceiver(port); |
| } |
| return -1; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetSendDestination |
| * Signature: (IILjava/lang/String)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_SetSendDestination( |
| JNIEnv * env, |
| jobject, |
| jint channel, |
| jint port, |
| jstring ipaddr) |
| { |
| |
| if (NULL == vieData.vie) |
| return -1; |
| |
| const char* ip = env->GetStringUTFChars(ipaddr, NULL); |
| if (!ip) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Could not get UTF string"); |
| return -1; |
| } |
| |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "SetSendDestination: channel=%d, port=%d, ip=%s\n", |
| channel, port, ip); |
| |
| if (vieData.transport.get()) { |
| return vieData.transport->SetSendDestination(ip, port); |
| } |
| return -1; |
| } |
| |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetReceiveCodec |
| * Signature: (IIIIII)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_SetReceiveCodec( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jint codecNum, |
| jint intbitRate, |
| jint width, |
| jint height, |
| jint frameRate) |
| { |
| if (NULL == vieData.codec) |
| return -1; |
| |
| //Create codec |
| webrtc::VideoCodec codec; |
| vieData.codec->GetCodec(codecNum, codec); |
| |
| __android_log_print( |
| ANDROID_LOG_DEBUG, |
| WEBRTC_LOG_TAG, |
| "SetReceiveCodec %s, pltype=%d, bitRate=%d, maxBitRate=%d," |
| " width=%d, height=%d, frameRate=%d \n", |
| codec.plName, codec.plType, codec.startBitrate, |
| codec.maxBitrate, codec.width, codec.height, |
| codec.maxFramerate); |
| int ret = vieData.codec->SetReceiveCodec(channel, codec); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "SetReceiveCodec return %d", ret); |
| return ret; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetSendCodec |
| * Signature: (IIIIII)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_SetSendCodec( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jint codecNum, |
| jint intbitRate, |
| jint width, |
| jint height, |
| jint frameRate) |
| { |
| if (NULL == vieData.codec) |
| return -1; |
| |
| //Create codec |
| webrtc::VideoCodec codec; |
| vieData.codec->GetCodec(codecNum, codec); |
| codec.startBitrate = intbitRate; |
| codec.maxBitrate = 600; |
| codec.width = width; |
| codec.height = height; |
| codec.maxFramerate = frameRate; |
| |
| for (int i = 0; i < vieData.codec->NumberOfCodecs(); ++i) { |
| webrtc::VideoCodec codecToList; |
| vieData.codec->GetCodec(i, codecToList); |
| __android_log_print( |
| ANDROID_LOG_DEBUG, |
| WEBRTC_LOG_TAG, |
| "Codec list %s, pltype=%d, bitRate=%d, maxBitRate=%d," |
| " width=%d, height=%d, frameRate=%d\n", |
| codecToList.plName, codecToList.plType, |
| codecToList.startBitrate, codecToList.maxBitrate, |
| codecToList.width, codecToList.height, |
| codecToList.maxFramerate); |
| } |
| __android_log_print( |
| ANDROID_LOG_DEBUG, |
| WEBRTC_LOG_TAG, |
| "SetSendCodec %s, pltype=%d, bitRate=%d, maxBitRate=%d, " |
| "width=%d, height=%d, frameRate=%d\n", |
| codec.plName, codec.plType, codec.startBitrate, |
| codec.maxBitrate, codec.width, codec.height, |
| codec.maxFramerate); |
| |
| return vieData.codec->SetSendCodec(channel, codec); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetSendCodec |
| * Signature: ()Z |
| */ |
| JNIEXPORT jobjectArray JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_GetCodecs( |
| JNIEnv *env, |
| jobject) |
| { |
| if (NULL == vieData.codec) { |
| return NULL; |
| } |
| |
| jobjectArray ret; |
| int num = vieData.codec->NumberOfCodecs(); |
| char info[32]; |
| |
| ret = (jobjectArray)env->NewObjectArray( |
| num, |
| env->FindClass("java/lang/String"), |
| env->NewStringUTF("")); |
| |
| for (int i = 0; i < num; ++i) { |
| webrtc::VideoCodec codecToList; |
| vieData.codec->GetCodec(i, codecToList); |
| sprintf(info, "%s pltype:%d", codecToList.plName, codecToList.plType); |
| env->SetObjectArrayElement(ret, i, env->NewStringUTF( info )); |
| |
| __android_log_print( |
| ANDROID_LOG_DEBUG, |
| WEBRTC_LOG_TAG, |
| "Codec[%d] %s, pltype=%d, bitRate=%d, maxBitRate=%d," |
| " width=%d, height=%d, frameRate=%d\n", |
| i, codecToList.plName, codecToList.plType, |
| codecToList.startBitrate, codecToList.maxBitrate, |
| codecToList.width, codecToList.height, |
| codecToList.maxFramerate); |
| } |
| |
| return ret; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: AddRemoteRenderer |
| * Signature: (ILjava/lang/Object;)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_AddRemoteRenderer( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jobject glSurface) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "AddRemoteRenderer"); |
| if (vieData.vie) { |
| return vieData.render->AddRenderer(channel, glSurface, 0, 0, 0, 1, 1); |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: RemoveRemoteRenderer |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_RemoveRemoteRenderer( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "RemoveRemoteRenderer"); |
| |
| if (vieData.vie) { |
| return vieData.render->RemoveRenderer(channel); |
| } |
| else { |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StartRender |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StartRender( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StartRender"); |
| |
| if (vieData.render) { |
| return vieData.render->StartRender(channel); |
| } |
| else { |
| return -1; |
| } |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StartCamera |
| * Signature: (II)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StartCamera( |
| JNIEnv * env, |
| jobject, |
| jint channel, |
| jint cameraNum) |
| { |
| if (NULL == vieData.vie) |
| return -1; |
| |
| int i = 0; |
| char deviceName[64]; |
| char deviceUniqueName[64]; |
| int re; |
| do { |
| re = vieData.capture->GetCaptureDevice(i, deviceName, |
| sizeof(deviceName), |
| deviceUniqueName, |
| sizeof(deviceUniqueName)); |
| __android_log_print( |
| ANDROID_LOG_DEBUG, |
| WEBRTC_LOG_TAG, |
| "GetCaptureDevice ret %d devicenum %d deviceUniqueName %s", |
| re, i, deviceUniqueName); |
| i++; |
| } while (re == 0); |
| |
| int ret; |
| int cameraId; |
| vieData.capture->GetCaptureDevice(cameraNum, deviceName, |
| sizeof(deviceName), deviceUniqueName, |
| sizeof(deviceUniqueName)); |
| vieData.capture->AllocateCaptureDevice(deviceUniqueName, |
| sizeof(deviceUniqueName), cameraId); |
| |
| if (cameraId >= 0) { //Connect the |
| ret = vieData.capture->ConnectCaptureDevice(cameraId, channel); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "ConnectCaptureDevice ret %d ", ret); |
| |
| ret = vieData.capture->StartCapture(cameraId); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "StartCapture ret %d ", ret); |
| } |
| |
| return cameraId; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StopCamera |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StopCamera( |
| JNIEnv *, |
| jobject, |
| jint cameraId) |
| { |
| if (NULL == vieData.capture) |
| return -1; |
| |
| int ret = vieData.capture->StopCapture(cameraId); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "StopCapture ret %d ", ret); |
| ret = vieData.capture->ReleaseCaptureDevice(cameraId); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "ReleaseCaptureDevice ret %d ", ret); |
| |
| return ret; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: GetCameraOrientation |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_GetCameraOrientation( |
| JNIEnv *, |
| jobject, |
| jint cameraNum) |
| { |
| char deviceName[64]; |
| char deviceUniqueName[64]; |
| int ret; |
| |
| ret = vieData.capture->GetCaptureDevice(cameraNum, deviceName, |
| sizeof(deviceName), |
| deviceUniqueName, |
| sizeof(deviceUniqueName)); |
| if (ret != 0) { |
| return -1; |
| } |
| |
| RotateCapturedFrame orientation; |
| ret = vieData.capture->GetOrientation(deviceUniqueName, orientation); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "GetOrientation ret %d orientation %d", ret, |
| orientation); |
| |
| return (jint) orientation; |
| |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetRotation |
| * Signature: (II)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_SetRotation( |
| JNIEnv *, |
| jobject, |
| jint captureId, |
| jint degrees) |
| { |
| |
| if (NULL == vieData.capture) |
| return -1; |
| RotateCapturedFrame rotation = RotateCapturedFrame_0; |
| if (degrees == 90) |
| rotation = RotateCapturedFrame_90; |
| else if (degrees == 180) |
| rotation = RotateCapturedFrame_180; |
| else if (degrees == 270) |
| rotation = RotateCapturedFrame_270; |
| |
| int ret = vieData.capture->SetRotateCapturedFrames(captureId, rotation); |
| return ret; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetExternalMediaCodecDecoderRenderer |
| * Signature: (ILjava/lang/Object;)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_SetExternalMediaCodecDecoderRenderer( |
| JNIEnv *env, |
| jobject, |
| jint channel, |
| jobject glSurface) |
| { |
| __android_log_write( |
| ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "SetExternalMediaCodecDecoder"); |
| |
| jclass cls = env->FindClass("org/webrtc/videoengine/ViEMediaCodecDecoder"); |
| env->NewGlobalRef(cls); |
| |
| AndroidMediaCodecDecoder* mediaCodecDecoder = |
| new AndroidMediaCodecDecoder(webrtcGlobalVM, glSurface, cls); |
| |
| // TODO(dwkang): Check the ownership of decoder object and release it |
| // if needed. |
| return vieData.externalCodec->RegisterExternalReceiveCodec( |
| channel, 120, mediaCodecDecoder, true); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: EnableNACK |
| * Signature: (IZ)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_EnableNACK( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jboolean enable) |
| { |
| if (NULL == vieData.rtp) |
| return -1; |
| |
| int ret = vieData.rtp->SetNACKStatus(channel, enable); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "EnableNACK(%d) ret:%d", enable, ret); |
| return ret; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: EnablePLI |
| * Signature: (IZ)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_EnablePLI( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jboolean enable) |
| { |
| if (NULL == vieData.rtp) |
| return -1; |
| |
| if (enable) |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "EnablePLI enable"); |
| else |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "EnablePLI disable"); |
| |
| int ret = vieData.rtp->SetKeyFrameRequestMethod(channel, |
| kViEKeyFrameRequestPliRtcp); |
| return ret; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: SetCallback |
| * Signature: (ILorg/webrtc/videoengineapp/IViEAndroidCallback;)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_SetCallback( |
| JNIEnv * env, |
| jobject, |
| jint channel, |
| jobject callback) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "SetCallback"); |
| |
| if (NULL == vieData.codec) |
| return -1; |
| if (vieData.callback == NULL) { |
| vieData.callback = new VideoCallbackAndroid(vieData, env, callback); |
| } |
| else if (vieData.codec) { |
| vieData.codec->DeregisterDecoderObserver(channel); // Wrong channel? |
| vieData.codec->DeregisterEncoderObserver(channel); |
| } |
| |
| vieData.codec->RegisterDecoderObserver(channel, *vieData.callback); |
| vieData.codec->RegisterEncoderObserver(channel, *vieData.callback); |
| |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StartIncomingRTPDump |
| * Signature: (ILjava/lang/String;)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StartIncomingRTPDump( |
| JNIEnv* env, |
| jobject, |
| jint channel, |
| jstring filename) { |
| if (NULL == vieData.rtp) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "video RTP_RTCP interface is null"); |
| return -1; |
| } |
| const char* file = env->GetStringUTFChars(filename, NULL); |
| if (!file) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Video StartRTPDump file name error"); |
| return -1; |
| } |
| if (vieData.rtp->StartRTPDump(channel, file, kRtpIncoming) != 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Video StartRTPDump error"); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: StopIncomingRTPDump |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_StopIncomingRTPDump( |
| JNIEnv *, |
| jobject, |
| jint channel) { |
| if (NULL == vieData.rtp) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "video RTP_RTCP interface is null"); |
| return -1; |
| } |
| if (vieData.rtp->StopRTPDump(channel, kRtpIncoming) != 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Video StopRTPDump error"); |
| return -1; |
| } |
| return 0; |
| } |
| |
| // |
| // VoiceEngine API wrapper functions |
| // |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_Create |
| * Signature: (Landroid/content/Context)Z |
| */ |
| JNIEXPORT jboolean JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1Create( |
| JNIEnv *env, |
| jobject context, |
| jobject ctx) { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "Create VoiceEngine"); |
| |
| VoiceEngine::SetAndroidObjects(webrtcGlobalVM, env, ctx); |
| |
| // Check if already created |
| if (voeData.ve) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "VoE already created"); |
| return false; |
| } |
| |
| // Create |
| voeData.ve = VoiceEngine::Create(); |
| if (!voeData.ve) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Create VoE failed"); |
| return false; |
| } |
| |
| // Get sub-APIs |
| if (!VE_GetSubApis()) { |
| // If not OK, release all sub-APIs and delete VoE |
| VE_ReleaseSubApis(); |
| if (!VoiceEngine::Delete(voeData.ve)) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Delete VoE failed"); |
| } |
| return false; |
| } |
| |
| return true; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_Delete |
| * Signature: ()Z |
| */ |
| JNIEXPORT jboolean JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1Delete( |
| JNIEnv *, |
| jobject) |
| { |
| // Check if exists |
| if (!voeData.ve) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "VoE does not exist"); |
| return false; |
| } |
| |
| // Release sub-APIs |
| VE_ReleaseSubApis(); |
| |
| // Delete |
| if (!VoiceEngine::Delete(voeData.ve)) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Delete VoE failed"); |
| return false; |
| } |
| |
| voeData.ve = NULL; |
| |
| // Clear instance independent Java objects |
| VoiceEngine::SetAndroidObjects(NULL, NULL, NULL); |
| |
| return true; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_Init |
| * Signature: (Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1Init( |
| JNIEnv *, |
| jobject, |
| jboolean enableTrace) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "VE_Init"); |
| |
| VALIDATE_BASE_POINTER; |
| |
| return voeData.base->Init(); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_Terminate |
| * Signature: ()I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1Terminate( |
| JNIEnv *, |
| jobject) |
| { |
| VALIDATE_BASE_POINTER; |
| |
| jint retVal = voeData.base->Terminate(); |
| return retVal; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_CreateChannel |
| * Signature: ()I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1CreateChannel( |
| JNIEnv *, |
| jobject) |
| { |
| VALIDATE_BASE_POINTER; |
| |
| webrtc::CodecInst voiceCodec; |
| int numOfVeCodecs = voeData.codec->NumOfCodecs(); |
| |
| //enum all the supported codec |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "Supported Voice Codec:\n"); |
| for (int i = 0; i < numOfVeCodecs; ++i) { |
| if (voeData.codec->GetCodec(i, voiceCodec) != -1) { |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "num: %d name: %s\n", i, voiceCodec.plname); |
| } |
| } |
| |
| jint channel = voeData.base->CreateChannel(); |
| voeData.transport.reset(new test::VoiceChannelTransport(voeData.netw, |
| channel)); |
| return channel; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_DeleteChannel |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1DeleteChannel( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| VALIDATE_BASE_POINTER; |
| voeData.transport.reset(NULL); |
| return voeData.base->DeleteChannel(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: ViE_DeleteChannel |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_ViE_1DeleteChannel( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| VALIDATE_BASE_POINTER; |
| vieData.transport.reset(NULL); |
| return vieData.base->DeleteChannel(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetLocalReceiver |
| * Signature: (II)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetLocalReceiver( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jint port) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "SetLocalReceiver"); |
| VALIDATE_BASE_POINTER; |
| if (voeData.transport.get()) { |
| return voeData.transport->SetLocalReceiver(port); |
| } |
| return -1; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetSendDestination |
| * Signature: (IILjava/lang/String;)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetSendDestination( |
| JNIEnv *env, |
| jobject, |
| jint channel, |
| jint port, |
| jstring ipaddr) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "SetSendDestination"); |
| VALIDATE_BASE_POINTER; |
| |
| const char* ipaddrNative = env->GetStringUTFChars(ipaddr, NULL); |
| if (!ipaddrNative) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Could not get UTF string"); |
| return -1; |
| } |
| if (voeData.transport.get()) { |
| jint retVal = voeData.transport->SetSendDestination(ipaddrNative, port); |
| env->ReleaseStringUTFChars(ipaddr, ipaddrNative); |
| return retVal; |
| } |
| env->ReleaseStringUTFChars(ipaddr, ipaddrNative); |
| return -1; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StartListen |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StartListen( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StartListen"); |
| VALIDATE_BASE_POINTER; |
| return voeData.base->StartReceive(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StartPlayout |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StartPlayout( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StartPlayout"); |
| VALIDATE_BASE_POINTER; |
| return voeData.base->StartPlayout(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StartSend |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StartSend( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| __android_log_write(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "StartSend"); |
| VALIDATE_BASE_POINTER; |
| return voeData.base->StartSend(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StopListen |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StopListen( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| VALIDATE_BASE_POINTER; |
| return voeData.base->StartReceive(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StopPlayout |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StopPlayout( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| VALIDATE_BASE_POINTER; |
| return voeData.base->StopPlayout(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StopSend |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StopSend( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| VALIDATE_BASE_POINTER; |
| return voeData.base->StopSend(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetSpeakerVolume |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetSpeakerVolume( |
| JNIEnv *, |
| jobject, |
| jint level) |
| { |
| VALIDATE_VOLUME_POINTER; |
| |
| if (voeData.volume->SetSpeakerVolume(level) != 0) { |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetLoudspeakerStatus |
| * Signature: (Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetLoudspeakerStatus( |
| JNIEnv *, |
| jobject, |
| jboolean enable) { |
| VALIDATE_HARDWARE_POINTER; |
| if (voeData.hardware->SetLoudspeakerStatus(enable) != 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "SetLoudspeakerStatus(%d) failed", enable); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StartPlayingFileLocally |
| * Signature: (ILjava/lang/String;Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StartPlayingFileLocally( |
| JNIEnv * env, |
| jobject, |
| jint channel, |
| jstring fileName, |
| jboolean loop) |
| { |
| VALIDATE_FILE_POINTER; |
| |
| const char* fileNameNative = env->GetStringUTFChars(fileName, NULL); |
| if (!fileNameNative) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Could not get UTF string"); |
| return -1; |
| } |
| |
| jint retVal = voeData.file->StartPlayingFileLocally(channel, |
| fileNameNative, |
| loop); |
| |
| env->ReleaseStringUTFChars(fileName, fileNameNative); |
| |
| return retVal; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StopPlayingFileLocally |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StopPlayingFileLocally( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| VALIDATE_FILE_POINTER; |
| return voeData.file->StopPlayingFileLocally(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StartPlayingFileAsMicrophone |
| * Signature: (ILjava/lang/String;Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StartPlayingFileAsMicrophone( |
| JNIEnv *env, |
| jobject, |
| jint channel, |
| jstring fileName, |
| jboolean loop) |
| { |
| VALIDATE_FILE_POINTER; |
| |
| const char* fileNameNative = env->GetStringUTFChars(fileName, NULL); |
| if (!fileNameNative) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Could not get UTF string"); |
| return -1; |
| } |
| |
| jint retVal = voeData.file->StartPlayingFileAsMicrophone(channel, |
| fileNameNative, |
| loop); |
| |
| env->ReleaseStringUTFChars(fileName, fileNameNative); |
| |
| return retVal; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StopPlayingFileAsMicrophone |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StopPlayingFileAsMicrophone( |
| JNIEnv *, |
| jobject, |
| jint channel) |
| { |
| VALIDATE_FILE_POINTER; |
| return voeData.file->StopPlayingFileAsMicrophone(channel); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_NumOfCodecs |
| * Signature: ()I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1NumOfCodecs( |
| JNIEnv *, |
| jobject) |
| { |
| VALIDATE_CODEC_POINTER; |
| return voeData.codec->NumOfCodecs(); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_NumOfCodecs |
| * Signature: ()I |
| */ |
| JNIEXPORT jobjectArray JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1GetCodecs( |
| JNIEnv *env, |
| jobject) |
| { |
| if (!voeData.codec) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Codec pointer doesn't exist"); |
| return NULL; |
| } |
| |
| jobjectArray ret; |
| int i; |
| int num = voeData.codec->NumOfCodecs(); |
| char info[256]; |
| |
| ret = (jobjectArray)env->NewObjectArray( |
| num, |
| env->FindClass("java/lang/String"), |
| env->NewStringUTF("")); |
| |
| for(i = 0; i < num; i++) { |
| webrtc::CodecInst codecToList; |
| voeData.codec->GetCodec(i, codecToList); |
| int written = snprintf(info, sizeof(info), |
| "%s type:%d freq:%d pac:%d ch:%d rate:%d", |
| codecToList.plname, codecToList.pltype, |
| codecToList.plfreq, codecToList.pacsize, |
| codecToList.channels, codecToList.rate); |
| assert(written >= 0 && written < static_cast<int>(sizeof(info))); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "VoiceEgnine Codec[%d] %s", i, info); |
| env->SetObjectArrayElement(ret, i, env->NewStringUTF( info )); |
| } |
| |
| return ret; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetSendCodec |
| * Signature: (II)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetSendCodec( |
| JNIEnv *, |
| jobject, |
| jint channel, |
| jint index) |
| { |
| VALIDATE_CODEC_POINTER; |
| |
| webrtc::CodecInst codec; |
| |
| for (int i = 0; i < voeData.codec->NumOfCodecs(); ++i) { |
| webrtc::CodecInst codecToList; |
| voeData.codec->GetCodec(i, codecToList); |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, |
| "VE Codec list %s, pltype=%d\n", |
| codecToList.plname, codecToList.pltype); |
| } |
| |
| if (voeData.codec->GetCodec(index, codec) != 0) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed to get codec"); |
| return -1; |
| } |
| __android_log_print(ANDROID_LOG_DEBUG, WEBRTC_LOG_TAG, "SetSendCodec %s\n", |
| codec.plname); |
| |
| return voeData.codec->SetSendCodec(channel, codec); |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetECStatus |
| * Signature: (Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetECStatus( |
| JNIEnv *, |
| jobject, |
| jboolean enable) { |
| VALIDATE_APM_POINTER; |
| if (voeData.apm->SetEcStatus(enable, kEcAecm) < 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed SetECStatus(%d,%d)", enable, kEcAecm); |
| return -1; |
| } |
| if (voeData.apm->SetAecmMode(kAecmSpeakerphone, false) != 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed SetAecmMode(%d,%d)", kAecmSpeakerphone, 0); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetAGCStatus |
| * Signature: (Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetAGCStatus( |
| JNIEnv *, |
| jobject, |
| jboolean enable) { |
| VALIDATE_APM_POINTER; |
| if (voeData.apm->SetAgcStatus(enable, kAgcFixedDigital) < 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed SetAgcStatus(%d,%d)", enable, kAgcFixedDigital); |
| return -1; |
| } |
| webrtc::AgcConfig config; |
| // The following settings are by default, explicitly set here. |
| config.targetLeveldBOv = 3; |
| config.digitalCompressionGaindB = 9; |
| config.limiterEnable = true; |
| if (voeData.apm->SetAgcConfig(config) != 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed SetAgcConfig(%d,%d,%d)", |
| config.targetLeveldBOv, |
| config.digitalCompressionGaindB, |
| config.limiterEnable); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_SetNSStatus |
| * Signature: (Z)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1SetNSStatus( |
| JNIEnv *, |
| jobject, |
| jboolean enable) { |
| VALIDATE_APM_POINTER; |
| if (voeData.apm->SetNsStatus(enable, kNsModerateSuppression) < 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Failed SetNsStatus(%d,%d)", |
| enable, kNsModerateSuppression); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StartDebugRecording |
| * Signature: (Ljava/lang/String)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StartDebugRecording( |
| JNIEnv* env, |
| jobject, |
| jstring filename) { |
| VALIDATE_APM_POINTER; |
| |
| const char* file = env->GetStringUTFChars(filename, NULL); |
| if (!file) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Voice StartDebugRecording file error"); |
| return -1; |
| } |
| if (voeData.apm->StartDebugRecording(file) != 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Voice StartDebugRecording error"); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StopDebugRecording |
| * Signature: ()I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StopDebugRecording( |
| JNIEnv *, |
| jobject) { |
| VALIDATE_APM_POINTER; |
| if (voeData.apm->StopDebugRecording() < 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Voice StopDebugRecording error"); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StartIncomingRTPDump |
| * Signature: (ILjava/lang/String;)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StartIncomingRTPDump( |
| JNIEnv* env, |
| jobject, |
| jint channel, |
| jstring filename) { |
| VALIDATE_RTP_POINTER; |
| const char* file = env->GetStringUTFChars(filename, NULL); |
| if (!file) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Voice StartRTPDump file error"); |
| return -1; |
| } |
| if (voeData.rtp->StartRTPDump(channel, file, kRtpIncoming) != 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Voice StartRTPDump error"); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /* |
| * Class: org_webrtc_videoengineapp_ViEAndroidJavaAPI |
| * Method: VoE_StopIncomingRTPDump |
| * Signature: (I)I |
| */ |
| JNIEXPORT jint JNICALL Java_org_webrtc_videoengineapp_ViEAndroidJavaAPI_VoE_1StopIncomingRTPDump( |
| JNIEnv *, |
| jobject, |
| jint channel) { |
| VALIDATE_RTP_POINTER; |
| if (voeData.rtp->StopRTPDump(channel) < 0) { |
| __android_log_print(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Voice StopRTPDump error"); |
| return -1; |
| } |
| return 0; |
| } |
| |
| // |
| // local function |
| // |
| |
| // Get all sub-APIs |
| bool VE_GetSubApis() { |
| bool getOK = true; |
| |
| // Base |
| voeData.base = VoEBase::GetInterface(voeData.ve); |
| if (!voeData.base) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get base sub-API failed"); |
| getOK = false; |
| } |
| |
| // Codec |
| voeData.codec = VoECodec::GetInterface(voeData.ve); |
| if (!voeData.codec) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get codec sub-API failed"); |
| getOK = false; |
| } |
| |
| // File |
| voeData.file = VoEFile::GetInterface(voeData.ve); |
| if (!voeData.file) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get file sub-API failed"); |
| getOK = false; |
| } |
| |
| // Network |
| voeData.netw = VoENetwork::GetInterface(voeData.ve); |
| if (!voeData.netw) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get network sub-API failed"); |
| getOK = false; |
| } |
| |
| // audioprocessing |
| voeData.apm = VoEAudioProcessing::GetInterface(voeData.ve); |
| if (!voeData.apm) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get VoEAudioProcessing sub-API failed"); |
| getOK = false; |
| } |
| |
| // Volume |
| voeData.volume = VoEVolumeControl::GetInterface(voeData.ve); |
| if (!voeData.volume) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get volume sub-API failed"); |
| getOK = false; |
| } |
| |
| // Hardware |
| voeData.hardware = VoEHardware::GetInterface(voeData.ve); |
| if (!voeData.hardware) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get hardware sub-API failed"); |
| getOK = false; |
| } |
| |
| // RTP |
| voeData.rtp = VoERTP_RTCP::GetInterface(voeData.ve); |
| if (!voeData.rtp) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Get rtp sub-API failed"); |
| getOK = false; |
| } |
| |
| return getOK; |
| } |
| |
| // Release all sub-APIs |
| bool VE_ReleaseSubApis() { |
| bool releaseOK = true; |
| |
| // Base |
| if (voeData.base) { |
| if (0 != voeData.base->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release base sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.base = NULL; |
| } |
| } |
| |
| // Codec |
| if (voeData.codec) { |
| if (0 != voeData.codec->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release codec sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.codec = NULL; |
| } |
| } |
| |
| // File |
| if (voeData.file) { |
| if (0 != voeData.file->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release file sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.file = NULL; |
| } |
| } |
| |
| // Network |
| if (voeData.netw) { |
| if (0 != voeData.netw->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release network sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.netw = NULL; |
| } |
| } |
| |
| // apm |
| if (voeData.apm) { |
| if (0 != voeData.apm->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release apm sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.apm = NULL; |
| } |
| } |
| |
| // Volume |
| if (voeData.volume) { |
| if (0 != voeData.volume->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release volume sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.volume = NULL; |
| } |
| } |
| |
| // Hardware |
| if (voeData.hardware) { |
| if (0 != voeData.hardware->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release hardware sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.hardware = NULL; |
| } |
| } |
| |
| if (voeData.rtp) { |
| if (0 != voeData.rtp->Release()) { |
| __android_log_write(ANDROID_LOG_ERROR, WEBRTC_LOG_TAG, |
| "Release rtp sub-API failed"); |
| releaseOK = false; |
| } |
| else { |
| voeData.rtp = NULL; |
| } |
| } |
| |
| return releaseOK; |
| } |