📄 dsp_mot.c
字号:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>
#include <std.h>
#include "dsp_type.h"
#include "dsp_global.h"
#include "dsp_stream.h"
#include "dsp_vop.h"
//#include "dsp_dct.h"
#include "dsp_vlc.h"
//对整个帧的运动估计
void
MotionEstimation (
unsigned char *curr_vop, /* <-- current Vop (for luminance) */
unsigned char *prev_rec_vop, /* <-- reference Vop (reconstructed)(1/2 pixel) */
//Int enable_8x8_mv, /* <-- 8x8 MV (=1) or only 16x16 MV (=0) */
//Int edge, /* <-- restricted(==0)/unrestricted(==edge) mode */
//Int f_code, /* <-- MV search range 1/2 pel: 1=32,2=64,...,7=2048*/
//unsigned char *curr_comp_vop, /* <-> motion compensated current VOP */
int *mad, /* <-> mad value of the ME/MC result */
char *motx, /* --> horizontal MV coordinates */
char *moty, /* --> vertical MV coordinates */
char *MBmode /* --> modes for each MB */
)
{
register short i,j;
int sad=0;
short MBnumX,MBnumY;
*mad = 0;
MBnumX = (p_par->width)/MB_SIZE; //MB_SIZE 16
MBnumY = (p_par->height)/MB_SIZE;
for ( j=0; j< MBnumY; j++)
{
for ( i=0; i< MBnumX; i++)
{
loadUCharMBdata(curr_vop,i,j,p_par->width,p_par->height, g_currMB);//g_currMB now is the source MB
//在这里VOP的大小实际就是参考帧图像的大小vop-width=p_par->width + 32, vop-height=p_par->height + 32
sad+=MBMotionEstimation( prev_rec_vop,i, j, p_par->hor_spat_ref, p_par->ver_spat_ref,
p_par->width + 32, p_par->height + 32, p_par->sr_for,
motx,moty,MBmode);
} /* end of i loop */
} /* end of j loop */
*mad = sad/(p_par->width * p_par->height);//平均绝对差值
}
//获取最大最小范围
void Obtain_Range(
short sr, /* <-- Serach range (radius) */
char type, /* <-- MBM_INTER16==16x16 search; MBM_INTER8==8x8 search */
short pmv_x, /* <-- predicted horizontal motion vector */
short pmv_y, /* <-- predicted horizontal motion vector */
short *mv_x_min, /* --> min horizontal range */
short *mv_x_max, /* --> max horizontal range */
short *mv_y_min, /* --> min vertical range */
short *mv_y_max /* --> max vertical range */
)
{
/****************************************************
range为搜索半径,pmv_x,y为预测向量值,AUX_X_MIN为搜索范围变化参考
******************************************************/
// short aux_x_min, aux_y_min,aux_x_max, aux_y_max;
short range;
range = sr;
*mv_x_min=-range; *mv_x_max= range - 1;
*mv_y_min=-range; *mv_y_max= range - 1;
/*
if (type==MBM_INTER8)//MBM_INTER8=4
{
aux_x_min=pmv_x - DEFAULT_8_WIN;//DEFAULT_8_WIN=2
aux_y_min=pmv_y - DEFAULT_8_WIN;
aux_x_max=pmv_x + DEFAULT_8_WIN;
aux_y_max=pmv_y + DEFAULT_8_WIN;
if(*mv_x_min < aux_x_min)
*mv_x_min = aux_x_min;//缩小向左搜索的范围
if(*mv_y_min < aux_y_min)
*mv_y_min = aux_y_min;//缩小向下搜索的范围
if(*mv_x_max > aux_x_max)
*mv_x_max = aux_x_max;//缩小向右搜索的范围
if(*mv_y_max > aux_y_max)
*mv_y_max = aux_y_max;//缩小向上搜索的范围
}
*/
}
void
RangeInSearchArea(
char i, /* <-- horizontal MBcoordinate in pixels */
char j, /* <-- vertical MB coordinate in pixels */
char block, /* <-- block position (0 16x16; 1-2-3-4 8x8) */
char prev_x, /* <-- absolute horizontal position of the previous vop*/
char prev_y, /* <-- absolute vertical position of the previous vop */
short vop_width, /* <-- horizontal vop dimension */
short vop_height, /* <-- vertical vop dimension */
// Int br_x, /* <-- absolute horizontal position of the current vop */
// Int br_y, /* <-- absolute vertical position of the current vop */
// Int edge, /* <-- edge arround the reference vop */
// Int f_code, /* <- MV search range 1/2 (or 1/4) pel: (0=16,) 1=32,2=64,...,7=2048 */
short *mv_x_min, /* <-- min horizontal range */
short *mv_x_max, /* <-- max horizontal range */
short *mv_y_min, /* <-- min vertical range */
short *mv_y_max, /* <-- max vertical range */
char *out /* --> the search area does not exist (the reference) */ /* and current BB does not overlap) */
)
{
short dim_curr_x_max,
dim_curr_y_max,
dim_curr_x_min,
dim_curr_y_min;
short dim_prev_x_max,
dim_prev_y_max,
dim_prev_x_min,
dim_prev_y_min;
char mb_b_size,
block_x,
block_y;
*out=0;
//MB_SIZE=16,B_SIZE=8
switch (block)
{
case 0: /* 8x8 or 16x16 block search */
block_x=0; /*****************************/
block_y=0; /** 1 2 ******** 0 *********/
mb_b_size=MB_SIZE; /** 3 4 ******** *********/
break; /*****************************/
case 1:
block_x=0;
block_y=0;
mb_b_size=B_SIZE;
break;
case 2:
block_x=B_SIZE;
block_y=0;
mb_b_size=B_SIZE;
break;
case 3:
block_x=0;
block_y=B_SIZE;
mb_b_size=B_SIZE;
break;
case 4:
block_x=B_SIZE;
block_y=B_SIZE;
mb_b_size=B_SIZE;
break;
default:
return;
}
/* min x/y 负方向*/
dim_curr_x_min=(short)(i*MB_SIZE+*mv_x_min+block_x);
dim_curr_y_min=(short)(j*MB_SIZE+*mv_y_min+block_y);
dim_prev_x_min=prev_x/*-edge参考帧中位置*/;
dim_prev_y_min=prev_y/*-edge*/;
/* max x/y 正方向*/
/*the MB right-pixels inside */
dim_curr_x_max=(short)(i*MB_SIZE+*mv_x_max+mb_b_size+block_x);
/*the MB bottom-pixels inside */
dim_curr_y_max=(Int)(j*MB_SIZE+*mv_y_max+mb_b_size+block_y);
dim_prev_x_max=prev_x+vop_width /*+edge*/;
dim_prev_y_max=prev_y+vop_height/*+edge*/;
/* range x/y min 判断搜索范围是否存在*/
if (dim_curr_x_min > dim_prev_x_max)
{
*out=1;
}
else if(dim_curr_x_min < dim_prev_x_min)//当前帧小所以要加
{
*mv_x_min = *mv_x_min + ( dim_prev_x_min - dim_curr_x_min ) ;
}
if(!(*out))
{
if (dim_curr_y_min > dim_prev_y_max)
{
*out=1;
}
else if(dim_curr_y_min < dim_prev_y_min)
{
*mv_y_min = *mv_y_min + ( dim_prev_y_min - dim_curr_y_min ) ;
}
}
/* range x/y max */
if(!(*out))
{
if(dim_curr_x_max < dim_prev_x_min)
{
*out=1;
}
if ((!(*out))&&(dim_curr_x_max > dim_prev_x_max))//当前帧大要减
{
*mv_x_max = *mv_x_max - ( dim_curr_x_max - dim_prev_x_max) ;
}
}
if(!(*out))
{
if(dim_curr_y_max < dim_prev_y_min)
{
*out=1; /* already set */
}
if ((!(*out))&&(dim_curr_y_max > dim_prev_y_max))
{
*mv_y_max = *mv_y_max - ( dim_curr_y_max - dim_prev_y_max) ;
}
}
if(*mv_x_min>*mv_x_max)
{
*out=1;
}
if ( (!(*out)) && (*mv_y_min>*mv_y_max))
{
*out=1;
}
return;
}
//这里没有进行半像素插值
inline Int
SAD_Macroblock(
unsigned char * restrict ii, /* <-- Pointer to the upper-left pel of first MB */
unsigned char * restrict act_block, /* <-- Id, second MB (width=16) */
short h_length, /* <-- Width of first area */
Int Min_FRAME /* <-- Minimum prediction error so far */
)
{
Int sad = 0;
unsigned char *kk;
kk = act_block;
sad=IMG_sad_16x16(kk,ii,h_length);//调用TI库函数进行宏块差值
if (sad > Min_FRAME)//MV_MAX_ERROR=0x2000000
return MV_MAX_ERROR;
else
return sad;
}
/*
inline Int
SAD_Block(
unsigned char * restrict ii, // <-- First area
unsigned char * restrict act_block, // <-- Id. second MB (width=16)
short h_length, //<-- Width of first area
Int min_sofar // <-- Minimum prediction error so far
)
{
Int sad = 0;
unsigned char *kk;
kk = act_block;
sad=IMG_sad_8x8(kk,ii,h_length);//调用TI库函数进行块差值
if (sad > min_sofar)
return INT_MAX;
else
return sad;
}
*/
//帧内/帧间编码方式的选择
char
TryIntraMode(unsigned char *curr,Int min_SAD)
{
register short j;
Int MB_mean = 0;
Int A = 0;
for (j = 0; j < 256; j++)//16*16
{
MB_mean += *(curr + j );//宏块所有像素值相加
}
MB_mean /= 256;//再取平均值
for (j = 0; j < 256; j++)
{
A += ABS( *(curr + j) - MB_mean );
}
if (A < (min_SAD - 2*256))//判断是否属于帧内形式
return 1;
else
return 0;
}
//宏块运动估计函数
Int
MBMotionEstimation(
unsigned char * prev,
char i, /* <-- horizontal MBcoordinate in pixels */
char j, /* <-- vertical MB coordinate in pixels */
char prev_x, /* <-- absolute horiz. position of previous vop */
char prev_y, /* <-- absolute verti. position of previous vop */
short vop_width, //reference vop's width ,including the edge 16
short vop_height, //reference vop's height,including the edge 16
short sr, /* <-- search range (corresponds to f_code) UB 990215*/
char *motx,// short is better,if use char ,then the mv sr should be <64
char *moty,// short is better,if use char ,then the mv sr should be <64
char *MB_md
)
{ //char enable_8x8_mv=0;//以前置为1
char mv16_x,mv16_y;//ldm
// char mv8_x[4],mv8_y[4];//ldm
char MBmode=MBM_INTRA;//初始化MBmode默认是帧内编码
// short x, y;
short k;
Int sad, sad_min=MV_MAX_ERROR;
short mv_x, mv_y;// short is better,if use char ,then the mv sr should be <64
short mv_x_min, mv_x_max, mv_y_min, mv_y_max;//Float
short mvm_width,posBlock0,posBlock1,posBlock2,posBlock3;
short x_curr = i*MB_SIZE, y_curr = j*MB_SIZE;
// char hb,vb,block;
char out;
char rel_ori_x;
char rel_ori_y;
Int min_error16, min_error_intra=0,min_error =0 ;
// int min_error8 = 0;
//#ifndef _FULL_SEARCH_ //如果不用全搜索则使用钻石搜索算法
typedef struct
{
char x;
char y;
char start_nmbr;
} DPoint;
typedef struct
{
DPoint point[8];
} Diamond;
char d_type=1,stop_flag=0,pt_nmbr=0,check_pts,total_check_pts=8,mot_dirn=0;
short d_centre_x=0,d_centre_y=0,check_pt_x,check_pt_y;
Diamond diamond[2]=
{
{
{ {0,1,0}, {1,0,0}, {0,-1,0}, {-1,0,0} }//小模板
}
,
{
{ {0,2,6}, {1,1,0}, {2,0,0}, {1,-1,2},
{0,-2,2}, {-1,-1,4}, {-2,0,4}, {-1,1,6} }//大模板
}
};
//#endif
rel_ori_x= 0 - prev_x;//16
rel_ori_y= 0 - prev_y;//16
Obtain_Range ( sr, MBM_INTER16, 0, 0, &mv_x_min, &mv_x_max, &mv_y_min, &mv_y_max);//先划定范围
RangeInSearchArea (i,j,0, prev_x, prev_y, vop_width, vop_height,
&mv_x_min, &mv_x_max,&mv_y_min, &mv_y_max,&out);//确定详细搜索范围
/* Compute */
//如果没有超出边界
if(!out)
{
//先初始化一个极大值
sad_min=MV_MAX_ERROR;
mv_x = mv_y = 2000; /* A very large MV */
///使用钻石搜索算法先从中心点开始*/
sad = SAD_Macroblock(prev+INDEX_BIG(x_curr+rel_ori_x,
y_curr+rel_ori_y), g_currMB, (vop_width/*+2*edge*/), MV_MAX_ERROR)- (128 + 1);
if (sad<sad_min)
{
sad_min=sad;
mv_x = mv_y = 0;
}
do
{
check_pts=total_check_pts;//total_check_pts=8 钻石搜索算法
do
{//先使用大模板再用小模板
check_pt_x = diamond[d_type].point[pt_nmbr].x + d_centre_x;//d_type=1 pt_nmbr=0 d_centre_x=0
check_pt_y = diamond[d_type].point[pt_nmbr].y + d_centre_y;
/* Restrict the search to the searching window ; Note: This constraint can be removed */
if ( check_pt_x < mv_x_min || check_pt_x > mv_x_max || check_pt_y < mv_y_min || check_pt_y > mv_y_max)
{
sad = MV_MAX_ERROR;
}
else
{
sad=SAD_Macroblock(prev+INDEX_BIG(x_curr+check_pt_x+rel_ori_x,y_curr+check_pt_y+rel_ori_y), g_currMB, (vop_width/*+2*edge*/), sad_min);//rel_ori_x= 0 - prev_x;//INDEX_BIG(x,y) ((x)+(y)*(vop_width))
// #ifdef _SAD_EXHAUS_
//fprintf(stdout,"+o+ [%2d,%2d] sad16(%3d,%3d)=%4d\n",i,j,x,y,sad);//关键输出函数
//#endif
}
if (sad<sad_min)//得到运动向量值
{
sad_min=sad;
mv_x=check_pt_x;
mv_y=check_pt_y;
mot_dirn=pt_nmbr;//pt_nmbr=0
}
else if (sad==sad_min)
if((ABS(check_pt_x)+ABS(check_pt_y)) < (ABS(mv_x)+ABS(mv_y)))
{
sad_min=sad;
mv_x=check_pt_x;
mv_y=check_pt_y;
mot_dirn=pt_nmbr;
}
pt_nmbr+=1;
if((pt_nmbr)>= 8) pt_nmbr-=8;
check_pts-=1;
}//do
while(check_pts>0);
//判断是否结束循环
if( d_type == 0)
{
stop_flag = 1;
}
else
{
if( (mv_x == d_centre_x) && (mv_y == d_centre_y) )
{
d_type=0;//开始使用小模板判断
pt_nmbr=0;
total_check_pts = 4;
}
else
{
if((mv_x==d_centre_x) ||(mv_y==d_centre_y))
total_check_pts=5;//进一步判断
else
total_check_pts=3;
pt_nmbr=diamond[d_type].point[mot_dirn].start_nmbr;
d_centre_x = mv_x;
d_centre_y = mv_y;
}
}
}//do
while(stop_flag!=1);
// #endif
MBmode = MBM_INTER16;//@
min_error16 = sad_min;//@
min_error = min_error16;//@
}//if(!out)
else
{
mv_x=mv_y=0;
sad_min=MV_MAX_ERROR;
}
mv16_x = mv_x; mv16_y = mv_y; //ldm向量赋值16*16
min_error16 = sad_min;
/* Compute 8x8 MVs
if(enable_8x8_mv==1)//宏块细分
{
if(!out)
{
for (block=0;block<4;block++)
{
// Obtain range
if(block==0)
{
hb=vb=0;
}
else if (block==1)
{
hb=1;vb=0;
}
else if (block==2)
{
hb=0;vb=1;
}
else
{
hb=vb=1;
}
// restrict the search range based on the current 16x16 MV inside a window around it
Obtain_Range(sr, MBM_INTER8,
mv16_x, mv16_y, &mv_x_min,&mv_x_max, &mv_y_min, &mv_y_max);
RangeInSearchArea(i,j, block+1, prev_x, prev_y,
vop_width, vop_height, &mv_x_min, &mv_x_max, &mv_y_min,&mv_y_max,&out);
if(!out)
{
sad_min=MV_MAX_ERROR;
mv_x = mv_y = 2000; //A very large MV
for (y= mv_y_min; y<= mv_y_max; y++)
for (x= mv_x_min; x<= mv_x_max; x++)
{
sad=SAD_Block(prev+
INDEX_BIG(x_curr + x + 8*(block==1||block==3)+rel_ori_x,
y_curr + y + 8*(block==2||block==3)+rel_ori_y),
g_currMB+INDEX_NOR(8*(block==1||block==3),
8*(block==2||block==3)),
(vop_width), sad_min);//获得差值
if (sad<sad_min)//判断运动向量
{
sad_min=sad;
mv_x=x;
mv_y=y;
}
else if (sad==sad_min)
if((ABS(x)+ABS(y)) < (ABS(mv_x)+ABS(mv_y)))
{
sad_min=sad;
mv_x=x;
mv_y=y;
}
}//for
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -