blob: d442811d9b460f5e055a5d8a4326367d1390b74f [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CONTENT_RENDERER_MEDIA_RTC_PEER_CONNECTION_HANDLER_H_
#define CONTENT_RENDERER_MEDIA_RTC_PEER_CONNECTION_HANDLER_H_
#include <stddef.h>
#include <map>
#include <memory>
#include <string>
#include "base/compiler_specific.h"
#include "base/macros.h"
#include "base/memory/ref_counted.h"
#include "base/memory/weak_ptr.h"
#include "base/single_thread_task_runner.h"
#include "base/threading/thread.h"
#include "base/threading/thread_checker.h"
#include "content/common/content_export.h"
#include "content/renderer/media/webrtc/media_stream_track_metrics.h"
#include "ipc/ipc_platform_file.h"
#include "third_party/WebKit/public/platform/WebMediaStreamSource.h"
#include "third_party/WebKit/public/platform/WebRTCPeerConnectionHandler.h"
#include "third_party/WebKit/public/platform/WebRTCStatsRequest.h"
#include "third_party/WebKit/public/platform/WebRTCStatsResponse.h"
namespace blink {
class WebFrame;
class WebRTCAnswerOptions;
class WebRTCDataChannelHandler;
class WebRTCLegacyStats;
class WebRTCOfferOptions;
class WebRTCPeerConnectionHandlerClient;
}
namespace content {
class PeerConnectionDependencyFactory;
class PeerConnectionTracker;
class RemoteMediaStreamImpl;
class RtcDataChannelHandler;
class WebRtcMediaStreamAdapter;
// Mockable wrapper for blink::WebRTCStatsResponse
class CONTENT_EXPORT LocalRTCStatsResponse
: public NON_EXPORTED_BASE(rtc::RefCountInterface) {
public:
explicit LocalRTCStatsResponse(const blink::WebRTCStatsResponse& impl)
: impl_(impl) {
}
virtual blink::WebRTCStatsResponse webKitStatsResponse() const;
virtual void addStats(const blink::WebRTCLegacyStats& stats);
protected:
~LocalRTCStatsResponse() override {}
// Constructor for creating mocks.
LocalRTCStatsResponse() {}
private:
blink::WebRTCStatsResponse impl_;
};
// Mockable wrapper for blink::WebRTCStatsRequest
class CONTENT_EXPORT LocalRTCStatsRequest
: public NON_EXPORTED_BASE(rtc::RefCountInterface) {
public:
explicit LocalRTCStatsRequest(blink::WebRTCStatsRequest impl);
// Constructor for testing.
LocalRTCStatsRequest();
virtual bool hasSelector() const;
virtual blink::WebMediaStreamTrack component() const;
virtual void requestSucceeded(const LocalRTCStatsResponse* response);
virtual scoped_refptr<LocalRTCStatsResponse> createResponse();
protected:
~LocalRTCStatsRequest() override;
private:
blink::WebRTCStatsRequest impl_;
};
// RTCPeerConnectionHandler is a delegate for the RTC PeerConnection API
// messages going between WebKit and native PeerConnection in libjingle. It's
// owned by WebKit.
// WebKit calls all of these methods on the main render thread.
// Callbacks to the webrtc::PeerConnectionObserver implementation also occur on
// the main render thread.
class CONTENT_EXPORT RTCPeerConnectionHandler
: NON_EXPORTED_BASE(public blink::WebRTCPeerConnectionHandler) {
public:
RTCPeerConnectionHandler(
blink::WebRTCPeerConnectionHandlerClient* client,
PeerConnectionDependencyFactory* dependency_factory);
~RTCPeerConnectionHandler() override;
// Destroy all existing RTCPeerConnectionHandler objects.
static void DestructAllHandlers();
void associateWithFrame(blink::WebFrame* frame);
// Initialize method only used for unit test.
bool InitializeForTest(
const blink::WebRTCConfiguration& server_configuration,
const blink::WebMediaConstraints& options,
const base::WeakPtr<PeerConnectionTracker>& peer_connection_tracker);
// blink::WebRTCPeerConnectionHandler implementation
bool initialize(const blink::WebRTCConfiguration& server_configuration,
const blink::WebMediaConstraints& options) override;
void createOffer(const blink::WebRTCSessionDescriptionRequest& request,
const blink::WebMediaConstraints& options) override;
void createOffer(const blink::WebRTCSessionDescriptionRequest& request,
const blink::WebRTCOfferOptions& options) override;
void createAnswer(const blink::WebRTCSessionDescriptionRequest& request,
const blink::WebMediaConstraints& options) override;
void createAnswer(const blink::WebRTCSessionDescriptionRequest& request,
const blink::WebRTCAnswerOptions& options) override;
void setLocalDescription(
const blink::WebRTCVoidRequest& request,
const blink::WebRTCSessionDescription& description) override;
void setRemoteDescription(
const blink::WebRTCVoidRequest& request,
const blink::WebRTCSessionDescription& description) override;
blink::WebRTCSessionDescription localDescription() override;
blink::WebRTCSessionDescription remoteDescription() override;
blink::WebRTCErrorType setConfiguration(
const blink::WebRTCConfiguration& configuration) override;
bool addICECandidate(const blink::WebRTCICECandidate& candidate) override;
bool addICECandidate(const blink::WebRTCVoidRequest& request,
const blink::WebRTCICECandidate& candidate) override;
virtual void OnaddICECandidateResult(const blink::WebRTCVoidRequest& request,
bool result);
bool addStream(const blink::WebMediaStream& stream,
const blink::WebMediaConstraints& options) override;
void removeStream(const blink::WebMediaStream& stream) override;
void getStats(const blink::WebRTCStatsRequest& request) override;
void getStats(
std::unique_ptr<blink::WebRTCStatsReportCallback> callback) override;
blink::WebRTCDataChannelHandler* createDataChannel(
const blink::WebString& label,
const blink::WebRTCDataChannelInit& init) override;
blink::WebRTCDTMFSenderHandler* createDTMFSender(
const blink::WebMediaStreamTrack& track) override;
void stop() override;
// Delegate functions to allow for mocking of WebKit interfaces.
// getStats takes ownership of request parameter.
virtual void getStats(const scoped_refptr<LocalRTCStatsRequest>& request);
// Asynchronously calls native_peer_connection_->getStats on the signaling
// thread. If the |track_id| is empty, the |track_type| parameter is ignored.
void GetStats(webrtc::StatsObserver* observer,
webrtc::PeerConnectionInterface::StatsOutputLevel level,
const std::string& track_id,
blink::WebMediaStreamSource::Type track_type);
// Tells the |client_| to close RTCPeerConnection.
// Make it virtual for testing purpose.
virtual void CloseClientPeerConnection();
// Start recording an event log.
void StartEventLog(IPC::PlatformFileForTransit file,
int64_t max_file_size_bytes);
// Stop recording an event log.
void StopEventLog();
protected:
webrtc::PeerConnectionInterface* native_peer_connection() {
return native_peer_connection_.get();
}
class Observer;
friend class Observer;
void OnSignalingChange(
webrtc::PeerConnectionInterface::SignalingState new_state);
void OnIceConnectionChange(
webrtc::PeerConnectionInterface::IceConnectionState new_state);
void OnIceGatheringChange(
webrtc::PeerConnectionInterface::IceGatheringState new_state);
void OnRenegotiationNeeded();
void OnAddStream(std::unique_ptr<RemoteMediaStreamImpl> stream);
void OnRemoveStream(
const scoped_refptr<webrtc::MediaStreamInterface>& stream);
void OnDataChannel(std::unique_ptr<RtcDataChannelHandler> handler);
void OnIceCandidate(const std::string& sdp, const std::string& sdp_mid,
int sdp_mline_index, int component, int address_family);
private:
// Record info about the first SessionDescription from the local and
// remote side to record UMA stats once both are set.
struct FirstSessionDescription {
FirstSessionDescription(const webrtc::SessionDescriptionInterface* desc);
bool audio = false;
bool video = false;
// If audio or video will use RTCP-mux (if there is no audio or
// video, then false).
bool rtcp_mux = false;
};
webrtc::SessionDescriptionInterface* CreateNativeSessionDescription(
const std::string& sdp, const std::string& type,
webrtc::SdpParseError* error);
// Report to UMA whether an IceConnectionState has occurred. It only records
// the first occurrence of a given state.
void ReportICEState(
webrtc::PeerConnectionInterface::IceConnectionState new_state);
// Reset UMA related members to the initial state. This is invoked at the
// constructor as well as after Ice Restart.
void ResetUMAStats();
void ReportFirstSessionDescriptions(
const FirstSessionDescription& local,
const FirstSessionDescription& remote);
// Virtual to allow mocks to override.
virtual scoped_refptr<base::SingleThreadTaskRunner> signaling_thread() const;
void RunSynchronousClosureOnSignalingThread(const base::Closure& closure,
const char* trace_event_name);
base::ThreadChecker thread_checker_;
// |client_| is a weak pointer to the blink object (blink::RTCPeerConnection)
// that owns this object.
// It is valid for the lifetime of this object.
blink::WebRTCPeerConnectionHandlerClient* const client_;
// True if this PeerConnection has been closed.
// After the PeerConnection has been closed, this object may no longer
// forward callbacks to blink.
bool is_closed_;
// |dependency_factory_| is a raw pointer, and is valid for the lifetime of
// RenderThreadImpl.
PeerConnectionDependencyFactory* const dependency_factory_;
blink::WebFrame* frame_ = nullptr;
ScopedVector<WebRtcMediaStreamAdapter> local_streams_;
base::WeakPtr<PeerConnectionTracker> peer_connection_tracker_;
MediaStreamTrackMetrics track_metrics_;
// Counter for a UMA stat reported at destruction time.
int num_data_channels_created_ = 0;
// Counter for number of IPv4 and IPv6 local candidates.
int num_local_candidates_ipv4_ = 0;
int num_local_candidates_ipv6_ = 0;
// To make sure the observers are released after native_peer_connection_,
// they have to come first.
scoped_refptr<Observer> peer_connection_observer_;
scoped_refptr<webrtc::UMAObserver> uma_observer_;
// |native_peer_connection_| is the libjingle native PeerConnection object.
scoped_refptr<webrtc::PeerConnectionInterface> native_peer_connection_;
// The last applied configuration. Used so that the constraints
// used when constructing the PeerConnection carry over when
// SetConfiguration is called.
webrtc::PeerConnectionInterface::RTCConfiguration configuration_;
// Record info about the first SessionDescription from the local and
// remote side to record UMA stats once both are set. We only check
// for the first offer or answer. "pranswer"s and "unknown"s (from
// unit tests) are ignored.
std::unique_ptr<FirstSessionDescription> first_local_description_;
std::unique_ptr<FirstSessionDescription> first_remote_description_;
std::map<webrtc::MediaStreamInterface*,
std::unique_ptr<content::RemoteMediaStreamImpl>>
remote_streams_;
base::TimeTicks ice_connection_checking_start_;
// Track which ICE Connection state that this PeerConnection has gone through.
bool ice_state_seen_[webrtc::PeerConnectionInterface::kIceConnectionMax] = {};
base::WeakPtrFactory<RTCPeerConnectionHandler> weak_factory_;
DISALLOW_COPY_AND_ASSIGN(RTCPeerConnectionHandler);
};
} // namespace content
#endif // CONTENT_RENDERER_MEDIA_RTC_PEER_CONNECTION_HANDLER_H_