灵活宏块顺序--FMO(flexible Order)

来源:互联网 发布:natlab用不了遗传算法 编辑:程序博客网 时间:2024/06/09 12:36

        FMO(flexible Macroblock Order)意即灵活宏块顺序,应用在baseline profile和extended profile。
        FMO是对Slice而言,Slice是一个独立的预测编码单元,假如你愿意,还可以作为一个独立传输单元,一个Slice的宏块不能采用另一个Slice的宏块预测,当一个Slice出现问题,另一个Slice还可以独立解码,这样子就不会导致误差扩散,通常情况下,H264对一个PIC编码,采取从上到下,从左到右的自然宏块光栅扫描顺序进行编码。

        如果启用FMO功能,那么JM,将打乱宏块自然顺序,采用某种乱序方式,通过宏块映射方式,分配到不同的Slice中间。

灵活宏块次序(FMO),它可用灵活的方法,把编码MB序列映射到解码图象中MB的分配用MB到片组之间的映射来确定,它表示每一个MB属于哪个片组。以下为MB到片组的各种映射类型。
0   交错   MB游程被依次分配给每一块组
1  散乱    每一片组中的MB被分散在整个图象中
2  前景和背景 
3  Box-out  从帧的中心开始,产生一个箱子,其MB属于片组0,其它MB属于片组
4 光栅扫描  片组0包含按光栅扫描次序从顶-左的所有MB,其余MB属片组1
5 手绢  片组0包含从顶-左垂直扫描次序的MB,其余MB属片组1
6 显式  每一Mbslice_group_id,用于指明它的片组(即MB映射完全是用户定义的)

        FMO映射划分图像的模式各种各样,重要的有棋盘模式、矩形模式等。FMO打乱了正常的宏块编号,也就是可能把相邻的宏块分开,干扰了预测机制,增大了编码延时,但是增强了码流的健壮性,在恶劣的传输信道里,误码率较高的情况下表现依然良好。

        下面对JM采用的各种宏块分组模式映射到Slice机制进行说明,首先打开JM支持FMO的选项,改配置为如下,JM最大支持8个Slice。
num_slice_groups_minus1 = 1
                  # Number of Slice Groups Minus 1,   0 == no FMO, 1 == two slice  groups, etc.
slice_group_map_type    = 0
                     # 0:  Interleave,
                     # 1:   Dispersed,  
                     # 2:   Foreground with left-over,前景,最突出的位置
                     # 3:  Box-out,  
                     # 4:  Raster Scan 
                     # 5:  Wipe
                     # 6:  Explicit, slice_group_id read from SliceGroupConfigFileName
此时,JM对一个PIC会映射出两个Slice,映射机制按照interleave方式,每个Slice多少MB,请读配置文件sg0conf.cfg.
Interleave就是Slice交积方式,
|-------------------------|
|XX0 Slice 0    |
|-------------------------|
|XX1 Slice 1    |
|-------------------------|
|XX2 Slice 0    |
|-------------------------|
|XX3 Slice 1    |
|-------------------------|
编码顺序Slice0-->Slice1,假如Slice1第一个宏块序号是XX1,那么如果按照光栅扫描序,XX1上面一个宏块位于Slice0.由于Slice规定预测宏块不可以位于另外一个Slice之中,另外当编码XX2宏块时,由于先对Slice0编码,Slice1还没有编码完成,当然也不能供预测,注意提供预测的必须是重建后的PIC(帧间)或者宏块(帧内),XX1,XX2宏块的B,C,D子块预测都不可用.至此,FMO问题到此结束.
判断宏块是否可用:
nt mb_is_available(int mbAddr, int currMbAddr)
{
  if ((mbAddr < 0) || (mbAddr > ((int)img->PicSizeInMbs - 1)))
    return 0;
  // the following line checks both: slice number and if the mb has been decoded
  if (!img->DeblockCall)
  {
    if (img->mb_data[mbAddr].slice_nr != img->mb_data[currMbAddr].slice_nr)
   //判断Slice编号,如果不属于同一个Slice,那么该宏块不能用来作为自己的预测.
      return 0;
  }
 
  return 1;
 
 通过当前宏块地址计算出相邻宏块是否可用。

0 0
原创粉丝点击