Refactoring WebRTC Java/JNI audio recording in C++ and Java.

This is a big refactoring of the existing C++/JNI/Java support for audio recording in native WebRTC:

- Removes unused code and old WEBRTC logging macros
- Now uses optimal sample rate and buffer size in Java AudioRecord (used hard-coded sample rate before)
- Makes code more inline with the implementation in Chrome
- Adds helper methods for JNI handling to improve readability
- Changes the threading model (high-prio audio thread now lives in Java-land and C++ only works as proxy)
- Adds basic thread checks
- Removes all locks in C++ land
- Removes all locks in Java
- Improves construction/destruction
- Additional cleanup

Tested using AppRTCDemo and WebRTCDemo APKs on N6, N5, N7, Samsung Galaxy S4 and
Samsung Galaxy S4 mini (which uses 44.1kHz as native sample rate).

BUG=NONE
R=magjed@webrtc.org, perkj@webrtc.org, pthatcher@webrtc.org

Review URL: https://webrtc-codereview.appspot.com/33969004

Cr-Commit-Position: refs/heads/master@{#8325}
git-svn-id: http://webrtc.googlecode.com/svn/trunk@8325 4adac7df-926f-26a2-2b94-8c16560cd09d
This commit is contained in:
henrika@webrtc.org 2015-02-11 08:38:35 +00:00
parent c2d0473320
commit 62f6e75673
21 changed files with 929 additions and 1727 deletions

View File

@ -147,16 +147,16 @@ bool PeerConnectionFactory::Initialize() {
cricket::DummyDeviceManager* device_manager(
new cricket::DummyDeviceManager());
// TODO: Need to make sure only one VoE is created inside
// WebRtcMediaEngine.
cricket::MediaEngineInterface* media_engine(
cricket::WebRtcMediaEngineFactory::Create(default_adm_.get(),
NULL, // No secondary adm.
video_encoder_factory_.get(),
video_decoder_factory_.get()));
cricket::MediaEngineInterface* media_engine =
worker_thread_->Invoke<cricket::MediaEngineInterface*>(rtc::Bind(
&PeerConnectionFactory::CreateMediaEngine_w, this));
channel_manager_.reset(new cricket::ChannelManager(
media_engine, device_manager, worker_thread_));
channel_manager_->SetVideoRtxEnabled(true);
if (!channel_manager_->Init()) {
return false;
@ -252,4 +252,11 @@ rtc::Thread* PeerConnectionFactory::worker_thread() {
return worker_thread_;
}
cricket::MediaEngineInterface* PeerConnectionFactory::CreateMediaEngine_w() {
ASSERT(worker_thread_ == rtc::Thread::Current());
return cricket::WebRtcMediaEngineFactory::Create(
default_adm_.get(), NULL, video_encoder_factory_.get(),
video_decoder_factory_.get());
}
} // namespace webrtc

View File

@ -90,6 +90,8 @@ class PeerConnectionFactory : public PeerConnectionFactoryInterface {
virtual ~PeerConnectionFactory();
private:
cricket::MediaEngineInterface* CreateMediaEngine_w();
bool owns_ptrs_;
bool wraps_current_thread_;
rtc::Thread* signaling_thread_;

View File

@ -589,6 +589,7 @@ WebRtcVoiceEngine::~WebRtcVoiceEngine() {
}
bool WebRtcVoiceEngine::Init(rtc::Thread* worker_thread) {
ASSERT(worker_thread == rtc::Thread::Current());
LOG(LS_INFO) << "WebRtcVoiceEngine::Init";
bool res = InitInternal();
if (res) {

View File

@ -149,6 +149,10 @@ ChannelManager::~ChannelManager() {
// shutdown.
ShutdownSrtp();
}
// Always delete the media engine on the worker thread to match how it was
// created.
worker_thread_->Invoke<void>(Bind(
&ChannelManager::DeleteMediaEngine_w, this));
}
bool ChannelManager::SetVideoRtxEnabled(bool enable) {
@ -215,17 +219,22 @@ bool ChannelManager::Init() {
if (initialized_) {
return false;
}
ASSERT(worker_thread_ != NULL);
if (worker_thread_) {
if (!worker_thread_) {
return false;
}
if (worker_thread_ != rtc::Thread::Current()) {
// Do not allow invoking calls to other threads on the worker thread.
worker_thread_->Invoke<bool>(rtc::Bind(
&rtc::Thread::SetAllowBlockingCalls, worker_thread_, false));
}
if (media_engine_->Init(worker_thread_)) {
initialized_ = true;
initialized_ = worker_thread_->Invoke<bool>(Bind(
&ChannelManager::InitMediaEngine_w, this));
ASSERT(initialized_);
if (!initialized_) {
return false;
}
// Now that we're initialized, apply any stored preferences. A preferred
// device might have been unplugged. In this case, we fallback to the
@ -283,21 +292,29 @@ bool ChannelManager::Init() {
if (default_video_encoder_config_.max_codec.id != 0) {
SetDefaultVideoEncoderConfig(default_video_encoder_config_);
}
}
}
return initialized_;
}
bool ChannelManager::InitMediaEngine_w() {
ASSERT(worker_thread_ == rtc::Thread::Current());
return (media_engine_->Init(worker_thread_));
}
void ChannelManager::Terminate() {
ASSERT(initialized_);
if (!initialized_) {
return;
}
worker_thread_->Invoke<void>(Bind(&ChannelManager::Terminate_w, this));
media_engine_->Terminate();
initialized_ = false;
}
void ChannelManager::DeleteMediaEngine_w() {
ASSERT(worker_thread_ == rtc::Thread::Current());
media_engine_.reset(NULL);
}
void ChannelManager::Terminate_w() {
ASSERT(worker_thread_ == rtc::Thread::Current());
// Need to destroy the voice/video channels
@ -313,6 +330,7 @@ void ChannelManager::Terminate_w() {
if (!SetCaptureDevice_w(NULL)) {
LOG(LS_WARNING) << "failed to delete video capturer";
}
media_engine_->Terminate();
}
VoiceChannel* ChannelManager::CreateVoiceChannel(

View File

@ -264,6 +264,8 @@ class ChannelManager : public rtc::MessageHandler,
DeviceManagerInterface* dm,
CaptureManager* cm,
rtc::Thread* worker_thread);
bool InitMediaEngine_w();
void DeleteMediaEngine_w();
void Terminate_w();
VoiceChannel* CreateVoiceChannel_w(
BaseSession* session, const std::string& content_name, bool rtcp);

View File

@ -19,7 +19,7 @@ using icu::UnicodeString;
jmethodID GetMethodID(JNIEnv* jni, jclass c, const std::string& name,
const char* signature) {
jmethodID m = jni->GetMethodID(c, name.c_str(), signature);
CHECK_EXCEPTION(jni, "error during GetMethodID");
CHECK_JNI_EXCEPTION(jni, "error during GetMethodID");
return m;
}
@ -37,11 +37,11 @@ jlong jlongFromPointer(void* ptr) {
// Given a (UTF-16) jstring return a new UTF-8 native string.
std::string JavaToStdString(JNIEnv* jni, const jstring& j_string) {
const jchar* jchars = jni->GetStringChars(j_string, NULL);
CHECK_EXCEPTION(jni, "Error during GetStringChars");
CHECK_JNI_EXCEPTION(jni, "Error during GetStringChars");
UnicodeString ustr(jchars, jni->GetStringLength(j_string));
CHECK_EXCEPTION(jni, "Error during GetStringLength");
CHECK_JNI_EXCEPTION(jni, "Error during GetStringLength");
jni->ReleaseStringChars(j_string, jchars);
CHECK_EXCEPTION(jni, "Error during ReleaseStringChars");
CHECK_JNI_EXCEPTION(jni, "Error during ReleaseStringChars");
std::string ret;
return ustr.toUTF8String(ret);
}
@ -72,10 +72,10 @@ jclass ClassReferenceHolder::GetClass(const std::string& name) {
void ClassReferenceHolder::LoadClass(JNIEnv* jni, const std::string& name) {
jclass localRef = jni->FindClass(name.c_str());
CHECK_EXCEPTION(jni, "Could not load class");
CHECK_JNI_EXCEPTION(jni, "Could not load class");
CHECK(localRef, name.c_str());
jclass globalRef = reinterpret_cast<jclass>(jni->NewGlobalRef(localRef));
CHECK_EXCEPTION(jni, "error during NewGlobalRef");
CHECK_JNI_EXCEPTION(jni, "error during NewGlobalRef");
CHECK(globalRef, name.c_str());
bool inserted = classes_.insert(std::make_pair(name, globalRef)).second;
CHECK(inserted, "Duplicate class name");

View File

@ -34,7 +34,7 @@
// Abort the process if |jni| has a Java exception pending, emitting |msg| to
// logcat.
#define CHECK_EXCEPTION(jni, msg) \
#define CHECK_JNI_EXCEPTION(jni, msg) \
if (0) { \
} else { \
if (jni->ExceptionCheck()) { \

View File

@ -115,7 +115,7 @@ class VideoDecodeEncodeObserver : public webrtc::ViEDecoderObserver,
jmethodID j_codec_ctor = GetMethodID(jni, j_codec_class, "<init>", "(J)V");
jobject j_codec =
jni->NewObject(j_codec_class, j_codec_ctor, jlongFromPointer(codec));
CHECK_EXCEPTION(jni, "error during NewObject");
CHECK_JNI_EXCEPTION(jni, "error during NewObject");
jni->CallVoidMethod(j_observer_, incoming_codec_changed_, video_channel,
j_codec);
}
@ -456,7 +456,7 @@ JOWW(jobject, VideoEngine_getCodec)(JNIEnv* jni, jobject j_vie, jint index) {
jmethodID j_codec_ctor = GetMethodID(jni, j_codec_class, "<init>", "(J)V");
jobject j_codec =
jni->NewObject(j_codec_class, j_codec_ctor, jlongFromPointer(codec));
CHECK_EXCEPTION(jni, "error during NewObject");
CHECK_JNI_EXCEPTION(jni, "error during NewObject");
return j_codec;
}
@ -515,7 +515,7 @@ JOWW(jobject,
jmethodID j_camera_ctor = GetMethodID(jni, j_camera_class, "<init>", "(J)V");
jobject j_camera = jni->NewObject(j_camera_class, j_camera_ctor,
jlongFromPointer(camera_info));
CHECK_EXCEPTION(jni, "error during NewObject");
CHECK_JNI_EXCEPTION(jni, "error during NewObject");
return j_camera;
}
@ -610,7 +610,7 @@ JOWW(jobject, VideoEngine_getReceivedRtcpStatistics)(JNIEnv* jni, jobject j_vie,
jni->NewObject(j_rtcp_statistics_class, j_rtcp_statistics_ctor,
fraction_lost, cumulative_lost, extended_max, jitter,
static_cast<int>(rtt_ms));
CHECK_EXCEPTION(jni, "error during NewObject");
CHECK_JNI_EXCEPTION(jni, "error during NewObject");
return j_rtcp_statistics;
}

View File

@ -327,7 +327,7 @@ JOWW(jobject, VoiceEngine_getCodec)(JNIEnv* jni, jobject j_voe, jint index) {
jmethodID j_codec_ctor = GetMethodID(jni, j_codec_class, "<init>", "(J)V");
jobject j_codec =
jni->NewObject(j_codec_class, j_codec_ctor, jlongFromPointer(codec));
CHECK_EXCEPTION(jni, "error during NewObject");
CHECK_JNI_EXCEPTION(jni, "error during NewObject");
return j_codec;
}

View File

@ -178,6 +178,7 @@ source_set("audio_device") {
}
deps = [
"../../base:rtc_base_approved",
"../../common_audio",
"../../system_wrappers",
"../utility",

View File

@ -15,6 +15,7 @@ namespace webrtc {
enum {
kDefaultSampleRate = 44100,
kBitsPerSample = 16,
kNumChannels = 1,
kDefaultBufSizeInSamples = kDefaultSampleRate * 10 / 1000,
};

View File

@ -11,8 +11,8 @@
#ifndef WEBRTC_MODULES_AUDIO_DEVICE_ANDROID_AUDIO_DEVICE_TEMPLATE_H_
#define WEBRTC_MODULES_AUDIO_DEVICE_ANDROID_AUDIO_DEVICE_TEMPLATE_H_
#include "webrtc/base/checks.h"
#include "webrtc/modules/audio_device/audio_device_generic.h"
#include "webrtc/system_wrappers/interface/trace.h"
namespace webrtc {
@ -22,13 +22,11 @@ namespace webrtc {
template <class InputType, class OutputType>
class AudioDeviceTemplate : public AudioDeviceGeneric {
public:
static int32_t SetAndroidAudioDeviceObjects(void* javaVM,
static void SetAndroidAudioDeviceObjects(void* javaVM,
void* env,
void* context) {
if (OutputType::SetAndroidAudioDeviceObjects(javaVM, env, context) == -1) {
return -1;
}
return InputType::SetAndroidAudioDeviceObjects(javaVM, env, context);
OutputType::SetAndroidAudioDeviceObjects(javaVM, env, context);
InputType::SetAndroidAudioDeviceObjects(javaVM, env, context);
}
static void ClearAndroidAudioDeviceObjects() {
@ -38,7 +36,8 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
explicit AudioDeviceTemplate(const int32_t id)
: output_(id),
input_(id, &output_) {
// TODO(henrika): provide proper delay estimate using input_(&output_).
input_() {
}
virtual ~AudioDeviceTemplate() {
@ -59,7 +58,7 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
}
bool Initialized() const {
return output_.Initialized() && input_.Initialized();
return output_.Initialized();
}
int16_t PlayoutDevices() {
@ -67,7 +66,7 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
}
int16_t RecordingDevices() {
return input_.RecordingDevices();
return 1;
}
int32_t PlayoutDeviceName(
@ -81,7 +80,7 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
uint16_t index,
char name[kAdmMaxDeviceNameSize],
char guid[kAdmMaxGuidSize]) {
return input_.RecordingDeviceName(index, name, guid);
return -1;
}
int32_t SetPlayoutDevice(uint16_t index) {
@ -94,12 +93,15 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
}
int32_t SetRecordingDevice(uint16_t index) {
return input_.SetRecordingDevice(index);
// OK to use but it has no effect currently since device selection is
// done using Andoid APIs instead.
return 0;
}
int32_t SetRecordingDevice(
AudioDeviceModule::WindowsDeviceType device) {
return input_.SetRecordingDevice(device);
FATAL() << "Should never be called";
return -1;
}
int32_t PlayoutIsAvailable(
@ -117,7 +119,8 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
int32_t RecordingIsAvailable(
bool& available) { // NOLINT
return input_.RecordingIsAvailable(available);
available = true;
return 0;
}
int32_t InitRecording() {
@ -153,17 +156,19 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
}
int32_t SetAGC(bool enable) {
return input_.SetAGC(enable);
if (enable) {
FATAL() << "Should never be called";
}
return -1;
}
bool AGC() const {
return input_.AGC();
return false;
}
int32_t SetWaveOutVolume(uint16_t volumeLeft,
uint16_t volumeRight) {
WEBRTC_TRACE(kTraceWarning, kTraceAudioDevice, 0,
" API call not supported on this platform");
FATAL() << "Should never be called";
return -1;
}
@ -184,11 +189,11 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
}
int32_t InitMicrophone() {
return input_.InitMicrophone();
return 0;
}
bool MicrophoneIsInitialized() const {
return input_.MicrophoneIsInitialized();
return true;
}
int32_t SpeakerVolumeIsAvailable(
@ -222,31 +227,38 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
int32_t MicrophoneVolumeIsAvailable(
bool& available) { // NOLINT
return input_.MicrophoneVolumeIsAvailable(available);
available = false;
FATAL() << "Should never be called";
return -1;
}
int32_t SetMicrophoneVolume(uint32_t volume) {
return input_.SetMicrophoneVolume(volume);
FATAL() << "Should never be called";
return -1;
}
int32_t MicrophoneVolume(
uint32_t& volume) const { // NOLINT
return input_.MicrophoneVolume(volume);
FATAL() << "Should never be called";
return -1;
}
int32_t MaxMicrophoneVolume(
uint32_t& maxVolume) const { // NOLINT
return input_.MaxMicrophoneVolume(maxVolume);
FATAL() << "Should never be called";
return -1;
}
int32_t MinMicrophoneVolume(
uint32_t& minVolume) const { // NOLINT
return input_.MinMicrophoneVolume(minVolume);
FATAL() << "Should never be called";
return -1;
}
int32_t MicrophoneVolumeStepSize(
uint16_t& stepSize) const { // NOLINT
return input_.MicrophoneVolumeStepSize(stepSize);
FATAL() << "Should never be called";
return -1;
}
int32_t SpeakerMuteIsAvailable(
@ -265,30 +277,36 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
int32_t MicrophoneMuteIsAvailable(
bool& available) { // NOLINT
return input_.MicrophoneMuteIsAvailable(available);
FATAL() << "Not implemented";
return -1;
}
int32_t SetMicrophoneMute(bool enable) {
return input_.SetMicrophoneMute(enable);
FATAL() << "Not implemented";
return -1;
}
int32_t MicrophoneMute(
bool& enabled) const { // NOLINT
return input_.MicrophoneMute(enabled);
FATAL() << "Not implemented";
return -1;
}
int32_t MicrophoneBoostIsAvailable(
bool& available) { // NOLINT
return input_.MicrophoneBoostIsAvailable(available);
FATAL() << "Should never be called";
return -1;
}
int32_t SetMicrophoneBoost(bool enable) {
return input_.SetMicrophoneBoost(enable);
FATAL() << "Should never be called";
return -1;
}
int32_t MicrophoneBoost(
bool& enabled) const { // NOLINT
return input_.MicrophoneBoost(enabled);
FATAL() << "Should never be called";
return -1;
}
int32_t StereoPlayoutIsAvailable(
@ -307,16 +325,18 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
int32_t StereoRecordingIsAvailable(
bool& available) { // NOLINT
return input_.StereoRecordingIsAvailable(available);
available = false;
return 0;
}
int32_t SetStereoRecording(bool enable) {
return input_.SetStereoRecording(enable);
return -1;
}
int32_t StereoRecording(
bool& enabled) const { // NOLINT
return input_.StereoRecording(enabled);
enabled = false;
return 0;
}
int32_t SetPlayoutBuffer(
@ -343,8 +363,7 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
int32_t CPULoad(
uint16_t& load) const { // NOLINT
WEBRTC_TRACE(kTraceWarning, kTraceAudioDevice, 0,
" API call not supported on this platform");
FATAL() << "Should never be called";
return -1;
}
@ -357,11 +376,11 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
}
bool RecordingWarning() const {
return input_.RecordingWarning();
return false;
}
bool RecordingError() const {
return input_.RecordingError();
return false;
}
void ClearPlayoutWarning() {
@ -372,13 +391,9 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
return output_.ClearPlayoutError();
}
void ClearRecordingWarning() {
return input_.ClearRecordingWarning();
}
void ClearRecordingWarning() {}
void ClearRecordingError() {
return input_.ClearRecordingError();
}
void ClearRecordingError() {}
void AttachAudioBuffer(
AudioDeviceBuffer* audioBuffer) {
@ -386,11 +401,6 @@ class AudioDeviceTemplate : public AudioDeviceGeneric {
input_.AttachAudioBuffer(audioBuffer);
}
int32_t SetRecordingSampleRate(
const uint32_t samplesPerSec) {
return input_.SetRecordingSampleRate(samplesPerSec);
}
int32_t SetPlayoutSampleRate(
const uint32_t samplesPerSec) {
return output_.SetPlayoutSampleRate(samplesPerSec);

View File

@ -8,42 +8,16 @@
* be found in the AUTHORS file in the root of the source tree.
*/
/*
* Android audio device utility implementation
*/
#include "webrtc/modules/audio_device/android/audio_device_utility_android.h"
#include "webrtc/system_wrappers/interface/critical_section_wrapper.h"
#include "webrtc/system_wrappers/interface/trace.h"
namespace webrtc {
namespace webrtc
{
AudioDeviceUtilityAndroid::AudioDeviceUtilityAndroid(const int32_t id) {}
AudioDeviceUtilityAndroid::AudioDeviceUtilityAndroid(const int32_t id) :
_critSect(*CriticalSectionWrapper::CreateCriticalSection()), _id(id)
{
WEBRTC_TRACE(kTraceMemory, kTraceAudioDevice, id,
"%s created", __FUNCTION__);
}
AudioDeviceUtilityAndroid::~AudioDeviceUtilityAndroid()
{
WEBRTC_TRACE(kTraceMemory, kTraceAudioDevice, _id,
"%s destroyed", __FUNCTION__);
{
CriticalSectionScoped lock(&_critSect);
}
delete &_critSect;
}
int32_t AudioDeviceUtilityAndroid::Init()
{
WEBRTC_TRACE(kTraceStateInfo, kTraceAudioDevice, _id,
" OS info: %s", "Android");
AudioDeviceUtilityAndroid::~AudioDeviceUtilityAndroid() {}
int32_t AudioDeviceUtilityAndroid::Init() {
return 0;
}

View File

@ -15,24 +15,22 @@
#ifndef WEBRTC_AUDIO_DEVICE_AUDIO_DEVICE_UTILITY_ANDROID_H
#define WEBRTC_AUDIO_DEVICE_AUDIO_DEVICE_UTILITY_ANDROID_H
#include <jni.h>
#include "webrtc/base/checks.h"
#include "webrtc/modules/audio_device/audio_device_utility.h"
#include "webrtc/modules/audio_device/include/audio_device.h"
namespace webrtc
{
class CriticalSectionWrapper;
namespace webrtc {
class AudioDeviceUtilityAndroid: public AudioDeviceUtility
{
// TODO(henrika): this utility class is not used but I would like to keep this
// file for the other helper methods which are unique for Android.
class AudioDeviceUtilityAndroid: public AudioDeviceUtility {
public:
AudioDeviceUtilityAndroid(const int32_t id);
~AudioDeviceUtilityAndroid();
virtual int32_t Init();
private:
CriticalSectionWrapper& _critSect;
int32_t _id;
};
} // namespace webrtc

File diff suppressed because it is too large Load Diff

View File

@ -13,166 +13,143 @@
#include <jni.h>
#include "webrtc/system_wrappers/interface/critical_section_wrapper.h"
#include "webrtc/base/thread_checker.h"
#include "webrtc/modules/audio_device/include/audio_device_defines.h"
#include "webrtc/modules/audio_device/audio_device_generic.h"
#include "webrtc/modules/utility/interface/helpers_android.h"
namespace webrtc {
class EventWrapper;
class ThreadWrapper;
class PlayoutDelayProvider;
const uint32_t N_REC_SAMPLES_PER_SEC = 16000; // Default is 16 kHz
const uint32_t N_REC_CHANNELS = 1; // default is mono recording
const uint32_t REC_BUF_SIZE_IN_SAMPLES = 480; // Handle max 10 ms @ 48 kHz
// Implements 16-bit mono PCM audio input support for Android using the Java
// AudioRecord interface. Most of the work is done by its Java counterpart in
// WebRtcAudioRecord.java. This class is created and lives on a thread in
// C++-land, but recorded audio buffers are delivered on a high-priority
// thread managed by the Java class.
//
// The Java class makes use of AudioEffect features (mainly AEC) which are
// first available in Jelly Bean. If it is instantiated running against earlier
// SDKs, the AEC provided by the APM in WebRTC must be used and enabled
// separately instead.
//
// An instance must be created and destroyed on one and the same thread.
// All public methods must also be called on the same thread. A thread checker
// will DCHECK if any method is called on an invalid thread.
// It is possible to call the two static methods (SetAndroidAudioDeviceObjects
// and ClearAndroidAudioDeviceObjects) from a different thread but both will
// CHECK that the calling thread is attached to a Java VM.
//
// All methods use AttachThreadScoped to attach to a Java VM if needed and then
// detach when method goes out of scope. We do so beacuse this class does not
// own the thread is is created and called on and other objects on the same
// thread might put us in a detached state at any time.
class AudioRecordJni {
public:
static int32_t SetAndroidAudioDeviceObjects(void* javaVM, void* env,
void* context);
// Use the invocation API to allow the native application to use the JNI
// interface pointer to access VM features.
// |jvm| denotes the Java VM, |env| is a pointer to the JNI interface pointer
// and |context| corresponds to android.content.Context in Java.
// This method also sets a global jclass object, |g_audio_record_class| for
// the "org/webrtc/voiceengine/WebRtcAudioRecord"-class.
static void SetAndroidAudioDeviceObjects(void* jvm, void* env, void* context);
// Always call this method after the object has been destructed. It deletes
// existing global references and enables garbage collection.
static void ClearAndroidAudioDeviceObjects();
AudioRecordJni(const int32_t id, PlayoutDelayProvider* delay_provider);
AudioRecordJni();
~AudioRecordJni();
// Main initializaton and termination
int32_t Init();
int32_t Terminate();
bool Initialized() const { return _initialized; }
// Device enumeration
int16_t RecordingDevices() { return 1; } // There is one device only
int32_t RecordingDeviceName(uint16_t index,
char name[kAdmMaxDeviceNameSize],
char guid[kAdmMaxGuidSize]);
// Device selection
int32_t SetRecordingDevice(uint16_t index);
int32_t SetRecordingDevice(
AudioDeviceModule::WindowsDeviceType device);
// Audio transport initialization
int32_t RecordingIsAvailable(bool& available); // NOLINT
int32_t InitRecording();
bool RecordingIsInitialized() const { return _recIsInitialized; }
bool RecordingIsInitialized() const { return initialized_; }
// Audio transport control
int32_t StartRecording();
int32_t StopRecording ();
bool Recording() const { return _recording; }
bool Recording() const { return recording_; }
// Microphone Automatic Gain Control (AGC)
int32_t SetAGC(bool enable);
bool AGC() const { return _AGC; }
int32_t RecordingDelay(uint16_t& delayMS) const;
// Audio mixer initialization
int32_t InitMicrophone();
bool MicrophoneIsInitialized() const { return _micIsInitialized; }
// Microphone volume controls
int32_t MicrophoneVolumeIsAvailable(bool& available); // NOLINT
// TODO(leozwang): Add microphone volume control when OpenSL APIs
// are available.
int32_t SetMicrophoneVolume(uint32_t volume);
int32_t MicrophoneVolume(uint32_t& volume) const; // NOLINT
int32_t MaxMicrophoneVolume(uint32_t& maxVolume) const; // NOLINT
int32_t MinMicrophoneVolume(uint32_t& minVolume) const; // NOLINT
int32_t MicrophoneVolumeStepSize(
uint16_t& stepSize) const; // NOLINT
// Microphone mute control
int32_t MicrophoneMuteIsAvailable(bool& available); // NOLINT
int32_t SetMicrophoneMute(bool enable);
int32_t MicrophoneMute(bool& enabled) const; // NOLINT
// Microphone boost control
int32_t MicrophoneBoostIsAvailable(bool& available); // NOLINT
int32_t SetMicrophoneBoost(bool enable);
int32_t MicrophoneBoost(bool& enabled) const; // NOLINT
// Stereo support
int32_t StereoRecordingIsAvailable(bool& available); // NOLINT
int32_t SetStereoRecording(bool enable);
int32_t StereoRecording(bool& enabled) const; // NOLINT
// Delay information and control
int32_t RecordingDelay(uint16_t& delayMS) const; // NOLINT
bool RecordingWarning() const;
bool RecordingError() const;
void ClearRecordingWarning();
void ClearRecordingError();
// Attach audio buffer
void AttachAudioBuffer(AudioDeviceBuffer* audioBuffer);
int32_t SetRecordingSampleRate(const uint32_t samplesPerSec);
bool BuiltInAECIsAvailable() const;
int32_t EnableBuiltInAEC(bool enable);
private:
void Lock() EXCLUSIVE_LOCK_FUNCTION(_critSect) {
_critSect.Enter();
}
void UnLock() UNLOCK_FUNCTION(_critSect) {
_critSect.Leave();
}
// Called from Java side so we can cache the address of the Java-manged
// |byte_buffer| in |direct_buffer_address_|. The size of the buffer
// is also stored in |direct_buffer_capacity_in_bytes_|.
// This method will be called by the WebRtcAudioRecord constructor, i.e.,
// on the same thread that this object is created on.
static void JNICALL CacheDirectBufferAddress(
JNIEnv* env, jobject obj, jobject byte_buffer, jlong nativeAudioRecord);
void OnCacheDirectBufferAddress(JNIEnv* env, jobject byte_buffer);
int32_t InitJavaResources();
int32_t InitSampleRate();
// Called periodically by the Java based WebRtcAudioRecord object when
// recording has started. Each call indicates that there are |length| new
// bytes recorded in the memory area |direct_buffer_address_| and it is
// now time to send these to the consumer.
// This method is called on a high-priority thread from Java. The name of
// the thread is 'AudioRecordThread'.
static void JNICALL DataIsRecorded(
JNIEnv* env, jobject obj, jint length, jlong nativeAudioRecord);
void OnDataIsRecorded(int length);
static bool RecThreadFunc(void*);
bool RecThreadProcess();
// Returns true if SetAndroidAudioDeviceObjects() has been called
// successfully.
bool HasDeviceObjects();
// TODO(leozwang): Android holds only one JVM, all these jni handling
// will be consolidated into a single place to make it consistant and
// reliable. Chromium has a good example at base/android.
static JavaVM* globalJvm;
static JNIEnv* globalJNIEnv;
static jobject globalContext;
static jclass globalScClass;
// Called from the constructor. Defines the |j_audio_record_| member.
void CreateJavaInstance();
JavaVM* _javaVM; // denotes a Java VM
JNIEnv* _jniEnvRec; // The JNI env for recording thread
jclass _javaScClass; // AudioDeviceAndroid class
jobject _javaScObj; // AudioDeviceAndroid object
jobject _javaRecBuffer;
void* _javaDirectRecBuffer; // Direct buffer pointer to rec buffer
jmethodID _javaMidRecAudio; // Method ID of rec in AudioDeviceAndroid
// Returns the native, or optimal, sample rate reported by the audio input
// device.
int GetNativeSampleRate();
AudioDeviceBuffer* _ptrAudioBuffer;
CriticalSectionWrapper& _critSect;
int32_t _id;
PlayoutDelayProvider* _delay_provider;
bool _initialized;
// Stores thread ID in constructor.
// We can then use ThreadChecker::CalledOnValidThread() to ensure that
// other methods are called from the same thread.
// Currently only does DCHECK(thread_checker_.CalledOnValidThread()).
rtc::ThreadChecker thread_checker_;
EventWrapper& _timeEventRec;
EventWrapper& _recStartStopEvent;
ThreadWrapper* _ptrThreadRec;
uint32_t _recThreadID;
bool _recThreadIsInitialized;
bool _shutdownRecThread;
// Stores thread ID in first call to OnDataIsRecorded() from high-priority
// thread in Java. Detached during construction of this object.
rtc::ThreadChecker thread_checker_java_;
int8_t _recBuffer[2 * REC_BUF_SIZE_IN_SAMPLES];
bool _recordingDeviceIsSpecified;
bool _recording;
bool _recIsInitialized;
bool _micIsInitialized;
// Should return the current playout delay.
// TODO(henrika): fix on Android. Reports zero today.
// PlayoutDelayProvider* delay_provider_;
bool _startRec;
// The Java WebRtcAudioRecord instance.
jobject j_audio_record_;
uint16_t _recWarning;
uint16_t _recError;
// Cached copy of address to direct audio buffer owned by |j_audio_record_|.
void* direct_buffer_address_;
uint16_t _delayRecording;
// Number of bytes in the direct audio buffer owned by |j_audio_record_|.
int direct_buffer_capacity_in_bytes_;
bool _AGC;
// Number audio frames per audio buffer. Each audio frame corresponds to
// one sample of PCM mono data at 16 bits per sample. Hence, each audio
// frame contains 2 bytes (given that the Java layer only supports mono).
// Example: 480 for 48000 Hz or 441 for 44100 Hz.
int frames_per_buffer_;
uint16_t _samplingFreqIn; // Sampling frequency for Mic
int _recAudioSource;
bool initialized_;
bool recording_;
// Raw pointer handle provided to us in AttachAudioBuffer(). Owned by the
// AudioDeviceModuleImpl class and called by AudioDeviceModuleImpl::Create().
AudioDeviceBuffer* audio_device_buffer_;
// Native sample rate set in AttachAudioBuffer() which uses JNI to ask the
// Java layer for the best possible sample rate for this particular device
// and audio configuration.
int sample_rate_hz_;
};

View File

@ -107,6 +107,7 @@ class AudioTrackJni : public PlayoutDelayProvider {
int32_t GetLoudspeakerStatus(bool& enable) const; // NOLINT
protected:
// TODO(henrika): improve this estimate.
virtual int PlayoutDelayMs() { return 0; }
private:

View File

@ -10,8 +10,13 @@
package org.webrtc.voiceengine;
import java.lang.System;
import java.lang.Thread;
import java.nio.ByteBuffer;
import java.util.concurrent.locks.ReentrantLock;
import java.util.concurrent.TimeUnit;
import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
import android.content.Context;
import android.media.AudioFormat;
@ -22,39 +27,161 @@ import android.media.AudioManager;
import android.media.AudioRecord;
import android.media.MediaRecorder.AudioSource;
import android.os.Build;
import android.os.Process;
import android.os.SystemClock;
import android.util.Log;
class WebRtcAudioRecord {
private AudioRecord _audioRecord = null;
private static final boolean DEBUG = false;
private Context _context;
private static final String TAG = "WebRtcAudioRecord";
private ByteBuffer _recBuffer;
private byte[] _tempBufRec;
// Use 44.1kHz as the default sampling rate.
private static final int SAMPLE_RATE_HZ = 44100;
private final ReentrantLock _recLock = new ReentrantLock();
// Mono recording is default.
private static final int CHANNELS = 1;
private boolean _doRecInit = true;
private boolean _isRecording = false;
// Default audio data format is PCM 16 bit per sample.
// Guaranteed to be supported by all devices.
private static final int BITS_PER_SAMPLE = 16;
private int _bufferedRecSamples = 0;
// Number of bytes per audio frame.
// Example: 16-bit PCM in stereo => 2*(16/8)=4 [bytes/frame]
private static final int BYTES_PER_FRAME = CHANNELS * (BITS_PER_SAMPLE / 8);
private AcousticEchoCanceler _aec = null;
private boolean _useBuiltInAEC = false;
// Requested size of each recorded buffer provided to the client.
private static final int CALLBACK_BUFFER_SIZE_MS = 10;
// Average number of callbacks per second.
private static final int BUFFERS_PER_SECOND = 1000 / CALLBACK_BUFFER_SIZE_MS;
private ByteBuffer byteBuffer;
private final int bytesPerBuffer;
private final int framesPerBuffer;
private final int sampleRate;
private final long nativeAudioRecord;
private final AudioManager audioManager;
private final Context context;
private AudioRecord audioRecord = null;
private AudioRecordThread audioThread = null;
private AcousticEchoCanceler aec = null;
private boolean useBuiltInAEC = false;
private final Set<Long> threadIds = new HashSet<Long>();
private static boolean runningOnJellyBeanOrHigher() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN;
}
WebRtcAudioRecord() {
try {
_recBuffer = ByteBuffer.allocateDirect(2 * 480); // Max 10 ms @ 48
// kHz
} catch (Exception e) {
DoLog(e.getMessage());
private static boolean runningOnJellyBeanMR1OrHigher() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1;
}
_tempBufRec = new byte[2 * 480];
/**
* Audio thread which keeps calling ByteBuffer.read() waiting for audio
* to be recorded. Feeds recorded data to the native counterpart as a
* periodic sequence of callbacks using DataIsRecorded().
* This thread uses a Process.THREAD_PRIORITY_URGENT_AUDIO priority.
*/
private class AudioRecordThread extends Thread {
private volatile boolean keepAlive = true;
public AudioRecordThread(String name) {
super(name);
}
@Override
public void run() {
Process.setThreadPriority(Process.THREAD_PRIORITY_URGENT_AUDIO);
DoLog("AudioRecordThread" + getThreadInfo());
AddThreadId();
try {
audioRecord.startRecording();
} catch (IllegalStateException e) {
DoLogErr("AudioRecord.startRecording failed: " + e.getMessage());
return;
}
assertIsTrue(audioRecord.getRecordingState()
== AudioRecord.RECORDSTATE_RECORDING);
long lastTime = System.nanoTime();
while (keepAlive) {
int bytesRead = audioRecord.read(byteBuffer, byteBuffer.capacity());
if (bytesRead == byteBuffer.capacity()) {
nativeDataIsRecorded(bytesRead, nativeAudioRecord);
} else {
DoLogErr("AudioRecord.read failed: " + bytesRead);
if (bytesRead == AudioRecord.ERROR_INVALID_OPERATION) {
keepAlive = false;
}
}
if (DEBUG) {
long nowTime = System.nanoTime();
long durationInMs =
TimeUnit.NANOSECONDS.toMillis((nowTime - lastTime));
lastTime = nowTime;
DoLog("bytesRead[" + durationInMs + "] " + bytesRead);
}
}
try {
audioRecord.stop();
} catch (IllegalStateException e) {
DoLogErr("AudioRecord.stop failed: " + e.getMessage());
}
RemoveThreadId();
}
public void joinThread() {
keepAlive = false;
while (isAlive()) {
try {
join();
} catch (InterruptedException e) {
// Ignore.
}
}
}
}
WebRtcAudioRecord(Context context, long nativeAudioRecord) {
DoLog("ctor" + getThreadInfo());
this.context = context;
this.nativeAudioRecord = nativeAudioRecord;
audioManager = ((AudioManager) context.getSystemService(
Context.AUDIO_SERVICE));
sampleRate = GetNativeSampleRate();
bytesPerBuffer = BYTES_PER_FRAME * (sampleRate / BUFFERS_PER_SECOND);
framesPerBuffer = sampleRate / BUFFERS_PER_SECOND;
byteBuffer = byteBuffer.allocateDirect(bytesPerBuffer);
DoLog("byteBuffer.capacity: " + byteBuffer.capacity());
// Rather than passing the ByteBuffer with every callback (requiring
// the potentially expensive GetDirectBufferAddress) we simply have the
// the native class cache the address to the memory once.
nativeCacheDirectBufferAddress(byteBuffer, nativeAudioRecord);
AddThreadId();
}
/**
* Returns the native or optimal input sample rate for this device's
* primary input stream. Unit is in Hz.
* Note that we actually query the output device but the same result is
* also valid for input.
*/
private int GetNativeSampleRate() {
if (!runningOnJellyBeanMR1OrHigher()) {
return SAMPLE_RATE_HZ;
}
String sampleRateString = audioManager.getProperty(
AudioManager.PROPERTY_OUTPUT_SAMPLE_RATE);
return (sampleRateString == null) ?
SAMPLE_RATE_HZ : Integer.parseInt(sampleRateString);
}
public static boolean BuiltInAECIsAvailable() {
@ -68,212 +195,162 @@ class WebRtcAudioRecord {
return AcousticEchoCanceler.isAvailable();
}
private int EnableBuiltInAEC(boolean enable) {
private boolean EnableBuiltInAEC(boolean enable) {
DoLog("EnableBuiltInAEC(" + enable + ')');
AddThreadId();
// AcousticEchoCanceler was added in API level 16 (Jelly Bean).
if (!runningOnJellyBeanOrHigher()) {
return -1;
return false;
}
_useBuiltInAEC = enable;
// Store the AEC state.
useBuiltInAEC = enable;
// Set AEC state if AEC has already been created.
if (_aec != null) {
int ret = _aec.setEnabled(enable);
if (aec != null) {
int ret = aec.setEnabled(enable);
if (ret != AudioEffect.SUCCESS) {
DoLogErr("AcousticEchoCanceler.setEnabled failed");
return -1;
return false;
}
DoLog("AcousticEchoCanceler.getEnabled: " + _aec.getEnabled());
DoLog("AcousticEchoCanceler.getEnabled: " + aec.getEnabled());
}
return true;
}
return 0;
}
@SuppressWarnings("unused")
private int InitRecording(int audioSource, int sampleRate) {
DoLog("InitRecording");
audioSource = AudioSource.VOICE_COMMUNICATION;
// get the minimum buffer size that can be used
int minRecBufSize = AudioRecord.getMinBufferSize(
private int InitRecording(int sampleRate) {
DoLog("InitRecording(sampleRate=" + sampleRate + ")");
AddThreadId();
// Get the minimum buffer size required for the successful creation of
// an AudioRecord object, in byte units.
// Note that this size doesn't guarantee a smooth recording under load.
// TODO(henrika): Do we need to make this larger to avoid underruns?
int minBufferSize = AudioRecord.getMinBufferSize(
sampleRate,
AudioFormat.CHANNEL_IN_MONO,
AudioFormat.ENCODING_PCM_16BIT);
DoLog("AudioRecord.getMinBufferSize: " + minBufferSize);
// DoLog("min rec buf size is " + minRecBufSize);
// double size to be more safe
int recBufSize = minRecBufSize * 2;
// On average half of the samples have been recorded/buffered and the
// recording interval is 1/100s.
_bufferedRecSamples = sampleRate / 200;
// DoLog("rough rec delay set to " + _bufferedRecSamples);
if (_aec != null) {
_aec.release();
_aec = null;
}
// release the object
if (_audioRecord != null) {
_audioRecord.release();
_audioRecord = null;
if (aec != null) {
aec.release();
aec = null;
}
if (audioRecord != null) {
audioRecord.release();
audioRecord = null;
}
int bufferSizeInBytes = Math.max(byteBuffer.capacity(), minBufferSize);
DoLog("bufferSizeInBytes: " + bufferSizeInBytes);
try {
_audioRecord = new AudioRecord(
audioSource,
audioRecord = new AudioRecord(AudioSource.VOICE_COMMUNICATION,
sampleRate,
AudioFormat.CHANNEL_IN_MONO,
AudioFormat.ENCODING_PCM_16BIT,
recBufSize);
bufferSizeInBytes);
} catch (Exception e) {
} catch (IllegalArgumentException e) {
DoLog(e.getMessage());
return -1;
}
assertIsTrue(audioRecord.getState() == AudioRecord.STATE_INITIALIZED);
// check that the audioRecord is ready to be used
if (_audioRecord.getState() != AudioRecord.STATE_INITIALIZED) {
// DoLog("rec not initialized " + sampleRate);
return -1;
}
// DoLog("rec sample rate set to " + sampleRate);
DoLog("AudioRecord " +
"session ID: " + audioRecord.getAudioSessionId() + ", " +
"audio format: " + audioRecord.getAudioFormat() + ", " +
"channels: " + audioRecord.getChannelCount() + ", " +
"sample rate: " + audioRecord.getSampleRate());
DoLog("AcousticEchoCanceler.isAvailable: " + BuiltInAECIsAvailable());
if (!BuiltInAECIsAvailable()) {
return _bufferedRecSamples;
return framesPerBuffer;
}
_aec = AcousticEchoCanceler.create(_audioRecord.getAudioSessionId());
if (_aec == null) {
aec = AcousticEchoCanceler.create(audioRecord.getAudioSessionId());
if (aec == null) {
DoLogErr("AcousticEchoCanceler.create failed");
return -1;
}
int ret = _aec.setEnabled(_useBuiltInAEC);
int ret = aec.setEnabled(useBuiltInAEC);
if (ret != AudioEffect.SUCCESS) {
DoLogErr("AcousticEchoCanceler.setEnabled failed");
return -1;
}
Descriptor descriptor = _aec.getDescriptor();
Descriptor descriptor = aec.getDescriptor();
DoLog("AcousticEchoCanceler " +
"name: " + descriptor.name + ", " +
"implementor: " + descriptor.implementor + ", " +
"uuid: " + descriptor.uuid);
DoLog("AcousticEchoCanceler.getEnabled: " + _aec.getEnabled());
return _bufferedRecSamples;
DoLog("AcousticEchoCanceler.getEnabled: " + aec.getEnabled());
return framesPerBuffer;
}
@SuppressWarnings("unused")
private int StartRecording() {
private boolean StartRecording() {
DoLog("StartRecording");
// start recording
try {
_audioRecord.startRecording();
} catch (IllegalStateException e) {
e.printStackTrace();
return -1;
AddThreadId();
if (audioRecord == null) {
DoLogErr("start() called before init()");
return false;
}
if (audioThread != null) {
DoLogErr("start() was already called");
return false;
}
audioThread = new AudioRecordThread("AudioRecordJavaThread");
audioThread.start();
return true;
}
_isRecording = true;
return 0;
}
@SuppressWarnings("unused")
private int StopRecording() {
private boolean StopRecording() {
DoLog("StopRecording");
_recLock.lock();
try {
// only stop if we are recording
if (_audioRecord.getRecordingState() ==
AudioRecord.RECORDSTATE_RECORDING) {
// stop recording
try {
_audioRecord.stop();
} catch (IllegalStateException e) {
e.printStackTrace();
return -1;
AddThreadId();
if (audioThread == null) {
DoLogErr("start() was never called, or stop() was already called");
return false;
}
audioThread.joinThread();
audioThread = null;
if (aec != null) {
aec.release();
aec = null;
}
// Release the AEC object.
if (_aec != null) {
_aec.release();
_aec = null;
if (audioRecord != null) {
audioRecord.release();
audioRecord = null;
}
// Release the AudioRecord object.
_audioRecord.release();
_audioRecord = null;
} finally {
// Ensure we always unlock, both for success, exception or error
// return.
_doRecInit = true;
_recLock.unlock();
return true;
}
_isRecording = false;
return 0;
}
@SuppressWarnings("unused")
private int RecordAudio(int lengthInBytes) {
_recLock.lock();
try {
if (_audioRecord == null) {
return -2; // We have probably closed down while waiting for rec
// lock
}
// Set priority, only do once
if (_doRecInit == true) {
try {
android.os.Process.setThreadPriority(
android.os.Process.THREAD_PRIORITY_URGENT_AUDIO);
} catch (Exception e) {
DoLog("Set rec thread priority failed: " + e.getMessage());
}
_doRecInit = false;
}
int readBytes = 0;
_recBuffer.rewind(); // Reset the position to start of buffer
readBytes = _audioRecord.read(_tempBufRec, 0, lengthInBytes);
// DoLog("read " + readBytes + "from SC");
_recBuffer.put(_tempBufRec);
if (readBytes != lengthInBytes) {
// DoLog("Could not read all data from sc (read = " + readBytes
// + ", length = " + lengthInBytes + ")");
return -1;
}
} catch (Exception e) {
DoLogErr("RecordAudio try failed: " + e.getMessage());
} finally {
// Ensure we always unlock, both for success, exception or error
// return.
_recLock.unlock();
}
return _bufferedRecSamples;
}
final String logTag = "WebRtcAudioRecord-Java";
private void DoLog(String msg) {
Log.d(logTag, msg);
Log.d(TAG, msg);
}
private void DoLogErr(String msg) {
Log.e(logTag, msg);
Log.e(TAG, msg);
}
/** Helper method for building a string of thread information.*/
private static String getThreadInfo() {
return "@[name=" + Thread.currentThread().getName()
+ ", id=" + Thread.currentThread().getId() + "]";
}
/** Helper method which throws an exception when an assertion has failed. */
private static void assertIsTrue(boolean condition) {
if (!condition) {
throw new AssertionError("Expected condition to be true");
}
}
private void AddThreadId() {
threadIds.add(Thread.currentThread().getId());
DoLog("threadIds: " + threadIds + " (#threads=" + threadIds.size() + ")");
}
private void RemoveThreadId() {
threadIds.remove(Thread.currentThread().getId());
DoLog("threadIds: " + threadIds + " (#threads=" + threadIds.size() + ")");
}
private native void nativeCacheDirectBufferAddress(
ByteBuffer byteBuffer, long nativeAudioRecord);
private native void nativeDataIsRecorded(int bytes, long nativeAudioRecord);
}

View File

@ -13,6 +13,7 @@
'type': 'static_library',
'dependencies': [
'webrtc_utility',
'<(webrtc_root)/base/base.gyp:rtc_base_approved',
'<(webrtc_root)/common_audio/common_audio.gyp:common_audio',
'<(webrtc_root)/system_wrappers/system_wrappers.gyp:system_wrappers',
],

View File

@ -12,9 +12,37 @@
#define WEBRTC_MODULES_UTILITY_INTERFACE_HELPERS_ANDROID_H_
#include <jni.h>
#include <string>
// Abort the process if |jni| has a Java exception pending.
// TODO(henrika): merge with CHECK_JNI_EXCEPTION() in jni_helpers.h.
#define CHECK_EXCEPTION(jni) \
CHECK(!jni->ExceptionCheck()) \
<< (jni->ExceptionDescribe(), jni->ExceptionClear(), "")
namespace webrtc {
// Return a |JNIEnv*| usable on this thread or NULL if this thread is detached.
JNIEnv* GetEnv(JavaVM* jvm);
// JNIEnv-helper methods that wraps the API which uses the JNI interface
// pointer (JNIEnv*). It allows us to CHECK success and that no Java exception
// is thrown while calling the method.
jmethodID GetMethodID (
JNIEnv* jni, jclass c, const std::string& name, const char* signature);
jclass FindClass(JNIEnv* jni, const std::string& name);
jobject NewGlobalRef(JNIEnv* jni, jobject o);
void DeleteGlobalRef(JNIEnv* jni, jobject o);
// Return thread ID as a string.
std::string GetThreadId();
// Return thread ID as string suitable for debug logging.
std::string GetThreadInfo();
// Attach thread to JVM if necessary and detach at scope end if originally
// attached.
class AttachThreadScoped {
@ -29,6 +57,23 @@ class AttachThreadScoped {
JNIEnv* env_;
};
// Scoped holder for global Java refs.
template<class T> // T is jclass, jobject, jintArray, etc.
class ScopedGlobalRef {
public:
ScopedGlobalRef(JNIEnv* jni, T obj)
: jni_(jni), obj_(static_cast<T>(NewGlobalRef(jni, obj))) {}
~ScopedGlobalRef() {
DeleteGlobalRef(jni_, obj_);
}
T operator*() const {
return obj_;
}
private:
JNIEnv* jni_;
T obj_;
};
} // namespace webrtc
#endif // WEBRTC_MODULES_UTILITY_INTERFACE_HELPERS_ANDROID_H_

View File

@ -8,27 +8,89 @@
* be found in the AUTHORS file in the root of the source tree.
*/
#include "webrtc/base/checks.h"
#include "webrtc/modules/utility/interface/helpers_android.h"
#include <android/log.h>
#include <assert.h>
#include <pthread.h>
#include <stddef.h>
#include <unistd.h>
#define TAG "HelpersAndroid"
#define ALOGD(...) __android_log_print(ANDROID_LOG_DEBUG, TAG, __VA_ARGS__)
namespace webrtc {
JNIEnv* GetEnv(JavaVM* jvm) {
void* env = NULL;
jint status = jvm->GetEnv(&env, JNI_VERSION_1_6);
CHECK(((env != NULL) && (status == JNI_OK)) ||
((env == NULL) && (status == JNI_EDETACHED)))
<< "Unexpected GetEnv return: " << status << ":" << env;
return reinterpret_cast<JNIEnv*>(env);
}
jmethodID GetMethodID (
JNIEnv* jni, jclass c, const std::string& name, const char* signature) {
jmethodID m = jni->GetMethodID(c, name.c_str(), signature);
CHECK_EXCEPTION(jni) << "Error during GetMethodID: " << name << ", "
<< signature;
CHECK(m) << name << ", " << signature;
return m;
}
jclass FindClass(JNIEnv* jni, const std::string& name) {
jclass c = jni->FindClass(name.c_str());
CHECK_EXCEPTION(jni) << "Error during FindClass: " << name;
CHECK(c) << name;
return c;
}
jobject NewGlobalRef(JNIEnv* jni, jobject o) {
jobject ret = jni->NewGlobalRef(o);
CHECK_EXCEPTION(jni) << "Error during NewGlobalRef";
CHECK(ret);
return ret;
}
void DeleteGlobalRef(JNIEnv* jni, jobject o) {
jni->DeleteGlobalRef(o);
CHECK_EXCEPTION(jni) << "Error during DeleteGlobalRef";
}
std::string GetThreadId() {
char buf[21]; // Big enough to hold a kuint64max plus terminating NULL.
int thread_id = gettid();
CHECK_LT(snprintf(buf, sizeof(buf), "%i", thread_id),
static_cast<int>(sizeof(buf))) << "Thread id is bigger than uint64??";
return std::string(buf);
}
std::string GetThreadInfo() {
return "@[tid=" + GetThreadId() + "]";
}
AttachThreadScoped::AttachThreadScoped(JavaVM* jvm)
: attached_(false), jvm_(jvm), env_(NULL) {
jint ret_val = jvm->GetEnv(reinterpret_cast<void**>(&env_), JNI_VERSION_1_4);
if (ret_val == JNI_EDETACHED) {
// Attach the thread to the Java VM.
ret_val = jvm_->AttachCurrentThread(&env_, NULL);
attached_ = ret_val == JNI_OK;
assert(attached_);
env_ = GetEnv(jvm);
if (!env_) {
// Adding debug log here so we can track down potential leaks and figure
// out why we sometimes see "Native thread exiting without having called
// DetachCurrentThread" in logcat outputs.
ALOGD("Attaching thread to JVM%s", GetThreadInfo().c_str());
jint res = jvm->AttachCurrentThread(&env_, NULL);
attached_ = (res == JNI_OK);
CHECK(attached_) << "AttachCurrentThread failed: " << res;
}
}
AttachThreadScoped::~AttachThreadScoped() {
if (attached_ && (jvm_->DetachCurrentThread() < 0)) {
assert(false);
if (attached_) {
ALOGD("Detaching thread from JVM%s", GetThreadInfo().c_str());
jint res = jvm_->DetachCurrentThread();
CHECK(res == JNI_OK) << "DetachCurrentThread failed: " << res;
CHECK(!GetEnv(jvm_));
}
}