123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210 |
- /**
- * Copyright (c) 2019 Paul-Louis Ageneau
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 2.1 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- */
- #ifndef RTC_PEER_CONNECTION_H
- #define RTC_PEER_CONNECTION_H
- #include "candidate.hpp"
- #include "configuration.hpp"
- #include "datachannel.hpp"
- #include "description.hpp"
- #include "include.hpp"
- #include "init.hpp"
- #include "message.hpp"
- #include "reliability.hpp"
- #include "rtc.hpp"
- #include "track.hpp"
- #include <atomic>
- #include <functional>
- #include <future>
- #include <list>
- #include <mutex>
- #include <shared_mutex>
- #include <thread>
- #include <unordered_map>
- namespace rtc {
- class Certificate;
- class Processor;
- class IceTransport;
- class DtlsTransport;
- class SctpTransport;
- using certificate_ptr = std::shared_ptr<Certificate>;
- using future_certificate_ptr = std::shared_future<certificate_ptr>;
- struct DataChannelInit {
- Reliability reliability = {};
- bool negotiated = false;
- std::optional<uint16_t> id = nullopt;
- string protocol = "";
- };
- class PeerConnection final : public std::enable_shared_from_this<PeerConnection> {
- public:
- enum class State : int {
- New = RTC_NEW,
- Connecting = RTC_CONNECTING,
- Connected = RTC_CONNECTED,
- Disconnected = RTC_DISCONNECTED,
- Failed = RTC_FAILED,
- Closed = RTC_CLOSED
- };
- enum class GatheringState : int {
- New = RTC_GATHERING_NEW,
- InProgress = RTC_GATHERING_INPROGRESS,
- Complete = RTC_GATHERING_COMPLETE
- };
- enum class SignalingState : int {
- Stable = RTC_SIGNALING_STABLE,
- HaveLocalOffer = RTC_SIGNALING_HAVE_LOCAL_OFFER,
- HaveRemoteOffer = RTC_SIGNALING_HAVE_REMOTE_OFFER,
- HaveLocalPranswer = RTC_SIGNALING_HAVE_LOCAL_PRANSWER,
- HaveRemotePranswer = RTC_SIGNALING_HAVE_REMOTE_PRANSWER,
- } rtcSignalingState;
- PeerConnection();
- PeerConnection(const Configuration &config);
- ~PeerConnection();
- void close();
- const Configuration *config() const;
- State state() const;
- GatheringState gatheringState() const;
- SignalingState signalingState() const;
- bool hasLocalDescription() const;
- bool hasRemoteDescription() const;
- bool hasMedia() const;
- std::optional<Description> localDescription() const;
- std::optional<Description> remoteDescription() const;
- std::optional<string> localAddress() const;
- std::optional<string> remoteAddress() const;
- bool getSelectedCandidatePair(Candidate *local, Candidate *remote);
- void setLocalDescription(Description::Type type = Description::Type::Unspec);
- void setRemoteDescription(Description description);
- void addRemoteCandidate(Candidate candidate);
- std::shared_ptr<DataChannel> addDataChannel(string label, DataChannelInit init = {});
- // Equivalent to calling addDataChannel() and setLocalDescription()
- std::shared_ptr<DataChannel> createDataChannel(string label, DataChannelInit init = {});
- void onDataChannel(std::function<void(std::shared_ptr<DataChannel> dataChannel)> callback);
- void onLocalDescription(std::function<void(Description description)> callback);
- void onLocalCandidate(std::function<void(Candidate candidate)> callback);
- void onStateChange(std::function<void(State state)> callback);
- void onGatheringStateChange(std::function<void(GatheringState state)> callback);
- void onSignalingStateChange(std::function<void(SignalingState state)> callback);
- // Stats
- void clearStats();
- size_t bytesSent();
- size_t bytesReceived();
- std::optional<std::chrono::milliseconds> rtt();
- // Track media support requires compiling with libSRTP
- std::shared_ptr<Track> addTrack(Description::Media description);
- void onTrack(std::function<void(std::shared_ptr<Track> track)> callback);
- private:
- std::shared_ptr<IceTransport> initIceTransport(Description::Role role);
- std::shared_ptr<DtlsTransport> initDtlsTransport();
- std::shared_ptr<SctpTransport> initSctpTransport();
- void closeTransports();
- void endLocalCandidates();
- bool checkFingerprint(const std::string &fingerprint) const;
- void forwardMessage(message_ptr message);
- void forwardMedia(message_ptr message);
- void forwardBufferedAmount(uint16_t stream, size_t amount);
- std::optional<std::string> getMidFromSsrc(uint32_t ssrc);
- std::shared_ptr<DataChannel> emplaceDataChannel(Description::Role role, string label,
- DataChannelInit init);
- std::shared_ptr<DataChannel> findDataChannel(uint16_t stream);
- void iterateDataChannels(std::function<void(std::shared_ptr<DataChannel> channel)> func);
- void openDataChannels();
- void closeDataChannels();
- void remoteCloseDataChannels();
- void incomingTrack(Description::Media description);
- void openTracks();
- void validateRemoteDescription(const Description &description);
- void processLocalDescription(Description description);
- void processLocalCandidate(Candidate candidate);
- void processRemoteDescription(Description description);
- void processRemoteCandidate(Candidate candidate);
- void triggerDataChannel(std::weak_ptr<DataChannel> weakDataChannel);
- void triggerTrack(std::shared_ptr<Track> track);
- bool changeState(State state);
- bool changeGatheringState(GatheringState state);
- bool changeSignalingState(SignalingState state);
- void resetCallbacks();
- void outgoingMedia(message_ptr message);
- const init_token mInitToken = Init::Token();
- const Configuration mConfig;
- const future_certificate_ptr mCertificate;
- const std::unique_ptr<Processor> mProcessor;
- std::optional<Description> mLocalDescription, mRemoteDescription;
- std::optional<Description> mCurrentLocalDescription;
- mutable std::mutex mLocalDescriptionMutex, mRemoteDescriptionMutex;
- std::shared_ptr<IceTransport> mIceTransport;
- std::shared_ptr<DtlsTransport> mDtlsTransport;
- std::shared_ptr<SctpTransport> mSctpTransport;
- std::unordered_map<uint16_t, std::weak_ptr<DataChannel>> mDataChannels; // by stream ID
- std::unordered_map<string, std::weak_ptr<Track>> mTracks; // by mid
- std::vector<std::weak_ptr<Track>> mTrackLines; // by SDP order
- std::shared_mutex mDataChannelsMutex, mTracksMutex;
- std::unordered_map<uint32_t, string> mMidFromSsrc; // cache
- std::atomic<State> mState;
- std::atomic<GatheringState> mGatheringState;
- std::atomic<SignalingState> mSignalingState;
- std::atomic<bool> mNegotiationNeeded;
- synchronized_callback<std::shared_ptr<DataChannel>> mDataChannelCallback;
- synchronized_callback<Description> mLocalDescriptionCallback;
- synchronized_callback<Candidate> mLocalCandidateCallback;
- synchronized_callback<State> mStateChangeCallback;
- synchronized_callback<GatheringState> mGatheringStateChangeCallback;
- synchronized_callback<SignalingState> mSignalingStateChangeCallback;
- synchronized_callback<std::shared_ptr<Track>> mTrackCallback;
- };
- } // namespace rtc
- std::ostream &operator<<(std::ostream &out, rtc::PeerConnection::State state);
- std::ostream &operator<<(std::ostream &out, rtc::PeerConnection::GatheringState state);
- std::ostream &operator<<(std::ostream &out, rtc::PeerConnection::SignalingState state);
- #endif
|