(2条消息) 【Live555】live555源码详解(九):ServerMediaSession、ServerMediaSubsession、live555MediaServer

【Live555】live555源码详解系列笔记

继承协作关系图

下面红色表示本博客将要介绍的三个类所在的位置:
ServerMediaSession、ServerMediaSubsession、DynamicRTSPServer
DynamicRTSPServer是live555MediaServer中实现的类,用来创建RTSP服务器

17、ServerMediaSession

ServerMediaSession 继承自 Medium,继承关系图:

协作图:

class ServerMediaSession: public Medium {
public:
/ 创建服务端媒体会话 ServerMediaSession
  static ServerMediaSession* createNew(UsageEnvironment& env,
       char const* streamName = NULL,
       char const* info = NULL,
       char const* description = NULL,
       Boolean isSSM = False,
       char const* miscSDPLines = NULL);

/ 通过名字获取服务端媒体会话 ServerMediaSession
  static Boolean lookupByName(UsageEnvironment& env,
                              char const* mediumName,
                              ServerMediaSession*& resultSession);

/ 生成会话描述,注意:调用者负责释放返回的字符串
  char* generateSDPDescription(); 

/ 返回流名称,例如媒体文件名
  char const* streamName() const { return fStreamName; }

/ 添加子会话
  Boolean addSubsession(ServerMediaSubsession* subsession);

/ 返回子会话数量
  unsigned numSubsessions() const { return fSubsessionCounter; }

/ 将“scale”设置为实际支持的比例:RTSP PLAY命令中的scale,通过调整这个数值,可以用来控制播放速度,也用来实现回放
  void testScaleFactor(float& scale);

/ 结果== 0表示:无限制会话(默认值)
/ 结果<  0表示:子会话持续时间不同;结果是-(最大的)。
/ 结果>  0表示:这是有界会话的持续时间
  float duration() const;

/ 当客户端访问此媒体时调用。
/ 默认实现什么都不做,但是子类可以重新定义它——例如,如果你想从服务器上删除长期未使用的“ServerMediaSession”。
  virtual void noteLiveness();

/ 返回引用数量,当引用数为0时,才可以删除
  unsigned referenceCount() const { return fReferenceCount; }

/ 增加引用
  void incrementReferenceCount() { ++fReferenceCount; }

/ 减少引用
  void decrementReferenceCount() { if (fReferenceCount > 0) --fReferenceCount; }

/ 判读是否可以删除
  Boolean& deleteWhenUnreferenced() { return fDeleteWhenUnreferenced; }

/ 删除所有由“addSubsession()”添加的子会话,返回到“空”状态
/ 注意:如果你已经添加了这个"ServerMediaSession"到"RTSPServer",
/那么,在调用这个函数之前,你必须先通过调用"RTSPServer::closeAllClientSessionsForServerMediaSession()"来关闭使用它的客户端连接。
  void deleteAllSubsessions();

18、ServerMediaSubsession

ServerMediaSubsession 是一个纯虚类,继承自 Medium,继承关系图:

协作图:

class ServerMediaSubsession: public Medium {
public:
/ 返回子会话处理的是哪个轨道的数据(例如:音频、视频、字幕等)
  unsigned trackNumber() const { return fTrackNumber; }
/ 以字符串形式返回轨道信息:track%d
  char const* trackId();

/ 返回SDP媒体会话的所有描述内容
  virtual char const* sdpLines() = 0;

/ 获取流参数
  virtual void getStreamParameters(unsigned clientSessionId, // in
   netAddressBits clientAddress, // in
   Port const& clientRTPPort, // in
   Port const& clientRTCPPort, // in
   int tcpSocketNum, // in (-1 means use UDP, not TCP)
   unsigned char rtpChannelId, // in (used if TCP)
   unsigned char rtcpChannelId, // in (used if TCP)
   netAddressBits& destinationAddress, // in out
   u_int8_t& destinationTTL, // in out
   Boolean& isMulticast, // out
   Port& serverRTPPort, // out
   Port& serverRTCPPort, // out
   void*& streamToken // out
   ) = 0;

/ 启动流
  virtual void startStream(unsigned clientSessionId, void* streamToken,
   TaskFunc* rtcpRRHandler,
   void* rtcpRRHandlerClientData,
   unsigned short& rtpSeqNum,
   unsigned& rtpTimestamp,
   ServerRequestAlternativeByteHandler* serverRequestAlternativeByteHandler,
   void* serverRequestAlternativeByteHandlerClientData) = 0;

/ 暂停流
  virtual void pauseStream(unsigned clientSessionId, void* streamToken);

/ 跳帧播放
  virtual void seekStream(unsigned clientSessionId, void* streamToken, double& seekNPT,double streamDuration, u_int64_t& numBytes);
  virtual void seekStream(unsigned clientSessionId, void* streamToken, char*& absStart, char*& absEnd);

/ 在"PLAY"命令没有指定的开始时间时调用???
  virtual void nullSeekStream(unsigned clientSessionId, void* streamToken, double streamEndTime, u_int64_t& numBytes);

/ 设置播放速度
  virtual void setStreamScale(unsigned clientSessionId, void* streamToken, float scale);

/获取当前正常播放时间NPT(Normal playing time)
  virtual float getCurrentNPT(void* streamToken);

/ 获取源
  virtual FramedSource* getStreamSource(void* streamToken);

/ 返回指向“streamToken”的“RTPSink”和“RTCPInstance”对象的指针。
/ 例如,如果您想要获得关联的“Groupsock”对象,这就很有用。
/ 您不能删除这些对象,或开始/停止播放它们;相反,这是使用“startStream()”和“deleteStream()”函数完成的。
  virtual void getRTPSinkandRTCP(void* streamToken, RTPSink const*& rtpSink, RTCPInstance const*& rtcp) = 0;
  virtual void deleteStream(unsigned clientSessionId, void*& streamToken);

/ 将“scale”设置为实际支持的比例:RTSP PLAY命令中的scale,通过调整这个数值,可以用来控制播放速度,也用来实现回放
  virtual void testScaleFactor(float& scale); // sets "scale" to the actual supported scale

/ 返回值 = 0 :持续播放
/ 返回值 > 0 :播放持续时间
  virtual float duration() const;

/ 子类可以通过“绝对”时间重新实现它们所支持的。
  virtual void getAbsoluteTimeRange(char*& absStartTime, char*& absEndTime) const;

/ (例如)SIP服务器可以调用以下命令,将SDP描述中的地址和端口号字段设置为非零:
  void setServerAddressAndPortForSDP(netAddressBits addressBits,  portNumBits portBits);

19、UserAuthenticationDatabase

用于可选用户/密码身份验证的数据结构,协作关系图:

class UserAuthenticationDatabase {
public:

/ 如果"passwordsAreMD5"为真,则存储到数据库或从数据库中删除的每个密码实际上都是md5计算的值(<username>:<realm>:<实际-密码>)。
  UserAuthenticationDatabase(char const* realm = NULL, Boolean passwordsAreMD5 = False);
  virtual ~UserAuthenticationDatabase();

/ 添加用户记录
  virtual void addUserRecord(char const* username, char const* password);

/ 删除用户记录
  virtual void removeUserRecord(char const* username);

/ 获取密码,如果用户名不存在,则返回NULL
  virtual char const* lookupPassword(char const* username);

/ 默认值是"LIVE555 Streaming Media"。(不清楚什么作用)
  char const* realm() { return fRealm; }

/ 判读密码是否经过MD5加密
  Boolean passwordsAreMD5() { return fPasswordsAreMD5; }
}

live555MediaServer源码分析

1、主流程

1.1 创建任务调度器和环境

TaskScheduler* scheduler = BasicTaskScheduler::createNew();
UsageEnvironment* env = BasicUsageEnvironment::createNew(*scheduler);

1.2 创建 RTSPServer

UserAuthenticationDatabase* authDB = NULL;
portNumBits rtspServerPortNum = 554;
RTSPServer* rtspServer = DynamicRTSPServer::createNew(*env, rtspServerPortNum, authDB);

1.3 尝试为 RTSP-over-HTTP 隧道创建一个HTTP服务器。

rtspServer->setUpTunnelingOverHTTP(80) // 尝试其它端口号:8000 8080

1.4 启动任务调度的事件循环

env->taskScheduler().doEventLoop();

2、DynamicRTSPServer 类分析

DynamicRTSPServer 继承自 RTSPServer
主要接口:

createNew:静态函数,调用 GenericMediaServer::setUpOurSocket ,然后创建 DynamicRTSPServer

重新实现的虚函数:

lookupServerMediaSession:获取 ServerMediaSession
live555MediaServer是将当前目录下的媒体文件通过RTSPServer发布出去。
lookupServerMediaSession 中首先根据名字streamName判读文件是否存在,如果存在,搜索 ServerMediaSession 是否已经创建,
结合其它条件,执行删除、创建、添加 ServerMediaSession 的操作

静态函数:

createNewSMS:创建服务会话
首先使用 ServerMediaSession::createNew 新建会话 ServerMediaSession *sms;
然后根据文件名的后缀创建不同的子会话,添加到会话sms中:sms->addSubsession(ADTSAudioFileServerMediaSubsession::createNew(...))

对RTSP抓包,分析通讯流程

1、C–>S(客户端向服务端):OPTIONS命令查询服务器提供的方法
Request: OPTIONS rtsp://192.168.1.10:554/12 RTSP/1.0\r\n
CSeq: 2\r\n
User-Agent: LibVLC/2.2.8 (LIVE555 Streaming Media v2016.02.22)\r\n
\r\n

2、S–>C:回应OPTIONS
Response: RTSP/1.0 200 OK\r\n
CSeq: 2\r\n
Cache-Control: no-cache\r\n
Server: Hisilicon RTSP Streaming Media Server/1.0.0\r\n
Date: Thu, Jan 01 1970 00:02:47 GMT\r\n
Public: OPTIONS, DESCRIBE, SETUP, PLAY, TEARDOWN, SET_PARAMETER, GET_PARAMETER\r\n
\r\n

3、C–>S:DESCRIBE命令查询媒体文件的SDP(Session Description Protocol)会话描述信息
Request: DESCRIBE rtsp://192.168.1.10:554/12 RTSP/1.0\r\n
CSeq: 3\r\n
User-Agent: LibVLC/2.2.8 (LIVE555 Streaming Media v2016.02.22)\r\n
Accept: application/sdp\r\n //请求获得sdp信息
\r\n

4、S–>C (RTSP/SDP):回应DESCRIBE,并携带SDP信息
Real Time Streaming Protocol
Response: RTSP/1.0 200 OK\r\n
CSeq: 3\r\n
Content-type: application/sdp
Cache-Control: no-cache\r\n
Server: Hisilicon RTSP Streaming Media Server/1.0.0\r\n
Content-length: 380
Date: Thu, Jan 01 1970 00:02:47 GMT\r\n
\r\n

v=0 //协议版本
o=StreamingServer 3331435948 1116907222000 IN IP4 192.168.1.10 //所有者和会话标识符
s=12//会话名称
c=IN IP4 0.0.0.0
t=0 0
a=control:*
a=range:npt=0-
m=video 0 RTP/AVP 96//m:媒体行
a=framerate:30//a:属性行
a=transform:1,0,0;0,1,0;0,0,1
a=control:trackID=0
a=rtpmap:96 H264/90000
a=fmtp:96 packetization-mode=1;profile-level-id=42002A;sprop-parameter-sets=Z0IAKp41wPAET8s3AQEBAg==,aM48gA==
a=x-dimensions:1920,1080

5、C–>S:SETUP命令告诉服务器 客户端用于接受媒体数据的端口号
Request: SETUP rtsp://192.168.1.10:554/12/trackID=0 RTSP/1.0\r\n
CSeq: 4\r\n
User-Agent: LibVLC/2.2.8 (LIVE555 Streaming Media v2016.02.22)\r\n
Transport: RTP/AVP;unicast;client_port=51374-51375 //unicast表示单播,用以区别组播;传输参数,RTP使用偶数,RTCP使用奇数
\r\n

6、S–>C:回应SETUP;至此流媒体连接建立完成
Response: RTSP/1.0 200 OK\r\n
CSeq: 4\r\n
Session: 106932986252082
Server: Hisilicon RTSP Streaming Media Server/1.0.0\r\n
Date: Thu, Jan 01 1970 00:02:47 GMT\r\n
Transport: RTP/AVP;unicast;client_port=51374-51375;server_port=5000-5001;ssrc=23bdcc95;mode=“PLAY”
\r\n

7、C–>S:PLAY命令,告诉服务端开始传输媒体流数据。
Request: PLAY rtsp://192.168.1.10:554/12 RTSP/1.0\r\n
CSeq: 5\r\n
User-Agent: LibVLC/2.2.8 (LIVE555 Streaming Media v2016.02.22)\r\n
Session: 106932986252082
Range: npt=0.000-\r\n
\r\n

8、S–>C:回应PLAY后,开始传输媒体数据流
Response: RTSP/1.0 200 OK\r\n
Server: Hisilicon RTSP Streaming Media Server/1.0.0\r\n
CSeq: 5\r\n
Session: 106932986252082
Date: Date: Thu, Jan 01 1970 00:02:49 GMT\r\n
Range: npt=0.000-\r\n
RTP-Info: url=rtsp://192.168.1.10:554/12/trackID=0;seq=0;rtptime=0,\r\n
\r\n

9、IPv4 RTP

x、C–>S
Request: GET_PARAMETER rtsp://192.168.1.10:554/12 RTSP/1.0\r\n
CSeq: 6\r\n
User-Agent: LibVLC/2.2.8 (LIVE555 Streaming Media v2016.02.22)\r\n
Session: 106932986252082
\r\n

n、IPv4 RTP

n+1、C–>S:TEARDOWN命令,终止媒体流传输
Request: TEARDOWN rtsp://192.168.1.10:554/12 RTSP/1.0\r\n
Method: TEARDOWN
URL: rtsp://192.168.1.10:554/12
CSeq: 7\r\n
User-Agent: LibVLC/2.2.8 (LIVE555 Streaming Media v2016.02.22)\r\n
Session: 378514285583215
\r\n

n+2、C–>S:RTCP Goodbye

(0)

相关推荐