📄 recon.cpp
字号:
// Recon.cpp: implementation of the CRecon class.
//
//////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "视频编解码器.h"
#include "Recon.h"
#ifdef _DEBUG
#undef THIS_FILE
static char THIS_FILE[]=__FILE__;
#define new DEBUG_NEW
#endif
#define sign(a) ((a) < 0 ? -1 : 1)
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
extern int MV[2][5][MBR_MAX+1][MBC_MAX+2];
extern int mv_outside_frame,adv_pred_mode,pb_frame,trd,trb;
extern int modemap[MBR_MAX+1][MBC_MAX+2];
extern int mb_width,coded_picture_width;
extern unsigned char *edgeframe[3],*newframe[3], *bframe[3],*oldrefframe[3];
CRecon::CRecon()
{
}
CRecon::~CRecon()
{
}
void CRecon::reconstruct(int bx, int by, int P, int bdx, int bdy)
{
int w,h,lx,lx2,dx,dy,xp,yp,comp,sum;
int x,y,mode,xvec,yvec;
unsigned char *src[3];
x = bx/16+1;
y = by/16+1;
lx = coded_picture_width;
if (mv_outside_frame) //
{
lx2 = coded_picture_width + 64;
src[0] = edgeframe[0];
src[1] = edgeframe[1];
src[2] = edgeframe[2];
}
else
{
lx2 = coded_picture_width;
src[0] = oldrefframe[0];
src[1] = oldrefframe[1];
src[2] = oldrefframe[2];
}
mode = modemap[y][x];
if (P) // P 预测模式
{
/* P prediction */
if (adv_pred_mode)
{
w = 8; h = 8;
/* Y*/
for (comp = 0; comp < 4; comp++)
{
xp = bx + ((comp&1)<<3);
yp = by + ((comp&2)<<2);
recon_comp_obmc(src[0],newframe[0], lx,lx2,comp,w,h,xp,yp);
}
if (mode == MODE_INTER4V)
{
sum = MV[0][1][y][x]+MV[0][2][y][x]+ MV[0][3][y][x]+MV[0][4][y][x];
dx = sign(sum)*(roundtab[abs(sum)%16] + (abs(sum)/16)*2);
sum = MV[1][1][y][x]+MV[1][2][y][x]+ MV[1][3][y][x]+MV[1][4][y][x];
dy = sign(sum)*(roundtab[abs(sum)%16] + (abs(sum)/16)*2);
}
else
{
dx = MV[0][0][y][x];
dy = MV[1][0][y][x];
// chroma rounding
dx = ( dx % 4 == 0 ? dx >> 1 : (dx>>1)|1 );
dy = ( dy % 4 == 0 ? dy >> 1 : (dy>>1)|1 );
}
lx>>=1;bx>>=1;
lx2>>=1; by>>=1;
// Chroma
recon_comp(src[1],newframe[1], lx,lx2,w,h,bx,by,dx,dy,1);
recon_comp(src[2],newframe[2], lx,lx2,w,h,bx,by,dx,dy,2);
}
else
{ // normal prediction mode
// P prediction
w = 16; h = 16;
dx = MV[0][0][y][x];
dy = MV[1][0][y][x];
// Y image
recon_comp(src[0],newframe[0], lx,lx2,w,h,bx,by,dx,dy,0);
//src[0]->newframe[0]
lx>>=1; w>>=1;
bx>>=1; lx2>>=1;
h>>=1; by>>=1;
// chroma rounding
dx = ( dx % 4 == 0 ? dx >> 1 : (dx>>1)|1 );
dy = ( dy % 4 == 0 ? dy >> 1 : (dy>>1)|1 );
// Chroma image
recon_comp(src[1],newframe[1], lx,lx2,w,h,bx,by,dx,dy,1);
recon_comp(src[2],newframe[2], lx,lx2,w,h,bx,by,dx,dy,2);
//src[]->newframe[]
}
}//end of(P)
else
{
// B forward prediction
if (adv_pred_mode)
{
if (mode == MODE_INTER4V)
{
w = 8; h = 8;
// Y Image
xvec = yvec = 0;
for (comp = 0; comp < 4; comp++)
{
xvec += dx = (trb)*MV[0][comp+1][y][x]/trd + bdx;
yvec += dy = (trb)*MV[1][comp+1][y][x]/trd + bdy;
xp = bx + ((comp&1)<<3);
yp = by + ((comp&2)<<2);
recon_comp(src[0],bframe[0], lx,lx2,w,h,xp,yp,dx,dy,0);
}
// chroma rounding (table 16/H.263)
dx = sign(xvec)*(roundtab[abs(xvec)%16] + (abs(xvec)/16)*2);
dy = sign(yvec)*(roundtab[abs(yvec)%16] + (abs(yvec)/16)*2);
lx>>=1; bx>>=1;
lx2>>=1;by>>=1;
// Chroma U V image
recon_comp(src[1],bframe[1], lx,lx2,w,h,bx,by,dx,dy,1);
recon_comp(src[2],bframe[2], lx,lx2,w,h,bx,by,dx,dy,2);
}
else
{ // adv_pred_mode but 16x16 vector
w = 16; h = 16;
dx = (trb)*MV[0][0][y][x]/trd + bdx;
dy = (trb)*MV[1][0][y][x]/trd + bdy;
// Y image
recon_comp(src[0],bframe[0], lx,lx2,w,h,bx,by,dx,dy,0);
lx>>=1; w>>=1; bx>>=1; lx2>>=1;
h>>=1; by>>=1;
xvec = 4*dx;
yvec = 4*dy;
// chroma rounding (table 16/H.263)
dx = sign(xvec)*(roundtab[abs(xvec)%16] + (abs(xvec)/16)*2);
dy = sign(yvec)*(roundtab[abs(yvec)%16] + (abs(yvec)/16)*2);
// Chroma U V image
recon_comp(src[1],bframe[1], lx,lx2,w,h,bx,by,dx,dy,1);
recon_comp(src[2],bframe[2], lx,lx2,w,h,bx,by,dx,dy,2);
}
}//end of(adv_pred_mode)
else
{ // normal B forward prediction
w = 16; h = 16;
dx = (trb)*MV[0][0][y][x]/trd + bdx;
dy = (trb)*MV[1][0][y][x]/trd + bdy;
/* Y */
recon_comp(src[0],bframe[0], lx,lx2,w,h,bx,by,dx,dy,0);
lx>>=1; w>>=1; bx>>=1; lx2>>=1;
h>>=1; by>>=1;
xvec = 4*dx;
yvec = 4*dy;
// chroma rounding (table 16/H.263)
dx = sign(xvec)*(roundtab[abs(xvec)%16] + (abs(xvec)/16)*2);
dy = sign(yvec)*(roundtab[abs(yvec)%16] + (abs(yvec)/16)*2);
// Chroma U V image
recon_comp(src[1],bframe[1], lx,lx2,w,h,bx,by,dx,dy,1);
recon_comp(src[2],bframe[2], lx,lx2,w,h,bx,by,dx,dy,2);
}
}
}
void CRecon::recon_comp(unsigned char *src, unsigned char *dst,
int lx, int lx2, int w, int h, int x, int y, int dx, int dy, int flag)
{
int xint, xh, yint, yh;
unsigned char *s, *d;
xint = dx>>1;
xh = dx & 1;
yint = dy>>1;
yh = dy & 1;
/* origins */
s = src + lx2*(y+yint) + x + xint;
d = dst + lx*y + x;
if (!xh && !yh)
if (w!=8)
rec(s,d,lx,lx2,h);
else
recc(s,d,lx,lx2,h);
else if (!xh && yh)
if (w!=8)
recv(s,d,lx,lx2,h);
else
recvc(s,d,lx,lx2,h);
else if (xh && !yh)
if (w!=8)
rech(s,d,lx,lx2,h);
else
rechc(s,d,lx,lx2,h);
else /* if (xh && yh) */
if (w!=8)
rec4(s,d,lx,lx2,h);
else
rec4c(s,d,lx,lx2,h);
}
void CRecon::recon_comp_obmc(unsigned char *src, unsigned char *dst, int lx, int lx2, int comp, int w, int h, int x, int y)
{
int j,k;
int xmb,ymb;
int c8,t8,l8,r8;
int xit,xib,xir,xil;
int yit,yib,yir,yil;
int vect,vecb,vecr,vecl;
int nx[5],ny[5],xint[5],yint[5],xh[5],yh[5];
int p[64],*pd;
unsigned char *d,*s[5];
xmb = (x>>4)+1;
ymb = (y>>4)+1;
c8 = (modemap[ymb][xmb] == MODE_INTER4V ? 1 : 0);
t8 = (modemap[ymb-1][xmb] == MODE_INTER4V ? 1 : 0);
l8 = (modemap[ymb][xmb-1] == MODE_INTER4V ? 1 : 0);
r8 = (modemap[ymb][xmb+1] == MODE_INTER4V ? 1 : 0);
switch (comp+1) {
case 1:
vect = (t8 ? 3 : 0) ; yit = ymb-1 ; xit = xmb;
vecb = (c8 ? 3 : 0) ; yib = ymb ; xib = xmb;
vecl = (l8 ? 2 : 0) ; yil = ymb ; xil = xmb-1;
vecr = (c8 ? 2 : 0) ; yir = ymb ; xir = xmb;
if (ymb == 1) {
yit = ymb;
vect = (c8 ? 1 : 0);
}
if (xmb == 1) {
xil = xmb;
vecl = (c8 ? 1 : 0);
}
break;
case 2:
vect = (t8 ? 4 : 0) ; yit = ymb-1 ; xit = xmb;
vecb = (c8 ? 4 : 0) ; yib = ymb ; xib = xmb;
vecl = (c8 ? 1 : 0) ; yil = ymb ; xil = xmb;
vecr = (r8 ? 1 : 0) ; yir = ymb ; xir = xmb+1;
if (ymb == 1) {
yit = ymb;
vect = (c8 ? 2 : 0);
}
if (xmb == mb_width) {
xir = xmb;
vecr = (c8 ? 2 : 0);
}
break;
case 3:
vect = (c8 ? 1 : 0) ; yit = ymb ; xit = xmb;
vecb = (c8 ? 3 : 0) ; yib = ymb ; xib = xmb;
vecl = (l8 ? 4 : 0) ; yil = ymb ; xil = xmb-1;
vecr = (c8 ? 4 : 0) ; yir = ymb ; xir = xmb;
if (xmb == 1) {
xil = xmb;
vecl = (c8 ? 3 : 0);
}
break;
case 4:
vect = (c8 ? 2 : 0) ; yit = ymb ; xit = xmb;
vecb = (c8 ? 4 : 0) ; yib = ymb ; xib = xmb;
vecl = (c8 ? 3 : 0) ; yil = ymb ; xil = xmb;
vecr = (r8 ? 3 : 0) ; yir = ymb ; xir = xmb+1;
if (xmb == mb_width) {
xir = xmb;
vecr = (c8 ? 4 : 0);
}
break;
default:
fprintf(stderr,"Illegal block number in recon_comp_obmc (recon.c)\n");
exit(1);
break;
}
nx[0] = MV[0][c8 ? comp + 1 : 0][ymb][xmb];
ny[0] = MV[1][c8 ? comp + 1 : 0][ymb][xmb];
nx[1] = MV[0][vect][yit][xit]; ny[1] = MV[1][vect][yit][xit];
nx[2] = MV[0][vecb][yib][xib]; ny[2] = MV[1][vecb][yib][xib];
nx[3] = MV[0][vecr][yir][xir]; ny[3] = MV[1][vecr][yir][xir];
nx[4] = MV[0][vecl][yil][xil]; ny[4] = MV[1][vecl][yil][xil];
for (k = 0; k < 5; k++) {
xint[k] = nx[k]>>1;
xh[k] = nx[k] & 1;
yint[k] = ny[k]>>1;
yh[k] = ny[k] & 1;
s[k] = src + lx2*(y+yint[k]) + x + xint[k];
}
d = dst + lx*y + x;
if (!xh[0] && !yh[0])
reco(s[0],&p[0],8,lx2,0,0,0,8,0,8);
else if (!xh[0] && yh[0])
recvo(s[0],&p[0],8,lx2,0,0,0,8,0,8);
else if (xh[0] && !yh[0])
recho(s[0],&p[0],8,lx2,0,0,0,8,0,8);
else /* if (xh[] && yh[]) */
rec4o(s[0],&p[0],8,lx2,0,0,0,8,0,8);
if (!xh[1] && !yh[1])
reco(s[1],&p[0],8,lx2,1,1,0,8,0,4);
else if (!xh[1] && yh[1])
recvo(s[1],&p[0],8,lx2,1,1,0,8,0,4);
else if (xh[1] && !yh[1])
recho(s[1],&p[0],8,lx2,1,1,0,8,0,4);
else /* if (xh[] && yh[]) */
rec4o(s[1],&p[0],8,lx2,1,1,0,8,0,4);
if (!xh[2] && !yh[2])
reco(s[2]+(lx2<<2),&p[32],8,lx2,1,2,0,8,4,8);
else if (!xh[2] && yh[2])
recvo(s[2]+(lx2<<2),&p[32],8,lx2,1,2,0,8,4,8);
else if (xh[2] && !yh[2])
recho(s[2]+(lx2<<2),&p[32],8,lx2,1,2,0,8,4,8);
else /* if (xh[] && yh[]) */
rec4o(s[2]+(lx2<<2),&p[32],8,lx2,1,2,0,8,4,8);
if (!xh[3] && !yh[3])
reco(s[3],&p[0],8,lx2,1,3,4,8,0,8);
else if (!xh[3] && yh[3])
recvo(s[3],&p[0],8,lx2,1,3,4,8,0,8);
else if (xh[3] && !yh[3])
recho(s[3],&p[0],8,lx2,1,3,4,8,0,8);
else /* if (xh[] && yh[]) */
rec4o(s[3],&p[0],8,lx2,1,3,4,8,0,8);
if (!xh[4] && !yh[4])
reco(s[4],&p[0],8,lx2,1,4,0,4,0,8);
else if (!xh[4] && yh[4])
recvo(s[4],&p[0],8,lx2,1,4,0,4,0,8);
else if (xh[4] && !yh[4])
recho(s[4],&p[0],8,lx2,1,4,0,4,0,8);
else /* if (xh[] && yh[]) */
rec4o(s[4],&p[0],8,lx2,1,4,0,4,0,8);
pd = &p[0];
for (j = 0; j < 8; j++) {
d[0] = (pd[0] + 4 )>>3;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -