generated from zhangwei/Matrixai
			
		
			
	
	
		
			155 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
		
		
			
		
	
	
			155 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
|  | // PHZ
 | |||
|  | // 2018-6-8
 | |||
|  | 
 | |||
|  | #ifndef XOP_MEDIA_SESSION_H
 | |||
|  | #define XOP_MEDIA_SESSION_H
 | |||
|  | 
 | |||
|  | #include <memory>
 | |||
|  | #include <string>
 | |||
|  | #include <vector>
 | |||
|  | #include <mutex>
 | |||
|  | #include <atomic>
 | |||
|  | #include <random>
 | |||
|  | #include <cstdint>
 | |||
|  | #include <unordered_set>
 | |||
|  | #include "media.h"
 | |||
|  | #include "H264Source.h"
 | |||
|  | #include "H265Source.h"
 | |||
|  | #include "VP8Source.h"
 | |||
|  | #include "G711ASource.h"
 | |||
|  | #include "AACSource.h"
 | |||
|  | #include "MediaSource.h"
 | |||
|  | #include "Socket.h"
 | |||
|  | #include "RingBuffer.h"
 | |||
|  | 
 | |||
|  | namespace xop | |||
|  | { | |||
|  | 
 | |||
|  | class RtpConnection; | |||
|  | 
 | |||
|  | class MediaSession | |||
|  | { | |||
|  | public: | |||
|  | 	using Ptr = std::shared_ptr<MediaSession>; | |||
|  | 	using NotifyConnectedCallback = std::function<void (MediaSessionId sessionId, std::string peer_ip, uint16_t peer_port)> ; | |||
|  | 	using NotifyDisconnectedCallback = std::function<void (MediaSessionId sessionId, std::string peer_ip, uint16_t peer_port)> ; | |||
|  | 
 | |||
|  | 	static MediaSession* CreateNew(std::string url_suffix="live"); | |||
|  | 	virtual ~MediaSession(); | |||
|  | 
 | |||
|  | 	bool AddSource(MediaChannelId channel_id, MediaSource* source); | |||
|  | 	bool RemoveSource(MediaChannelId channel_id); | |||
|  | 
 | |||
|  | 	bool StartMulticast(); | |||
|  | 
 | |||
|  | 	void AddNotifyConnectedCallback(const NotifyConnectedCallback& callback); | |||
|  | 	void AddNotifyDisconnectedCallback(const NotifyDisconnectedCallback& callback); | |||
|  | 
 | |||
|  | 	std::string GetRtspUrlSuffix() const | |||
|  | 	{ return suffix_; } | |||
|  | 
 | |||
|  | 	void SetRtspUrlSuffix(std::string& suffix) | |||
|  | 	{ suffix_ = suffix; } | |||
|  | 
 | |||
|  | 	std::string GetSdpMessage(std::string ip, std::string session_name =""); | |||
|  | 
 | |||
|  | 	MediaSource* GetMediaSource(MediaChannelId channel_id); | |||
|  | 
 | |||
|  | 	bool HandleFrame(MediaChannelId channel_id, AVFrame frame); | |||
|  | 
 | |||
|  | 	bool AddClient(SOCKET rtspfd, std::shared_ptr<RtpConnection> rtp_conn); | |||
|  | 	void RemoveClient(SOCKET rtspfd); | |||
|  | 
 | |||
|  | 	MediaSessionId GetMediaSessionId() | |||
|  | 	{ return session_id_; } | |||
|  | 
 | |||
|  | 	uint32_t GetNumClient() const | |||
|  | 	{ return (uint32_t)clients_.size(); } | |||
|  | 
 | |||
|  | 	bool IsMulticast() const | |||
|  | 	{ return is_multicast_; } | |||
|  | 
 | |||
|  | 	std::string GetMulticastIp() const | |||
|  | 	{ return multicast_ip_; } | |||
|  | 
 | |||
|  | 	uint16_t GetMulticastPort(MediaChannelId channel_id) const | |||
|  | 	{ | |||
|  | 		if (channel_id >= MAX_MEDIA_CHANNEL) { | |||
|  | 			return 0; | |||
|  | 		}          | |||
|  | 		return multicast_port_[channel_id]; | |||
|  | 	} | |||
|  | 
 | |||
|  | private: | |||
|  | 	friend class MediaSource; | |||
|  | 	friend class RtspServer; | |||
|  | 	MediaSession(std::string url_suffxx); | |||
|  | 
 | |||
|  | 	MediaSessionId session_id_ = 0; | |||
|  | 	std::string suffix_; | |||
|  | 	std::string sdp_; | |||
|  | 
 | |||
|  | 	std::vector<std::unique_ptr<MediaSource>> media_sources_; | |||
|  | 	std::vector<RingBuffer<AVFrame>> buffer_; | |||
|  | 
 | |||
|  | 	std::vector<NotifyConnectedCallback> notify_connected_callbacks_; | |||
|  | 	std::vector<NotifyDisconnectedCallback> notify_disconnected_callbacks_; | |||
|  | 	std::mutex mutex_; | |||
|  | 	std::mutex map_mutex_; | |||
|  | 	std::map<SOCKET, std::weak_ptr<RtpConnection>> clients_; | |||
|  | 
 | |||
|  | 	bool is_multicast_ = false; | |||
|  | 	uint16_t multicast_port_[MAX_MEDIA_CHANNEL]; | |||
|  | 	std::string multicast_ip_; | |||
|  | 	std::atomic_bool has_new_client_; | |||
|  | 
 | |||
|  | 	static std::atomic_uint last_session_id_; | |||
|  | }; | |||
|  | 
 | |||
|  | class MulticastAddr | |||
|  | { | |||
|  | public: | |||
|  | 	static MulticastAddr& instance() | |||
|  | 	{ | |||
|  | 		static MulticastAddr s_multi_addr; | |||
|  | 		return s_multi_addr; | |||
|  | 	} | |||
|  | 
 | |||
|  | 	std::string GetAddr() | |||
|  | 	{ | |||
|  | 		std::lock_guard<std::mutex> lock(mutex_); | |||
|  | 		std::string addr_str; | |||
|  | 		struct sockaddr_in addr = { 0 }; | |||
|  | 		std::random_device rd; | |||
|  | 
 | |||
|  | 		for (int n = 0; n <= 10; n++) { | |||
|  | 			uint32_t range = 0xE8FFFFFF - 0xE8000100; | |||
|  | 			addr.sin_addr.s_addr = htonl(0xE8000100 + (rd()) % range); | |||
|  | 			addr_str = inet_ntoa(addr.sin_addr); | |||
|  | 
 | |||
|  | 			if (addrs_.find(addr_str) != addrs_.end()) { | |||
|  | 				addr_str.clear(); | |||
|  | 			} | |||
|  | 			else { | |||
|  | 				addrs_.insert(addr_str); | |||
|  | 				break; | |||
|  | 			} | |||
|  | 		} | |||
|  | 
 | |||
|  | 		return addr_str; | |||
|  | 	} | |||
|  | 
 | |||
|  | 	void Release(std::string addr) { | |||
|  | 		std::lock_guard<std::mutex> lock(mutex_); | |||
|  | 		addrs_.erase(addr); | |||
|  | 	} | |||
|  | 
 | |||
|  | private: | |||
|  | 	std::mutex mutex_; | |||
|  | 	std::unordered_set<std::string> addrs_; | |||
|  | }; | |||
|  | 
 | |||
|  | } | |||
|  | 
 | |||
|  | #endif
 |