MP4文件格式

1.先说几个基本概念 
  Sample: 采样,对于音视频来说就是一个编码帧;Sample_count即总帧数,Sample_index即帧下标。 

                  在一个Mp4文件里面,所有Box处理的Samples都是严格按照帧序号排列的。
                  删除或者修改一帧,很多个Box里面的内容需要从新计算。

  Chunk: 块,一个Chunk包括一个或者多个同类型Samples,使用Chunk的目的是为了加快Sample数据访问效率; 
                在一个Chunk里面Sample顺序紧凑排列。 
  Tarck: 轨,是音频或者视频流信息定义的集合。一个轨包括一个或者多个Chunk。各轨之间有同步的问题。 
  Box(也叫atom): 盒,MP4文件各种信息定义的结构。一个MP4文件就是有很多个各种类型的Box组成的。
        Box有固定的格式 
        4Bytes             4Bytes              8Bytes               (Box_length-8)Bytes

      Box_length | Box_type | Box_long_length |        Box_data 

      a. Box_long_length 仅当Box_length=1时出现, 他定义超长Box, 超过32位数字的范围的就用64位表述 
      b. box_length 包含了自己和4字节的type。因此一般情况下Box的负载是Box_length-8字节。 
      Box有两大类,一种是Container Box,这种Box里可以嵌套别的Box。它的负载就是其他的Box。 
                                       下图加^的Box就是Container Box。 
                               另外一种就是单独的Box,里面定义某一些信息数据。 
  time_scale: 很多个Box里面都有time_scale,它定义了该媒体在1秒中内的采样刻度, 可以理解为一秒钟有多少个单元。 
  duration: 相对于time_scale的时间长度。真实的时长 = duration/time_scale。 
      例如我手头这个MP4文件,视频track的time_scale为25000. duration=9670000. 
      这样time = 9670000/25000 = 386.8s = 6m27s 
  不同Box定义的time_scale和duration可能是不同的,原因在于: 
      a. 文件时长和track时长本身就不一定是相同的,Mp4格式容许杂合多个不同长短不同起始时间的Tracks在同一个文件里面; 
      b. 一般mvhd,vmhd, smhd里面的time_scale,duration都不相同。所以一定注意要用同一组来计算。 
 
2.一个基本的MP4文件Box结构如下: 
    Root  (虚拟的没有这个box类型) 
     |----ftyp 
     |----moov^ 
     |     |----mvhd   
     |     |----trak^ 
     |     |     |----tkhd 
     |     |     |----mdia^ 
     |     |           |----mdhd 
     |     |           |----hdlr 
     |     |           |----minf^ 
     |     |                 |----vmhd/smhd 
     |     |                 |----dinf^ 
     |     |                 |     |----dref^ 
     |     |                 |           |----url 
     |     |                 |----stbl^ 
     |     |                       |----stsd^ 
     |     |                       |     |----mp4v/mp4a^   
     |     |                       |----stts 
     |     |                       |----stss 
     |     |                       |----stsc 
     |     |                       |----stsz 
     |     |                       |----stco 
     |     |----trak^ 
     |     |     |---- .... 
     |     |            .... 
     |     |
     |     |----udta^ 
     |           |---- meta^ 
     |                  |----hdlr 
     |                  |----ilist^   
     |----mdat 
     |----free 
 
 
3. 几个重要的Box,先总结如下,再各个叙述。 
   stbl Box是Mp4文件里面最复杂的Box. 有多种stxxBox, 其中st是Sample Table. 
   stsd: Sample Description, 具体音视频解码器的定义                
   stts: Time to Sample, 定义每个Sample的duration, 这个duration也是相对time_scale的单位 
   stss: Sync Sample, 定义同步Sample Index, 即I帧Sample的Index. 
   stsc: Sample to Chunk,定义Sample在多个Chunk里面的划分情况。 
   stsz: Sample siZe,定义每个Sample的字节长度 
   stco: Chunk Offset,定义各个Chunk在文件中的起始地址。 
  
   stts: Sample duration. 定义的方法类似于游程编码。 
         (Sample_number1, duration1), (Sample_number2, duration2), ... (Sample_numberN, durationN).
         一个游程对定义了相同duration的连续Sample个数。N定义了有多少个这种游程对。 
         最常见的一种情况是所有的Sample具有相同的duration, 这样stts: 
         stts.entry_count = 1,   
         stts.sample_couint[0] = Sample_Count; 
         stts.sample_delta[0] = duration. 
         当给定一个时间,找对应的sample_index时需要用到stts. 
 
   stss: I帧Sample Index. 
         由于视频编码帧间依赖性,不是从任意Samplea开始都可以连续解码的,只有从I帧的Sample处才可以。 
         stss定义了随机访问点的Index值。在做Seek的时候需要用到stss 
   
   stsc: Sample在Chunk里面的分布表。有4个主要参数。 
         a. uint32_t entry_count, 定义一个Track里面包含多少个Chunk组,注意这里不是Chunk count,而是 chunk组 count.  
            所谓chunk组是指那些包含相同个数Samples,而且这些Samples的Sample Description相同的Chunk集合,类似游程编码。  
         b. uint32_t *first_chunk, 定义每个Chunk组起始Chunk index 
         c. uint32_t *samples_per_chunk, 定义每个Chunk组内各个Chunk包含Sample个数 
         d. uint32_t *sample_description_index,定义每个Chunk组内各个Chunk所包含的Sample的解码描述。
          
          for example 
          inx     first_chunk   samples_per_chunk    sample_description_index 
           0              1                         13                                          1
           1              2                         12                                          1
           2           806                         9                                          1
          
          意义:有3个chunk组. 
                组0 有 (  2-1)个chunk (下标为 1), 每个chunk含13个samples。 组0有(2-1)*13帧 
                组1 有 (806-2)个chunk (下标为2..805), 每个chunk含12个Sample。组1有(806-2)*12帧 
                组2 有 1 个chunk* (下标为806), 这个chunk含9个samples。组2有1*9帧. 
                一个Chunk组内chunk数目chunk_count = first_chunk[n+1] - first_chunk[n]。    
                一般最后一个chunk组就只有一个chunk. 在stco里面有chunk count的定义。 
                如果不放心可以用那个数字来计算最后一个chunk组的chunk数目。 
          根据这个表,可以展开得到每一个Sample在Chunk中的分布。  
     
    stsz: 定义每个Sample长度,这个Box很大。 
          在计算某一个Sample的偏移地址时需要stsz。 
 
    stco/co64:定义每个Chunk相对于文件头的偏移地址。stco里面的地址是32位,co64里面的是64位,根据实际情况选用。 
           
