mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-13 13:50:40 +01:00

This avoids the need of hard coding the values to use networkIgnoreMask on Android platform. Bug: None Change-Id: Ib5e860913cec2c6d41cfa1b778cb122d0bfe1300 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/311780 Reviewed-by: Sergey Silkin <ssilkin@webrtc.org> Commit-Queue: Sergey Silkin <ssilkin@webrtc.org> Cr-Commit-Position: refs/heads/main@{#40541}
634 lines
25 KiB
Java
634 lines
25 KiB
Java
/*
|
|
* Copyright 2013 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.
|
|
*/
|
|
|
|
package org.webrtc;
|
|
|
|
import android.content.Context;
|
|
import android.os.Process;
|
|
import androidx.annotation.Nullable;
|
|
import java.util.List;
|
|
import org.webrtc.Logging.Severity;
|
|
import org.webrtc.MediaStreamTrack;
|
|
import org.webrtc.PeerConnection;
|
|
import org.webrtc.RtpCapabilities;
|
|
import org.webrtc.audio.AudioDeviceModule;
|
|
import org.webrtc.audio.JavaAudioDeviceModule;
|
|
|
|
/**
|
|
* Java wrapper for a C++ PeerConnectionFactoryInterface. Main entry point to
|
|
* the PeerConnection API for clients.
|
|
*/
|
|
public class PeerConnectionFactory {
|
|
public static final String TRIAL_ENABLED = "Enabled";
|
|
@Deprecated public static final String VIDEO_FRAME_EMIT_TRIAL = "VideoFrameEmit";
|
|
|
|
private static final String TAG = "PeerConnectionFactory";
|
|
private static final String VIDEO_CAPTURER_THREAD_NAME = "VideoCapturerThread";
|
|
|
|
/** Helper class holding both Java and C++ thread info. */
|
|
private static class ThreadInfo {
|
|
final Thread thread;
|
|
final int tid;
|
|
|
|
public static ThreadInfo getCurrent() {
|
|
return new ThreadInfo(Thread.currentThread(), Process.myTid());
|
|
}
|
|
|
|
private ThreadInfo(Thread thread, int tid) {
|
|
this.thread = thread;
|
|
this.tid = tid;
|
|
}
|
|
}
|
|
|
|
private static volatile boolean internalTracerInitialized;
|
|
|
|
// Remove these once deprecated static printStackTrace() is gone.
|
|
@Nullable private static ThreadInfo staticNetworkThread;
|
|
@Nullable private static ThreadInfo staticWorkerThread;
|
|
@Nullable private static ThreadInfo staticSignalingThread;
|
|
|
|
private long nativeFactory;
|
|
@Nullable private volatile ThreadInfo networkThread;
|
|
@Nullable private volatile ThreadInfo workerThread;
|
|
@Nullable private volatile ThreadInfo signalingThread;
|
|
|
|
public static class InitializationOptions {
|
|
final Context applicationContext;
|
|
final String fieldTrials;
|
|
final boolean enableInternalTracer;
|
|
final NativeLibraryLoader nativeLibraryLoader;
|
|
final String nativeLibraryName;
|
|
@Nullable Loggable loggable;
|
|
@Nullable Severity loggableSeverity;
|
|
|
|
private InitializationOptions(Context applicationContext, String fieldTrials,
|
|
boolean enableInternalTracer, NativeLibraryLoader nativeLibraryLoader,
|
|
String nativeLibraryName, @Nullable Loggable loggable,
|
|
@Nullable Severity loggableSeverity) {
|
|
this.applicationContext = applicationContext;
|
|
this.fieldTrials = fieldTrials;
|
|
this.enableInternalTracer = enableInternalTracer;
|
|
this.nativeLibraryLoader = nativeLibraryLoader;
|
|
this.nativeLibraryName = nativeLibraryName;
|
|
this.loggable = loggable;
|
|
this.loggableSeverity = loggableSeverity;
|
|
}
|
|
|
|
public static Builder builder(Context applicationContext) {
|
|
return new Builder(applicationContext);
|
|
}
|
|
|
|
public static class Builder {
|
|
private final Context applicationContext;
|
|
private String fieldTrials = "";
|
|
private boolean enableInternalTracer;
|
|
private NativeLibraryLoader nativeLibraryLoader = new NativeLibrary.DefaultLoader();
|
|
private String nativeLibraryName = "jingle_peerconnection_so";
|
|
@Nullable private Loggable loggable;
|
|
@Nullable private Severity loggableSeverity;
|
|
|
|
Builder(Context applicationContext) {
|
|
this.applicationContext = applicationContext;
|
|
}
|
|
|
|
public Builder setFieldTrials(String fieldTrials) {
|
|
this.fieldTrials = fieldTrials;
|
|
return this;
|
|
}
|
|
|
|
public Builder setEnableInternalTracer(boolean enableInternalTracer) {
|
|
this.enableInternalTracer = enableInternalTracer;
|
|
return this;
|
|
}
|
|
|
|
public Builder setNativeLibraryLoader(NativeLibraryLoader nativeLibraryLoader) {
|
|
this.nativeLibraryLoader = nativeLibraryLoader;
|
|
return this;
|
|
}
|
|
|
|
public Builder setNativeLibraryName(String nativeLibraryName) {
|
|
this.nativeLibraryName = nativeLibraryName;
|
|
return this;
|
|
}
|
|
|
|
public Builder setInjectableLogger(Loggable loggable, Severity severity) {
|
|
this.loggable = loggable;
|
|
this.loggableSeverity = severity;
|
|
return this;
|
|
}
|
|
|
|
public PeerConnectionFactory.InitializationOptions createInitializationOptions() {
|
|
return new PeerConnectionFactory.InitializationOptions(applicationContext, fieldTrials,
|
|
enableInternalTracer, nativeLibraryLoader, nativeLibraryName, loggable,
|
|
loggableSeverity);
|
|
}
|
|
}
|
|
}
|
|
|
|
public static class Options {
|
|
// Keep in sync with webrtc/rtc_base/network.h!
|
|
//
|
|
// These bit fields are defined for `networkIgnoreMask` below.
|
|
public static final int ADAPTER_TYPE_UNKNOWN = 0;
|
|
public static final int ADAPTER_TYPE_ETHERNET = 1 << 0;
|
|
public static final int ADAPTER_TYPE_WIFI = 1 << 1;
|
|
public static final int ADAPTER_TYPE_CELLULAR = 1 << 2;
|
|
public static final int ADAPTER_TYPE_VPN = 1 << 3;
|
|
public static final int ADAPTER_TYPE_LOOPBACK = 1 << 4;
|
|
public static final int ADAPTER_TYPE_ANY = 1 << 5;
|
|
|
|
public int networkIgnoreMask;
|
|
public boolean disableEncryption;
|
|
public boolean disableNetworkMonitor;
|
|
|
|
@CalledByNative("Options")
|
|
int getNetworkIgnoreMask() {
|
|
return networkIgnoreMask;
|
|
}
|
|
|
|
@CalledByNative("Options")
|
|
boolean getDisableEncryption() {
|
|
return disableEncryption;
|
|
}
|
|
|
|
@CalledByNative("Options")
|
|
boolean getDisableNetworkMonitor() {
|
|
return disableNetworkMonitor;
|
|
}
|
|
}
|
|
|
|
public static class Builder {
|
|
@Nullable private Options options;
|
|
@Nullable private AudioDeviceModule audioDeviceModule;
|
|
private AudioEncoderFactoryFactory audioEncoderFactoryFactory =
|
|
new BuiltinAudioEncoderFactoryFactory();
|
|
private AudioDecoderFactoryFactory audioDecoderFactoryFactory =
|
|
new BuiltinAudioDecoderFactoryFactory();
|
|
@Nullable private VideoEncoderFactory videoEncoderFactory;
|
|
@Nullable private VideoDecoderFactory videoDecoderFactory;
|
|
@Nullable private AudioProcessingFactory audioProcessingFactory;
|
|
@Nullable private FecControllerFactoryFactoryInterface fecControllerFactoryFactory;
|
|
@Nullable private NetworkControllerFactoryFactory networkControllerFactoryFactory;
|
|
@Nullable private NetworkStatePredictorFactoryFactory networkStatePredictorFactoryFactory;
|
|
@Nullable private NetEqFactoryFactory neteqFactoryFactory;
|
|
|
|
private Builder() {}
|
|
|
|
public Builder setOptions(Options options) {
|
|
this.options = options;
|
|
return this;
|
|
}
|
|
|
|
public Builder setAudioDeviceModule(AudioDeviceModule audioDeviceModule) {
|
|
this.audioDeviceModule = audioDeviceModule;
|
|
return this;
|
|
}
|
|
|
|
public Builder setAudioEncoderFactoryFactory(
|
|
AudioEncoderFactoryFactory audioEncoderFactoryFactory) {
|
|
if (audioEncoderFactoryFactory == null) {
|
|
throw new IllegalArgumentException(
|
|
"PeerConnectionFactory.Builder does not accept a null AudioEncoderFactoryFactory.");
|
|
}
|
|
this.audioEncoderFactoryFactory = audioEncoderFactoryFactory;
|
|
return this;
|
|
}
|
|
|
|
public Builder setAudioDecoderFactoryFactory(
|
|
AudioDecoderFactoryFactory audioDecoderFactoryFactory) {
|
|
if (audioDecoderFactoryFactory == null) {
|
|
throw new IllegalArgumentException(
|
|
"PeerConnectionFactory.Builder does not accept a null AudioDecoderFactoryFactory.");
|
|
}
|
|
this.audioDecoderFactoryFactory = audioDecoderFactoryFactory;
|
|
return this;
|
|
}
|
|
|
|
public Builder setVideoEncoderFactory(VideoEncoderFactory videoEncoderFactory) {
|
|
this.videoEncoderFactory = videoEncoderFactory;
|
|
return this;
|
|
}
|
|
|
|
public Builder setVideoDecoderFactory(VideoDecoderFactory videoDecoderFactory) {
|
|
this.videoDecoderFactory = videoDecoderFactory;
|
|
return this;
|
|
}
|
|
|
|
public Builder setAudioProcessingFactory(AudioProcessingFactory audioProcessingFactory) {
|
|
if (audioProcessingFactory == null) {
|
|
throw new NullPointerException(
|
|
"PeerConnectionFactory builder does not accept a null AudioProcessingFactory.");
|
|
}
|
|
this.audioProcessingFactory = audioProcessingFactory;
|
|
return this;
|
|
}
|
|
|
|
public Builder setFecControllerFactoryFactoryInterface(
|
|
FecControllerFactoryFactoryInterface fecControllerFactoryFactory) {
|
|
this.fecControllerFactoryFactory = fecControllerFactoryFactory;
|
|
return this;
|
|
}
|
|
|
|
public Builder setNetworkControllerFactoryFactory(
|
|
NetworkControllerFactoryFactory networkControllerFactoryFactory) {
|
|
this.networkControllerFactoryFactory = networkControllerFactoryFactory;
|
|
return this;
|
|
}
|
|
|
|
public Builder setNetworkStatePredictorFactoryFactory(
|
|
NetworkStatePredictorFactoryFactory networkStatePredictorFactoryFactory) {
|
|
this.networkStatePredictorFactoryFactory = networkStatePredictorFactoryFactory;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Sets a NetEqFactoryFactory for the PeerConnectionFactory. When using a
|
|
* custom NetEqFactoryFactory, the AudioDecoderFactoryFactory will be set
|
|
* to null. The AudioDecoderFactoryFactory should be wrapped in the
|
|
* NetEqFactoryFactory.
|
|
*/
|
|
public Builder setNetEqFactoryFactory(NetEqFactoryFactory neteqFactoryFactory) {
|
|
this.neteqFactoryFactory = neteqFactoryFactory;
|
|
return this;
|
|
}
|
|
|
|
public PeerConnectionFactory createPeerConnectionFactory() {
|
|
checkInitializeHasBeenCalled();
|
|
if (audioDeviceModule == null) {
|
|
audioDeviceModule = JavaAudioDeviceModule.builder(ContextUtils.getApplicationContext())
|
|
.createAudioDeviceModule();
|
|
}
|
|
return nativeCreatePeerConnectionFactory(ContextUtils.getApplicationContext(), options,
|
|
audioDeviceModule.getNativeAudioDeviceModulePointer(),
|
|
audioEncoderFactoryFactory.createNativeAudioEncoderFactory(),
|
|
audioDecoderFactoryFactory.createNativeAudioDecoderFactory(), videoEncoderFactory,
|
|
videoDecoderFactory,
|
|
audioProcessingFactory == null ? 0 : audioProcessingFactory.createNative(),
|
|
fecControllerFactoryFactory == null ? 0 : fecControllerFactoryFactory.createNative(),
|
|
networkControllerFactoryFactory == null
|
|
? 0
|
|
: networkControllerFactoryFactory.createNativeNetworkControllerFactory(),
|
|
networkStatePredictorFactoryFactory == null
|
|
? 0
|
|
: networkStatePredictorFactoryFactory.createNativeNetworkStatePredictorFactory(),
|
|
neteqFactoryFactory == null ? 0 : neteqFactoryFactory.createNativeNetEqFactory());
|
|
}
|
|
}
|
|
|
|
public static Builder builder() {
|
|
return new Builder();
|
|
}
|
|
|
|
/**
|
|
* Loads and initializes WebRTC. This must be called at least once before creating a
|
|
* PeerConnectionFactory. Replaces all the old initialization methods. Must not be called while
|
|
* a PeerConnectionFactory is alive.
|
|
*/
|
|
public static void initialize(InitializationOptions options) {
|
|
ContextUtils.initialize(options.applicationContext);
|
|
NativeLibrary.initialize(options.nativeLibraryLoader, options.nativeLibraryName);
|
|
nativeInitializeAndroidGlobals();
|
|
nativeInitializeFieldTrials(options.fieldTrials);
|
|
if (options.enableInternalTracer && !internalTracerInitialized) {
|
|
initializeInternalTracer();
|
|
}
|
|
if (options.loggable != null) {
|
|
Logging.injectLoggable(options.loggable, options.loggableSeverity);
|
|
nativeInjectLoggable(new JNILogging(options.loggable), options.loggableSeverity.ordinal());
|
|
} else {
|
|
Logging.d(TAG,
|
|
"PeerConnectionFactory was initialized without an injected Loggable. "
|
|
+ "Any existing Loggable will be deleted.");
|
|
Logging.deleteInjectedLoggable();
|
|
nativeDeleteLoggable();
|
|
}
|
|
}
|
|
|
|
private static void checkInitializeHasBeenCalled() {
|
|
if (!NativeLibrary.isLoaded() || ContextUtils.getApplicationContext() == null) {
|
|
throw new IllegalStateException(
|
|
"PeerConnectionFactory.initialize was not called before creating a "
|
|
+ "PeerConnectionFactory.");
|
|
}
|
|
}
|
|
|
|
private static void initializeInternalTracer() {
|
|
internalTracerInitialized = true;
|
|
nativeInitializeInternalTracer();
|
|
}
|
|
|
|
public static void shutdownInternalTracer() {
|
|
internalTracerInitialized = false;
|
|
nativeShutdownInternalTracer();
|
|
}
|
|
|
|
// Field trial initialization. Must be called before PeerConnectionFactory
|
|
// is created.
|
|
// Deprecated, use PeerConnectionFactory.initialize instead.
|
|
@Deprecated
|
|
public static void initializeFieldTrials(String fieldTrialsInitString) {
|
|
nativeInitializeFieldTrials(fieldTrialsInitString);
|
|
}
|
|
|
|
// Wrapper of webrtc::field_trial::FindFullName. Develop the feature with default behaviour off.
|
|
// Example usage:
|
|
// if (PeerConnectionFactory.fieldTrialsFindFullName("WebRTCExperiment").equals("Enabled")) {
|
|
// method1();
|
|
// } else {
|
|
// method2();
|
|
// }
|
|
public static String fieldTrialsFindFullName(String name) {
|
|
return NativeLibrary.isLoaded() ? nativeFindFieldTrialsFullName(name) : "";
|
|
}
|
|
// Start/stop internal capturing of internal tracing.
|
|
public static boolean startInternalTracingCapture(String tracingFilename) {
|
|
return nativeStartInternalTracingCapture(tracingFilename);
|
|
}
|
|
|
|
public static void stopInternalTracingCapture() {
|
|
nativeStopInternalTracingCapture();
|
|
}
|
|
|
|
@CalledByNative
|
|
PeerConnectionFactory(long nativeFactory) {
|
|
checkInitializeHasBeenCalled();
|
|
if (nativeFactory == 0) {
|
|
throw new RuntimeException("Failed to initialize PeerConnectionFactory!");
|
|
}
|
|
this.nativeFactory = nativeFactory;
|
|
}
|
|
|
|
/**
|
|
* Internal helper function to pass the parameters down into the native JNI bridge.
|
|
*/
|
|
@Nullable
|
|
PeerConnection createPeerConnectionInternal(PeerConnection.RTCConfiguration rtcConfig,
|
|
MediaConstraints constraints, PeerConnection.Observer observer,
|
|
SSLCertificateVerifier sslCertificateVerifier) {
|
|
checkPeerConnectionFactoryExists();
|
|
long nativeObserver = PeerConnection.createNativePeerConnectionObserver(observer);
|
|
if (nativeObserver == 0) {
|
|
return null;
|
|
}
|
|
long nativePeerConnection = nativeCreatePeerConnection(
|
|
nativeFactory, rtcConfig, constraints, nativeObserver, sslCertificateVerifier);
|
|
if (nativePeerConnection == 0) {
|
|
return null;
|
|
}
|
|
return new PeerConnection(nativePeerConnection);
|
|
}
|
|
|
|
/**
|
|
* Deprecated. PeerConnection constraints are deprecated. Supply values in rtcConfig struct
|
|
* instead and use the method without constraints in the signature.
|
|
*/
|
|
@Nullable
|
|
@Deprecated
|
|
public PeerConnection createPeerConnection(PeerConnection.RTCConfiguration rtcConfig,
|
|
MediaConstraints constraints, PeerConnection.Observer observer) {
|
|
return createPeerConnectionInternal(
|
|
rtcConfig, constraints, observer, /* sslCertificateVerifier= */ null);
|
|
}
|
|
|
|
/**
|
|
* Deprecated. PeerConnection constraints are deprecated. Supply values in rtcConfig struct
|
|
* instead and use the method without constraints in the signature.
|
|
*/
|
|
@Nullable
|
|
@Deprecated
|
|
public PeerConnection createPeerConnection(List<PeerConnection.IceServer> iceServers,
|
|
MediaConstraints constraints, PeerConnection.Observer observer) {
|
|
PeerConnection.RTCConfiguration rtcConfig = new PeerConnection.RTCConfiguration(iceServers);
|
|
rtcConfig.sdpSemantics = PeerConnection.SdpSemantics.UNIFIED_PLAN;
|
|
return createPeerConnection(rtcConfig, constraints, observer);
|
|
}
|
|
|
|
@Nullable
|
|
public PeerConnection createPeerConnection(
|
|
List<PeerConnection.IceServer> iceServers, PeerConnection.Observer observer) {
|
|
PeerConnection.RTCConfiguration rtcConfig = new PeerConnection.RTCConfiguration(iceServers);
|
|
rtcConfig.sdpSemantics = PeerConnection.SdpSemantics.UNIFIED_PLAN;
|
|
return createPeerConnection(rtcConfig, observer);
|
|
}
|
|
|
|
@Nullable
|
|
public PeerConnection createPeerConnection(
|
|
PeerConnection.RTCConfiguration rtcConfig, PeerConnection.Observer observer) {
|
|
return createPeerConnection(rtcConfig, null /* constraints */, observer);
|
|
}
|
|
|
|
@Nullable
|
|
public PeerConnection createPeerConnection(
|
|
PeerConnection.RTCConfiguration rtcConfig, PeerConnectionDependencies dependencies) {
|
|
return createPeerConnectionInternal(rtcConfig, null /* constraints */,
|
|
dependencies.getObserver(), dependencies.getSSLCertificateVerifier());
|
|
}
|
|
|
|
public MediaStream createLocalMediaStream(String label) {
|
|
checkPeerConnectionFactoryExists();
|
|
return new MediaStream(nativeCreateLocalMediaStream(nativeFactory, label));
|
|
}
|
|
|
|
/**
|
|
* Create video source with given parameters. If alignTimestamps is false, the caller is
|
|
* responsible for aligning the frame timestamps to rtc::TimeNanos(). This can be used to achieve
|
|
* higher accuracy if there is a big delay between frame creation and frames being delivered to
|
|
* the returned video source. If alignTimestamps is true, timestamps will be aligned to
|
|
* rtc::TimeNanos() when they arrive to the returned video source.
|
|
*/
|
|
public VideoSource createVideoSource(boolean isScreencast, boolean alignTimestamps) {
|
|
checkPeerConnectionFactoryExists();
|
|
return new VideoSource(nativeCreateVideoSource(nativeFactory, isScreencast, alignTimestamps));
|
|
}
|
|
|
|
/**
|
|
* Same as above with alignTimestamps set to true.
|
|
*
|
|
* @see #createVideoSource(boolean, boolean)
|
|
*/
|
|
public VideoSource createVideoSource(boolean isScreencast) {
|
|
return createVideoSource(isScreencast, /* alignTimestamps= */ true);
|
|
}
|
|
|
|
public VideoTrack createVideoTrack(String id, VideoSource source) {
|
|
checkPeerConnectionFactoryExists();
|
|
return new VideoTrack(
|
|
nativeCreateVideoTrack(nativeFactory, id, source.getNativeVideoTrackSource()));
|
|
}
|
|
|
|
public AudioSource createAudioSource(MediaConstraints constraints) {
|
|
checkPeerConnectionFactoryExists();
|
|
return new AudioSource(nativeCreateAudioSource(nativeFactory, constraints));
|
|
}
|
|
|
|
public AudioTrack createAudioTrack(String id, AudioSource source) {
|
|
checkPeerConnectionFactoryExists();
|
|
return new AudioTrack(nativeCreateAudioTrack(nativeFactory, id, source.getNativeAudioSource()));
|
|
}
|
|
|
|
public RtpCapabilities getRtpReceiverCapabilities(MediaStreamTrack.MediaType mediaType) {
|
|
checkPeerConnectionFactoryExists();
|
|
return nativeGetRtpReceiverCapabilities(nativeFactory, mediaType);
|
|
}
|
|
|
|
public RtpCapabilities getRtpSenderCapabilities(MediaStreamTrack.MediaType mediaType) {
|
|
checkPeerConnectionFactoryExists();
|
|
return nativeGetRtpSenderCapabilities(nativeFactory, mediaType);
|
|
}
|
|
|
|
// Starts recording an AEC dump. Ownership of the file is transfered to the
|
|
// native code. If an AEC dump is already in progress, it will be stopped and
|
|
// a new one will start using the provided file.
|
|
public boolean startAecDump(int file_descriptor, int filesize_limit_bytes) {
|
|
checkPeerConnectionFactoryExists();
|
|
return nativeStartAecDump(nativeFactory, file_descriptor, filesize_limit_bytes);
|
|
}
|
|
|
|
// Stops recording an AEC dump. If no AEC dump is currently being recorded,
|
|
// this call will have no effect.
|
|
public void stopAecDump() {
|
|
checkPeerConnectionFactoryExists();
|
|
nativeStopAecDump(nativeFactory);
|
|
}
|
|
|
|
public void dispose() {
|
|
checkPeerConnectionFactoryExists();
|
|
nativeFreeFactory(nativeFactory);
|
|
networkThread = null;
|
|
workerThread = null;
|
|
signalingThread = null;
|
|
nativeFactory = 0;
|
|
}
|
|
|
|
/** Returns a pointer to the native webrtc::PeerConnectionFactoryInterface. */
|
|
public long getNativePeerConnectionFactory() {
|
|
checkPeerConnectionFactoryExists();
|
|
return nativeGetNativePeerConnectionFactory(nativeFactory);
|
|
}
|
|
|
|
/** Returns a pointer to the native OwnedFactoryAndThreads object */
|
|
public long getNativeOwnedFactoryAndThreads() {
|
|
checkPeerConnectionFactoryExists();
|
|
return nativeFactory;
|
|
}
|
|
|
|
private void checkPeerConnectionFactoryExists() {
|
|
if (nativeFactory == 0) {
|
|
throw new IllegalStateException("PeerConnectionFactory has been disposed.");
|
|
}
|
|
}
|
|
|
|
private static void printStackTrace(
|
|
@Nullable ThreadInfo threadInfo, boolean printNativeStackTrace) {
|
|
if (threadInfo == null) {
|
|
// Thread callbacks have not been completed yet, ignore call.
|
|
return;
|
|
}
|
|
final String threadName = threadInfo.thread.getName();
|
|
StackTraceElement[] stackTraces = threadInfo.thread.getStackTrace();
|
|
if (stackTraces.length > 0) {
|
|
Logging.w(TAG, threadName + " stacktrace:");
|
|
for (StackTraceElement stackTrace : stackTraces) {
|
|
Logging.w(TAG, stackTrace.toString());
|
|
}
|
|
}
|
|
if (printNativeStackTrace) {
|
|
// Imitate output from debuggerd/tombstone so that stack trace can easily be symbolized with
|
|
// ndk-stack.
|
|
Logging.w(TAG, "*** *** *** *** *** *** *** *** *** *** *** *** *** *** *** ***");
|
|
Logging.w(TAG,
|
|
"pid: " + Process.myPid() + ", tid: " + threadInfo.tid + ", name: " + threadName
|
|
+ " >>> WebRTC <<<");
|
|
nativePrintStackTrace(threadInfo.tid);
|
|
}
|
|
}
|
|
|
|
/** Deprecated, use non-static version instead. */
|
|
@Deprecated
|
|
public static void printStackTraces() {
|
|
printStackTrace(staticNetworkThread, /* printNativeStackTrace= */ false);
|
|
printStackTrace(staticWorkerThread, /* printNativeStackTrace= */ false);
|
|
printStackTrace(staticSignalingThread, /* printNativeStackTrace= */ false);
|
|
}
|
|
|
|
/**
|
|
* Print the Java stack traces for the critical threads used by PeerConnectionFactory, namely;
|
|
* signaling thread, worker thread, and network thread. If printNativeStackTraces is true, also
|
|
* attempt to print the C++ stack traces for these threads.
|
|
*/
|
|
public void printInternalStackTraces(boolean printNativeStackTraces) {
|
|
printStackTrace(signalingThread, printNativeStackTraces);
|
|
printStackTrace(workerThread, printNativeStackTraces);
|
|
printStackTrace(networkThread, printNativeStackTraces);
|
|
}
|
|
|
|
@CalledByNative
|
|
private void onNetworkThreadReady() {
|
|
networkThread = ThreadInfo.getCurrent();
|
|
staticNetworkThread = networkThread;
|
|
Logging.d(TAG, "onNetworkThreadReady");
|
|
}
|
|
|
|
@CalledByNative
|
|
private void onWorkerThreadReady() {
|
|
workerThread = ThreadInfo.getCurrent();
|
|
staticWorkerThread = workerThread;
|
|
Logging.d(TAG, "onWorkerThreadReady");
|
|
}
|
|
|
|
@CalledByNative
|
|
private void onSignalingThreadReady() {
|
|
signalingThread = ThreadInfo.getCurrent();
|
|
staticSignalingThread = signalingThread;
|
|
Logging.d(TAG, "onSignalingThreadReady");
|
|
}
|
|
|
|
// Must be called at least once before creating a PeerConnectionFactory
|
|
// (for example, at application startup time).
|
|
private static native void nativeInitializeAndroidGlobals();
|
|
private static native void nativeInitializeFieldTrials(String fieldTrialsInitString);
|
|
private static native String nativeFindFieldTrialsFullName(String name);
|
|
private static native void nativeInitializeInternalTracer();
|
|
// Internal tracing shutdown, called to prevent resource leaks. Must be called after
|
|
// PeerConnectionFactory is gone to prevent races with code performing tracing.
|
|
private static native void nativeShutdownInternalTracer();
|
|
private static native boolean nativeStartInternalTracingCapture(String tracingFilename);
|
|
private static native void nativeStopInternalTracingCapture();
|
|
|
|
private static native PeerConnectionFactory nativeCreatePeerConnectionFactory(Context context,
|
|
Options options, long nativeAudioDeviceModule, long audioEncoderFactory,
|
|
long audioDecoderFactory, VideoEncoderFactory encoderFactory,
|
|
VideoDecoderFactory decoderFactory, long nativeAudioProcessor,
|
|
long nativeFecControllerFactory, long nativeNetworkControllerFactory,
|
|
long nativeNetworkStatePredictorFactory, long neteqFactory);
|
|
|
|
private static native long nativeCreatePeerConnection(long factory,
|
|
PeerConnection.RTCConfiguration rtcConfig, MediaConstraints constraints, long nativeObserver,
|
|
SSLCertificateVerifier sslCertificateVerifier);
|
|
private static native long nativeCreateLocalMediaStream(long factory, String label);
|
|
private static native long nativeCreateVideoSource(
|
|
long factory, boolean is_screencast, boolean alignTimestamps);
|
|
private static native long nativeCreateVideoTrack(
|
|
long factory, String id, long nativeVideoSource);
|
|
private static native long nativeCreateAudioSource(long factory, MediaConstraints constraints);
|
|
private static native long nativeCreateAudioTrack(long factory, String id, long nativeSource);
|
|
private static native boolean nativeStartAecDump(
|
|
long factory, int file_descriptor, int filesize_limit_bytes);
|
|
private static native void nativeStopAecDump(long factory);
|
|
private static native void nativeFreeFactory(long factory);
|
|
private static native long nativeGetNativePeerConnectionFactory(long factory);
|
|
private static native void nativeInjectLoggable(JNILogging jniLogging, int severity);
|
|
private static native void nativeDeleteLoggable();
|
|
private static native void nativePrintStackTrace(int tid);
|
|
private static native RtpCapabilities nativeGetRtpSenderCapabilities(
|
|
long factory, MediaStreamTrack.MediaType mediaType);
|
|
private static native RtpCapabilities nativeGetRtpReceiverCapabilities(
|
|
long factory, MediaStreamTrack.MediaType mediaType);
|
|
}
|