当前位置:   article > 正文

2、RTSP协议的实现_stm32网络摄像头rtsp

stm32网络摄像头rtsp

写在前面


此系列只追求精简,旨在学习RTSP协议的实现过程,不追求复杂完美,所以这里要实现的RTSP服务器为了简单,实现上同一时间只能有一个客户端,下面开始介绍实现过程

在写一个RTSP服务器之前,我们必须知道一个RTSP服务器最简单的包含两部分,一部分是RTSP的交互,一部分是RTP发送,本文先实现RTSP交互过程

一、创建套接字


想一下我们在vlc输入rtsp://127.0.0.1:8554后发生了什么事?

在这种情况下,vlc其实是一个rtsp客户端,当输入这个url后,vlc知道目的IP为127.0.0.1,目的端口号为8854,这时vlc会发起一个tcp连接取连接服务器,连接成功后就开始发送请求,服务端响应

所以我们要写一个rtsp服务器,第一步肯定是创建tcp服务器

首先创建tcp套接字,绑定端口,监听

  1. //1.创建套接字
  2. serverSockfd = socket(AF_INET, SOCK_STREAM, 0);
  3. setsockopt(serverSockfd, SOL_SOCKET, SO_REUSEADDR, (const char*)&on, sizeof(on));
  4. //2.绑定地址和端口号 这个示例绑定的地址是INADDR_ANY,端口号为8554
  5. bind(serverSockfd, (struct sockaddr *)&addr, sizeof(struct sockaddr)
  6. //3.开始监听
  7. listen(serverSockfd, 10);

RTSP服务器传输音视频数据和信息使用的是RTP和RTCP,所以我们还要为RTP和RTCP创建UDP套接字,并绑定号端口

  1. //1.创建套接字
  2. serverRtpSockfd = createUdpSocket();
  3. serverRtcpSockfd = createUdpSocket();
  4. //2.绑定端口号 当创建好套接字还有绑定号端口后,就可以接收客户端请求了
  5. bindSocketAddr(serverRtpSockfd, "0.0.0.0", SERVER_RTP_PORT);
  6. bindSocketAddr(serverRtcpSockfd, "0.0.0.0", SERVER_RTCP_PORT);
  7. //3.开始accept等待客户端连接
  8. clientfd = accept(serverSockfd, (struct sockaddr *)&addr, &len);

 

二、解析请求

当rtsp客户端连接成功后就会开始发送请求,服务器这是需要接收客户端请求并开始解析,再采取相应得操作

这里我们做得最简单,首先解析第一行得到方法,对于OPTIONSDESCRIBEPLAYTEARDOWN我们只解析CSeq。对于SETUP,我们讲client_port解析出来

所以我们要做的第一步就是解析请求中的信息

 

解析完请求命令后,接下来就是更具不同得方法做不同的响应了,如下

  1. /*
  2. * 作者:_JT_
  3. * 博客:https://blog.csdn.net/weixin_42462202
  4. */
  5. if(!strcmp(method, "OPTIONS"))
  6. {
  7. handleCmd_OPTIONS();
  8. }
  9. else if(!strcmp(method, "DESCRIBE"))
  10. {
  11. handleCmd_DESCRIBE();
  12. }
  13. else if(!strcmp(method, "SETUP"))
  14. {
  15. handleCmd_SETUP();
  16. }
  17. else if(!strcmp(method, "PLAY"))
  18. {
  19. handleCmd_PLAY();
  20. }
  21. else if(!strcmp(method, "TEARDOWN"))
  22. {
  23. handleCmd_TEARDOWN();
  24. }

 

三、OPTIONS响应

OPTIONS是客户端向服务端请求可用的方法,我们这里就向客户端回复我们当前可用的方法

  1. sprintf(sBuf, "RTSP/1.0 200 OK\r\n"
  2. "CSeq: %d\r\n"
  3. "Public: OPTIONS, DESCRIBE, SETUP, PLAY\r\n"
  4. "\r\n",
  5. cseq);
  6. send(clientSockfd, sBuf, strlen(sBuf));

四、DESCRIBE响应

DESCRIBE是客户端向服务器请求媒体信息,这是服务器需要回复sdp描述文件,这个例子中的媒体是H.264

  • sdp文件生成

  1. sprintf(sdp, "v=0\r\n"
  2. "o=- 9%ld 1 IN IP4 %s\r\n"
  3. "t=0 0\r\n"
  4. "a=control:*\r\n"
  5. "m=video 0 RTP/AVP 96\r\n"
  6. "a=rtpmap:96 H264/90000\r\n"
  7. "a=control:track0\r\n",
  8. time(NULL), localIp);

回复

  1. sprintf(sBuf, "RTSP/1.0 200 OK\r\n"
  2. "CSeq: %d\r\n"
  3. "Content-Base: %s\r\n"
  4. "Content-type: application/sdp\r\n"
  5. "Content-length: %d\r\n\r\n"
  6. "%s",
  7. cseq,
  8. url,
  9. strlen(sdp),
  10. sdp);
  11. send(clientSockfd, sBuf, strlen(sBuf));

五、SETUP响应

SETUP是客户端请求建立会话连接,并发送了客户端的RTP端口和RTCP端口,那么此时服务端需要回复服务端的RTP端口和RTCP端口

六、PLAY响应

PLAY时客户端向服务器请求播放,这时服务端回复完请求后就开始通过setup过程中创建的udp套接字发送RTP包

七、源码

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <stdint.h>
  4. #include <string.h>
  5. #include <sys/types.h>
  6. #include <sys/socket.h>
  7. #include <sys/socket.h>
  8. #include <netinet/in.h>
  9. #include <arpa/inet.h>
  10. #include <time.h>
  11. #define SERVER_PORT 8554
  12. #define SERVER_RTP_PORT 55532
  13. #define SERVER_RTCP_PORT 55533
  14. #define BUF_MAX_SIZE (1024*1024)
  15. static int createTcpSocket()
  16. {
  17. int sockfd;
  18. int on = 1;
  19. sockfd = socket(AF_INET, SOCK_STREAM, 0);
  20. if(sockfd < 0)
  21. return -1;
  22. setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (const char*)&on, sizeof(on));
  23. return sockfd;
  24. }
  25. static int createUdpSocket()
  26. {
  27. int sockfd;
  28. int on = 1;
  29. sockfd = socket(AF_INET, SOCK_DGRAM, 0);
  30. if(sockfd < 0)
  31. return -1;
  32. setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (const char*)&on, sizeof(on));
  33. return sockfd;
  34. }
  35. static int bindSocketAddr(int sockfd, const char* ip, int port)
  36. {
  37. struct sockaddr_in addr;
  38. addr.sin_family = AF_INET;
  39. addr.sin_port = htons(port);
  40. addr.sin_addr.s_addr = inet_addr(ip);
  41. if(bind(sockfd, (struct sockaddr *)&addr, sizeof(struct sockaddr)) < 0)
  42. return -1;
  43. return 0;
  44. }
  45. static int acceptClient(int sockfd, char* ip, int* port)
  46. {
  47. int clientfd;
  48. socklen_t len = 0;
  49. struct sockaddr_in addr;
  50. memset(&addr, 0, sizeof(addr));
  51. len = sizeof(addr);
  52. clientfd = accept(sockfd, (struct sockaddr *)&addr, &len);
  53. if(clientfd < 0)
  54. return -1;
  55. strcpy(ip, inet_ntoa(addr.sin_addr));
  56. *port = ntohs(addr.sin_port);
  57. return clientfd;
  58. }
  59. static char* getLineFromBuf(char* buf, char* line)
  60. {
  61. while(*buf != '\n')
  62. {
  63. *line = *buf;
  64. line++;
  65. buf++;
  66. }
  67. *line = '\n';
  68. ++line;
  69. *line = '\0';
  70. ++buf;
  71. return buf;
  72. }
  73. static int handleCmd_OPTIONS(char* result, int cseq)
  74. {
  75. sprintf(result, "RTSP/1.0 200 OK\r\n"
  76. "CSeq: %d\r\n"
  77. "Public: OPTIONS, DESCRIBE, SETUP, PLAY\r\n"
  78. "\r\n",
  79. cseq);
  80. return 0;
  81. }
  82. static int handleCmd_DESCRIBE(char* result, int cseq, char* url)
  83. {
  84. char sdp[500];
  85. char localIp[100];
  86. sscanf(url, "rtsp://%[^:]:", localIp);
  87. sprintf(sdp, "v=0\r\n"
  88. "o=- 9%ld 1 IN IP4 %s\r\n"
  89. "t=0 0\r\n"
  90. "a=control:*\r\n"
  91. "m=video 0 RTP/AVP 96\r\n"
  92. "a=rtpmap:96 H264/90000\r\n"
  93. "a=control:track0\r\n",
  94. time(NULL), localIp);
  95. sprintf(result, "RTSP/1.0 200 OK\r\nCSeq: %d\r\n"
  96. "Content-Base: %s\r\n"
  97. "Content-type: application/sdp\r\n"
  98. "Content-length: %d\r\n\r\n"
  99. "%s",
  100. cseq,
  101. url,
  102. strlen(sdp),
  103. sdp);
  104. return 0;
  105. }
  106. static int handleCmd_SETUP(char* result, int cseq, int clientRtpPort)
  107. {
  108. sprintf(result, "RTSP/1.0 200 OK\r\n"
  109. "CSeq: %d\r\n"
  110. "Transport: RTP/AVP;unicast;client_port=%d-%d;server_port=%d-%d\r\n"
  111. "Session: 66334873\r\n"
  112. "\r\n",
  113. cseq,
  114. clientRtpPort,
  115. clientRtpPort+1,
  116. SERVER_RTP_PORT,
  117. SERVER_RTCP_PORT);
  118. return 0;
  119. }
  120. static int handleCmd_PLAY(char* result, int cseq)
  121. {
  122. sprintf(result, "RTSP/1.0 200 OK\r\n"
  123. "CSeq: %d\r\n"
  124. "Range: npt=0.000-\r\n"
  125. "Session: 66334873; timeout=60\r\n\r\n",
  126. cseq);
  127. return 0;
  128. }
  129. static void doClient(int clientSockfd, const char* clientIP, int clientPort,
  130. int serverRtpSockfd, int serverRtcpSockfd)
  131. {
  132. char method[40];
  133. char url[100];
  134. char version[40];
  135. int cseq;
  136. int clientRtpPort, clientRtcpPort;
  137. char *bufPtr;
  138. char* rBuf = malloc(BUF_MAX_SIZE);
  139. char* sBuf = malloc(BUF_MAX_SIZE);
  140. char line[400];
  141. while(1)
  142. {
  143. int recvLen;
  144. recvLen = recv(clientSockfd, rBuf, BUF_MAX_SIZE, 0);
  145. if(recvLen <= 0)
  146. goto out;
  147. rBuf[recvLen] = '\0';
  148. printf("---------------C->S--------------\n");
  149. printf("%s", rBuf);
  150. /* 解析方法 */
  151. bufPtr = getLineFromBuf(rBuf, line);
  152. if(sscanf(line, "%s %s %s\r\n", method, url, version) != 3)
  153. {
  154. printf("parse err\n");
  155. goto out;
  156. }
  157. /* 解析序列号 */
  158. bufPtr = getLineFromBuf(bufPtr, line);
  159. if(sscanf(line, "CSeq: %d\r\n", &cseq) != 1)
  160. {
  161. printf("parse err\n");
  162. goto out;
  163. }
  164. /* 如果是SETUP,那么就再解析client_port */
  165. if(!strcmp(method, "SETUP"))
  166. {
  167. while(1)
  168. {
  169. bufPtr = getLineFromBuf(bufPtr, line);
  170. if(!strncmp(line, "Transport:", strlen("Transport:")))
  171. {
  172. sscanf(line, "Transport: RTP/AVP;unicast;client_port=%d-%d\r\n",
  173. &clientRtpPort, &clientRtcpPort);
  174. break;
  175. }
  176. }
  177. }
  178. if(!strcmp(method, "OPTIONS"))
  179. {
  180. if(handleCmd_OPTIONS(sBuf, cseq))
  181. {
  182. printf("failed to handle options\n");
  183. goto out;
  184. }
  185. }
  186. else if(!strcmp(method, "DESCRIBE"))
  187. {
  188. if(handleCmd_DESCRIBE(sBuf, cseq, url))
  189. {
  190. printf("failed to handle describe\n");
  191. goto out;
  192. }
  193. }
  194. else if(!strcmp(method, "SETUP"))
  195. {
  196. if(handleCmd_SETUP(sBuf, cseq, clientRtpPort))
  197. {
  198. printf("failed to handle setup\n");
  199. goto out;
  200. }
  201. }
  202. else if(!strcmp(method, "PLAY"))
  203. {
  204. if(handleCmd_PLAY(sBuf, cseq))
  205. {
  206. printf("failed to handle play\n");
  207. goto out;
  208. }
  209. }
  210. else
  211. {
  212. goto out;
  213. }
  214. printf("---------------S->C--------------\n");
  215. printf("%s", sBuf);
  216. send(clientSockfd, sBuf, strlen(sBuf), 0);
  217. }
  218. out:
  219. close(clientSockfd);
  220. free(rBuf);
  221. free(sBuf);
  222. }
  223. int main(int argc, char* argv[])
  224. {
  225. int serverSockfd;
  226. int serverRtpSockfd, serverRtcpSockfd;
  227. int ret;
  228. serverSockfd = createTcpSocket();
  229. if(serverSockfd < 0)
  230. {
  231. printf("failed to create tcp socket\n");
  232. return -1;
  233. }
  234. ret = bindSocketAddr(serverSockfd, "0.0.0.0", SERVER_PORT);
  235. if(ret < 0)
  236. {
  237. printf("failed to bind addr\n");
  238. return -1;
  239. }
  240. ret = listen(serverSockfd, 10);
  241. if(ret < 0)
  242. {
  243. printf("failed to listen\n");
  244. return -1;
  245. }
  246. serverRtpSockfd = createUdpSocket();
  247. serverRtcpSockfd = createUdpSocket();
  248. if(serverRtpSockfd < 0 || serverRtcpSockfd < 0)
  249. {
  250. printf("failed to create udp socket\n");
  251. return -1;
  252. }
  253. if(bindSocketAddr(serverRtpSockfd, "0.0.0.0", SERVER_RTP_PORT) < 0 ||
  254. bindSocketAddr(serverRtcpSockfd, "0.0.0.0", SERVER_RTCP_PORT) < 0)
  255. {
  256. printf("failed to bind addr\n");
  257. return -1;
  258. }
  259. printf("rtsp://127.0.0.1:%d\n", SERVER_PORT);
  260. while(1)
  261. {
  262. int clientSockfd;
  263. char clientIp[40];
  264. int clientPort;
  265. clientSockfd = acceptClient(serverSockfd, clientIp, &clientPort);
  266. if(clientSockfd < 0)
  267. {
  268. printf("failed to accept client\n");
  269. return -1;
  270. }
  271. printf("accept client;client ip:%s,client port:%d\n", clientIp, clientPort);
  272. doClient(clientSockfd, clientIp, clientPort, serverRtpSockfd, serverRtcpSockfd);
  273. }
  274. return 0;
  275. }

八、测试

编译运行源码,打开vlc,输入rtsp://127.0.0.1:8554,点击开始播放,可以看到控制台会打印出交互过程,或是用wireshak抓包

本篇文章到这里结束,至此完成了RTSP协议的交互部分,在PLAY之后并没有开始发送RTP包,所以暂时还看不到视频,究竟如何发送RTP包,请看下一篇文章
 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/46217
推荐阅读
相关标签
  

闽ICP备14008679号