本身動手寫RTP服務器—用RTP協議傳輸TS流

預備知識

關於TS流的格式:TS流封裝的具體格式請參考文檔ISO/IEC 13818-1。這裏咱們只須要了解一些簡單的信息就好。首先TS流是有許多的TS Packet組成的,每一個TS Packet的長度固定爲188 bytes,每一個packet都是以sync_byte:0x47開頭。服務器

MTU(Maximum Transmission Unit): 最大傳輸單元。是指一種通訊協議的某一層上面所能經過的最大數據包大小(以字節爲單位)。最大傳輸單元這個參數一般與通訊接口有關(網絡接口卡、串口等)。例如:太網沒法接收大於1500 字節的數據包。網絡

參考代碼

下面我會把本身寫的簡單的代碼貼出來,而且一步步地說明。socket

新建main.c文件,內容以下:ide

  1. #include <stdio.h>  
  2. #include <string.h>  
  3. #include <sys/types.h>  
  4. #include <sys/socket.h>  
  5. #include <netinet/in.h>  
  6.   
  7. #define TS_PACKET_SIZE 188  
  8. #define MTU 1500  
說明:包含一些必要的頭文件,而且定義了TS Packet的長度(188 bytes),MTU的限制(1500 bytes)。
  1. struct rtp_header{  
  2.     unsigned char cc:4;  
  3.     unsigned char x:1;  
  4.     unsigned char p:1;  
  5.     unsigned char v:2;  
  6.       
  7.     unsigned char pt:7;  
  8.     unsigned char m:1;  
  9.   
  10.     unsigned short sequence_number;  
  11.     unsigned int timestamp;  
  12.     unsigned int ssrc;  
  13. };  
  14.   
  15. void init_rtp_header(struct rtp_header *h){  
  16.     h->v = 2;  
  17.     h->p = 0;  
  18.     h->x = 0;  
  19.     h->cc = 0;  
  20.     h->m = 0;  
  21.     h->pt = 33;  
  22.     h->sequence_number = 123;  
  23.     h->timestamp = 123;  
  24.     h->ssrc = 123;  
  25. }  
說明:這裏定義了RTP Header的結構體,以及初始化的方法。這裏用到了位域,這是實現協議的時候經常會用到的方法。

須要注意的是:函數

你會發現這裏定義RTP Header的時候,上一篇講到的具體順序不一樣。緣由是本機和網絡字節流的順序相反,若是按照v p x cc的順序來定義一個byte,在這個byte內部v p x cc就會按照從低位到高位的順序放置;而在RTP流中,應該是順序從高位到低位放置的。因此每一個byte我都把順序作了倒置。測試

初始化RTP Header的函數的初始化值的意義請參考rfc3550。爲了實現簡單,其中的sequence_number、timestamp、ssrc,都是隨意填寫的。在發送包的時候須要將sequence_number遞增。大數據

  1. void sequence_number_increase(struct rtp_header *header){  
  2.     unsigned short sequence = ntohs(header->sequence_number);  
  3.     sequence++;  
  4.     header->sequence_number = htons(sequence);  
  5. }  
說明:這個函數的目的就是讓sequence_number加一,仍是因爲本機與網絡字節序不一樣的緣由,因此顯得略微複雜些。
  1. int main(){  
  2.     // RTP Packet we will send  
  3.     char buf[MTU];  
  4.     unsigned int count = 0;  
  5.   
  6.     // Init RTP Header  
  7.     init_rtp_header((struct rtp_header*)buf);  
  8.     count = sizeof(struct rtp_header);  
  9.   
  10.     // Init socket  
  11.     int sock = socket(AF_INET, SOCK_DGRAM, 0);  
  12.     struct sockaddr_in dest_addr;  
  13.       
  14.     dest_addr.sin_family=AF_INET;  
  15.     dest_addr.sin_port = htons(6666);  
  16.     dest_addr.sin_addr.s_addr = INADDR_ANY;  
  17.     bzero(&(dest_addr.sin_zero),8);  
  18.       
  19.     // Open TS file  
  20.     FILE *ts_file = fopen("/home/baby/Videos/480p.ts""r+");  
說 明:終於到了main函數了,main函數的開始很簡單,四個部分的初始化:表明RTP Packet的buffer,RTP Header,Socket,TS流媒體文件。若是你手頭沒有現成的TS文件,能夠用ffmpeg轉碼獲得一個ts文件:「ffmpeg -i video.xxx video.ts」, 其中 video.xxx 表示輸入的視頻文件,video.ts 爲輸出的TS文件。
  1. while(!feof(ts_file)){  
  2.     int read_len = fread(buf+count, 1, TS_PACKET_SIZE, ts_file);  
  3.     if(*(buf+count) != 0x47){  
  4.         fprintf(stderr, "Bad sync header!\n");  
  5.         continue;  
  6.     }  
  7.     count += read_len;  
  8.       
  9.     if (count + TS_PACKET_SIZE > MTU){// We should send  
  10.         sequence_number_increase((struct rtp_header*)buf);  
  11.         sendto(sock, buf, count, 0, (const struct sockaddr*)&dest_addr, sizeof(dest_addr));  
  12.         count = sizeof(struct rtp_header);  
  13.         usleep(10000);  
  14.     }  
  15. }  
  16.   
  17. fclose(ts_file);  
說 明:一切就緒後就能夠不斷的用UDP發送RTP Packet了。每次從ts_file中讀取188 bytes,附加到buf以後,若是buf的長度還沒用到達MTU的限制,那麼就繼續添加,不然就將buf發送出去。每次發送會將 sequence_number加一,而且間隔10000微秒。固然這只是個簡單的例子,實際發送視頻是要根據時間戳的。

測試

短短几十行代碼是否就能完成一個RTP服務器?咱們須要用實驗來驗證。spa

個人測試環境是Linux,用gcc編譯經過,使用VLC(MPlayer 測試也能夠經過了)做爲接收端。.net

首先啓動咱們的發送端程序,而後再執行「vlc rtp://127.0.0.1:6666」,等待幾秒後,發現真的能夠進行播放啦!ssr

相關文章
相關標籤/搜索