blob: ca1d3654e59c8b2301d765e07526aacbae1d42de [file] [log] [blame]
/*
* 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;
}