common/common.h
x264_scan8这个表格储存的是什么资料。开始的时候,的确让人费解,越看越糊涂。
在很多地方都直接或者间接通过这个表格来保存资料。比如:
h->mb.cache.intra4x4_pred_mode[x264_scan8[0] - 8] = h->mb.intra4x4_pred_mode[i_top_xy][0];
h->mb.cache.intra4x4_pred_mode[x264_scan8[1] - 8] = h->mb.intra4x4_pred_mode[i_top_xy][1];
h->mb.cache.intra4x4_pred_mode[x264_scan8[4] - 8] = h->mb.intra4x4_pred_mode[i_top_xy][2];
h->mb.cache.intra4x4_pred_mode[x264_scan8[5] - 8] = h->mb.intra4x4_pred_mode[i_top_xy][3];
很多地方都是通过x264_scan8找到数组的下标,那么这个x264_scan8表格是如何定义的呢,
我们看
static const int x264_scan8[16+2*4] =
{
/* Luma */
4+1*8, 5+1*8, 4+2*8, 5+2*8,
6+1*8, 7+1*8, 6+2*8, 7+2*8,
4+3*8, 5+3*8, 4+4*8, 5+4*8,
6+3*8, 7+3*8, 6+4*8, 7+4*8,
1+1*8, 2+1*8,
1+2*8, 2+2*8,
1+4*8, 2+4*8,
1+5*8, 2+5*8,
};
/*
0 1 2 3 4 5 6 7
0
1 B B L L L L
2 B B L L L L
3 L L L L
4 R R L L L L
5 R R
*/
YcbCr取样时,色度是水平和垂直方向都是二取一,所以,这个4x4色度块对应于宏块的8x8个像素。也就是说,对于一个宏块(16x16像素),它的Y分量是16x16个字节,色度Cb,水平方向,两个像素样点,共用一个色度值,垂直方向也是二个样点共用一个色度值,也就是,4个亮度值,对应于一个Cb和一个Cr。
事实上,第0行,第0列,第3列,第3行都有空块.这些空块用来存储什么呢,通过分析,用来存储当前宏块的top宏块,left宏块的邻近像素,比如:
宏块的top宏块下边缘的扫描(对Luma)。其他的Cb,Cr分量依次类推。
到此x264_scan8表格分析完成,结论:存储的是当前宏块各个分量4X4块在8X6表格中的扫描序号。
事实上,第0行,第0列,第3列,第3行都有空块.这些空块用来存储什么呢,通过分析,用来存储当前宏块的top宏块,left宏块的4X4子块的扫描,比如x264_scan8[0],代表第一个Luma的扫描,扫描序为12,
位于上图表格的(4,1)处,(5,1)处是第二个Luma,扫描序为13,位于x264_scan8[1],(6,1)处这时候扫描序号是14,位于x264_scan8[4],
(7,1)扫描序号是15位于x264_scan8[5].所以x264_scan8[0],x264_scan8[1],x264_scan8[4],x264_scan8[5],代表整个宏块的
第一行的4X4块的扫描序号,由于我们在帧内亮度,色度预测,帧间运动矢量预测的时候,都要运用top,left侧的子块
做Left,top,或者中值预测.所以上面空出来的扫描序号用来存储top,或者left宏块的边缘子块(4X4)扫描。比如(4,0)~(7,0)存储当前
宏块的top宏块下边缘的扫描(对Luma)。其他的Cb,Cr分量依次类推。
到此x264_scan8表格分析完成,结论:存储的是当前宏块各个分量4X4块在8X6表格中的扫描序号。