媒體格式分析之flv -- 基於FFMPEG

轉自:http://www.cnblogs.com/qingquan/archive/2011/08/11/2135531.html


FLV 是FLASH VIDEO的簡稱,FLV流媒體格式是隨着Flash MX的推出發展而來的視頻格式。由於它形成的文件極小、加載速度極快,使得網絡觀看視頻文件成爲可能.當前主流的媒體網站像國內的優酷、國外youtube其標清格式的文件均採用flv的格式。

FLV文件結構解析

FLV是一個二進制文件,其文件格式如下圖 ,由文件頭(FLV header)和很多tag組成。tag又可以分成三類:audio,video,script,分別代表音頻流,視頻流,腳本流(關鍵字或者文件信息之類)。

flv_struct

FLV Header

FLV的Header信息一般比較簡單,包括文件類型之類的全局信息。如下圖中解析:

flv_header 

文件類型3bytes 總是FLV(0x46 0x4C 0x56),否則就不是在ffmpeg中在沒有指定文件格式的情況下,也是通過這個字段來探測文件是否屬於FLV格式的。

版本1byte 一般是0x01,表示FLV version 1

流信息1byte 倒數第一bit是1表示有視頻,倒數第三bit是1表示有音頻,其他都應該是0(有些軟件如flvtool2可能造成倒數第四bit是1,不過也沒發現有什麼不對)

header長度4bytes 整個文件頭的長度,一般是9(3+1+1+4),當然頭部字段也有可能包含其它信息這個時間其長度就不是9了。

FLV Body

FLV body就是由很多tag組成的,一個tag包括下列信息:

      previoustagsize 4bytes 前一個tag的長度,第一個tag就是0

tag類型1byte 共分爲三類:

* 8 -- 音頻tag

* 9 -- 視頻tag

* 18 -- 腳本tag

數據區長度3bytes 時間戳3bytes 單位毫秒同時還有1bytes的擴展時間戳,放在最高位,大部分時間時間戳爲媒體的dts信息,如果是腳本tag就是0

streamsID 3bytes 總是0(不知道幹啥用)

數據區:根據不同的tag類型就有不同的數據區

腳本tag :

腳本tag一般是用文本方式表示,如下圖flv的metadata信息:

flv_16

從中可以看出是通過文本的方式來標記的,其解析後其header信息爲:

meta_data_header

從中可以看出其type爲18。time stamp爲0.data size爲33638.

metadata tag data信息解析後爲:

metadata

其中有一些媒體信息:

例如視頻的:高和寬它的codec id。幀率。音頻的信息例如:音頻的sample rate,codec id,sample size及是否立體聲。還有整個文件的大小等等。

音頻的tag信息:

音頻的tag信息如下圖:

audio tag

其中time stamp 爲0是因爲其爲第一個音頻tag.

視頻tag

video

這是文件中的第6個tag所以其time stamp不爲0。因爲其爲視頻tag所以其type爲9。

ffmpeg中的flv文件格式解析的實現:

其中flv_read_header主要是從文件中讀取一些頭信息,同時作一些初始化化的工作

static int flv_read_header(AVFormatContext *s,AVFormatParameters *ap) 
{

         ……

    url_fskip(s->pb, 4); //將flv的頭去掉。 
    flags = get_byte(s->pb);//讀出flv的video和audio flag信息。

         ……  
        if(flags & FLV_HEADER_FLAG_HASVIDEO){ 
        if(!create_stream(s, 0))  //創建視頻流 
            return AVERROR(ENOMEM); 
    } 
    if(flags & FLV_HEADER_FLAG_HASAUDIO){ 
        if(!create_stream(s, 1)) //創建音頻流 
            return AVERROR(ENOMEM); 
    }

    offset = get_be32(s->pb); //獲取文件頭長度

        …… 
}

其它tag的讀取:

static int flv_read_packet(AVFormatContext *s, AVPacket *pkt) 
{

     …… 
for(;;url_fskip(s->pb, 4)){ /* pkt size is repeated at end. skip it */ 
    pos = url_ftell(s->pb); 
    type = get_byte(s->pb); //獲取tag的類型,前面已經提到flv的tag大概有以下三種 :FLV_TAG_TYPE_AUDIO = 0x08,FLV_TAG_TYPE_VIDEO = 0x09,FLV_TAG_TYPE_META  = 0x12, 
    size = get_be24(s->pb);//獲取tag的長度 
    dts = get_be24(s->pb); 
    dts |= get_byte(s->pb) << 24; //計算tag的timestamp也就是dts信息

    …… 
    if (type == FLV_TAG_TYPE_AUDIO) { //判斷是否爲audio tag

       …… 
         } else if (type == FLV_TAG_TYPE_VIDEO) {//判斷是否爲video tag

       …… 
               if ((flags & 0xf0) == 0x50) /* video info / command frame */ 
            goto skip; 
    } else { 
        if (type == FLV_TAG_TYPE_META && size > 13+1+4)//判斷是否爲meta tag,如果是meta信息則會將信息存放在一個map表中。 
         ……

}


發佈了28 篇原創文章 · 獲贊 6 · 訪問量 16萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章