4.计算 
     输入i_chunk, 输出inx是Chunk组下标      
    uint32_t sample_to_chunk_group_index(MP4_Box_data_trak_t *p_track, uint32_t i_chunk) {
        uint32_t i; 
        for(i=0; i<p_track->i_stsc_count-1; i++)  
            if(i_chunk >= p_track->i_stsc_first_chunk[i] && i_chunk < p_track->i_stsc_first_chunk[i+1]) break;
        return i; 
    } 
 
    // 展开stsc,计算得到每个Chunk的i_sample_first, i_sample_count,i_sample_description_index 和 i_offset
    unsigned int i_chunk; 
    uint32_t i_first = 0; 
    uint32_t i_chunk_group_index = 0; 
    for( i_chunk = 0; i_chunk < p_track->i_chunk_count; i_chunk++ ) { 
        mp4_chunk_t *ck = &p_track->chunk[i_chunk]; 
        ck->i_offset = p_track->i_co64_sample_offset[i_chunk]; 
        i_chunk_group_index = chunk_group_index(p_track, i_chunk);  // 输出Chunk组下标
        ck->i_sample_description_index = p_track->i_stsc_sample_description_index[i_chunk_group_index];
        ck->i_sample_count = p_track->i_stsc_samples_per_chunk[i_chunk_group_index];
        ck->i_sample_first = i_first; 
        i_first += p_track->i_stsc_samples_per_chunk[i_chunk_group_index]; 
    } 
 
// 由Sample_index计算chunk_index  
uint32_t sample_to_chunk(MP4_Box_data_trak_t *p_track, uint64_t sample) { 
    uint32_t i; 
    for(i=0; i<p_track->i_chunk_count-1; i++) 
        if(sample >= p_track->chunk[i].i_sample_first && sample < p_track->chunk[i+1].i_sample_first) break;
    return i; 

 
// 由Sample_index计算该Sample数据起始地址 
// sample起始地址 = 该sample所在Chunk偏移地址(stco) + 该sample在Chunk里面的偏移地址 (stsz) 
// sample在chunk里面的偏移地址 = 该chunk的第一帧到该帧之前的Sample长度和  
uint64_t smaple_to_offset(MP4_Box_data_trak_t *p_track, uint64_t sample) { 
    uint32_t chunk_inx = sample_to_chunk(p_track, sample); 
    uint64_t start_address = p_track->i_co64_sample_offset[chunk_inx]; 
    if(p_track->i_stsz_sample_size == 0) { 
        uint32_t i; 
        for(i=p_track->chunk[chunk_inx].i_sample_first; i<sample; i++) 
            start_address += p_track->i_stsz_entry_size[i]; 
    } else 
        start_address += (sample - p_track->chunk[chunk_inx].i_sample_first) * p_track->i_stsz_sample_size;
    return start_address; 

 
// time的单位为毫秒 
uint64_t time_to_duration(MP4_Box_data_trak_t *p_track, uint64_t time) { 
    return  time * p_track->i_timescale/1000; 
}   
 
// 由time(单位是毫秒)计算sample_index.  
// 根据stts计算,应该把每个Chunk第一个sample对应的duration一次性计算出来,这样查找起来会更快一些。 
uint64_t time_to_sample(MP4_Box_data_trak_t *p_track, uint64_t time) { 
    uint64_t i_start = time_to_duration(p_track, time); 
    if(i_start > p_track->i_duration) return 0; 
 
    uint32_t i; 
    uint64_t i_sample = 0; 
    if(p_track->i_stts_count == 1) { // all the sample has same duration 
        i = 0; 
    } else { 
        for(i=0; i<p_track->i_stts_count; i++) { 
            uint64_t range = p_track->i_stts_sample_count[i] * p_track->i_stts_sample_delta[i];
            if(range > i_start) break; 
            i_start -= range; 
            i_sample += p_track->i_stts_sample_count[i]; 
        } 
    } 
    i_sample += i_start / p_track->i_stts_sample_delta[i]; 
    if(i_sample > p_track->i_stsz_sample_count) i_sample = 0; 
 
    return i_sample; 
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值