You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1423 lines
49 KiB
1423 lines
49 KiB
/* File: sv_h263_getpic.c */
|
|
/*****************************************************************************
|
|
** Copyright (c) Digital Equipment Corporation, 1995, 1997 **
|
|
** **
|
|
** All Rights Reserved. Unpublished rights reserved under the copyright **
|
|
** laws of the United States. **
|
|
** **
|
|
** The software contained on this media is proprietary to and embodies **
|
|
** the confidential technology of Digital Equipment Corporation. **
|
|
** Possession, use, duplication or dissemination of the software and **
|
|
** media is authorized only pursuant to a valid written license from **
|
|
** Digital Equipment Corporation. **
|
|
** **
|
|
** RESTRICTED RIGHTS LEGEND Use, duplication, or disclosure by the U.S. **
|
|
** Government is subject to restrictions as set forth in Subparagraph **
|
|
** (c)(1)(ii) of DFARS 252.227-7013, or in FAR 52.227-19, as applicable. **
|
|
******************************************************************************/
|
|
|
|
/*
|
|
#define _SLIBDEBUG_
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include "sv_h263.h"
|
|
#include "sv_intrn.h"
|
|
#include "SC_err.h"
|
|
#include "proto.h"
|
|
|
|
#ifdef _SLIBDEBUG_
|
|
#include "sc_debug.h"
|
|
|
|
#define _DEBUG_ 0 /* detailed debuging statements */
|
|
#define _VERBOSE_ 1 /* show progress */
|
|
#define _VERIFY_ 0 /* verify correct operation */
|
|
#define _WARN_ 1 /* warnings about strange behavior */
|
|
#endif
|
|
|
|
/* private prototypes*/
|
|
static void getMBs(SvH263DecompressInfo_t *H263Info, ScBitstream_t *BSIn);
|
|
static void clearblock(SvH263DecompressInfo_t *H263Info, int comp);
|
|
static int motion_decode(SvH263DecompressInfo_t *H263Info, int vec, int pmv);
|
|
static int find_pmv(SvH263DecompressInfo_t *H263Info, int x, int y, int block, int comp);
|
|
static void addblock(SvH263DecompressInfo_t *H263Info, int comp, int bx, int by, int addflag);
|
|
static void reconblock_b(SvH263DecompressInfo_t *H263Info, int comp, int bx, int by,
|
|
int mode, int bdx, int bdy);
|
|
static void find_bidir_limits(int vec, int *start, int*stop, int nhv);
|
|
static void find_bidir_chroma_limits(int vec, int *start, int*stop);
|
|
static void make_edge_image(unsigned char *src, unsigned char *dst, int width, int height, int edge);
|
|
void interpolate_image(unsigned char *in, unsigned char *out, int width, int height);
|
|
|
|
static void IDCT_add(int comp,int bx,int by,int addflag) ;
|
|
|
|
static int H263_roundtab[16]= {0,0,0,1,1,1,1,1,1,1,1,1,1,1,2,2};
|
|
|
|
static int H263_codtab[2]= {0,1};
|
|
static int H263_mcbpctab[21] = {0,16,32,48,1,17,33,49,2,18,34,50,3,19,35,51,4,20,36,52,255};
|
|
static int H263_mcbpc_intratab[9] = {3,19,35,51,4,20,36,52,255};
|
|
static int H263_modb_tab[3] = {0, 1, 2};
|
|
static int H263_ycbpb_tab[2] = {0, 1};
|
|
static int H263_uvcbpb_tab[2] = {0, 1};
|
|
static int H263_cbpytab[16] = {15,14,13,12,11,10,9,8,7,6,5,4,3,2,1,0};
|
|
static int H263_cbpy_intratab[16] = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15};
|
|
static int H263_dquanttab[4] = {1,0,3,4};
|
|
static int H263_mvdtab[64] = {32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31};
|
|
static int H263_intradctab[254] = {1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,119,120,121,122,123,124,125,126,127,255,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175,176,177,178,179,180,181,182,183,184,185,186,187,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,203,204,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,220,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,238,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,254};
|
|
|
|
static int H263_cumf_COD[3]={16383, 6849, 0};
|
|
static int H263_cumf_MCBPC[22]={16383, 4105, 3088, 2367, 1988, 1621, 1612, 1609, 1608, 496, 353, 195, 77, 22, 17, 12, 5, 4, 3, 2, 1, 0};
|
|
static int H263_cumf_MCBPC_intra[10]={16383, 7410, 6549, 5188, 442, 182, 181, 141, 1, 0};
|
|
static int H263_cumf_MODB[4]={16383, 6062, 2130, 0};
|
|
static int H263_cumf_YCBPB[3]={16383,6062,0};
|
|
static int H263_cumf_UVCBPB[3]={16383,491,0};
|
|
static int H263_cumf_CBPY[17]={16383, 14481, 13869, 13196, 12568, 11931, 11185, 10814, 9796, 9150, 8781, 7933, 6860, 6116, 4873, 3538, 0};
|
|
static int H263_cumf_CBPY_intra[17]={16383, 13619, 13211, 12933, 12562, 12395, 11913, 11783, 11004, 10782, 10689, 9928, 9353, 8945, 8407, 7795, 0};
|
|
static int H263_cumf_DQUANT[5]={16383, 12287, 8192, 4095, 0};
|
|
static int H263_cumf_MVD[65]={16383, 16380, 16369, 16365, 16361, 16357, 16350, 16343, 16339, 16333, 16326, 16318, 16311, 16306, 16298, 16291, 16283, 16272, 16261, 16249, 16235, 16222, 16207, 16175, 16141, 16094, 16044, 15936, 15764, 15463, 14956, 13924, 11491, 4621, 2264, 1315, 854, 583, 420, 326, 273, 229, 196, 166, 148, 137, 123, 114, 101, 91, 82, 76, 66, 59, 53, 46, 36, 30, 26, 24, 18, 14, 10, 5, 0};
|
|
static int H263_cumf_INTRADC[255]={16383, 16380, 16379, 16378, 16377, 16376, 16370, 16361, 16360, 16359, 16358, 16357, 16356, 16355, 16343, 16238, 16237, 16236, 16230, 16221, 16220, 16205, 16190, 16169, 16151, 16130, 16109, 16094, 16070, 16037, 16007, 15962, 15938, 15899, 15854, 15815, 15788, 15743, 15689, 15656, 15617, 15560, 15473, 15404, 15296, 15178, 15106, 14992, 14868, 14738, 14593, 14438, 14283, 14169, 14064, 14004, 13914, 13824, 13752, 13671, 13590, 13515, 13458, 13380, 13305, 13230, 13143, 13025, 12935, 12878, 12794, 12743, 12656, 12596, 12521, 12443, 12359, 12278, 12200, 12131, 12047, 12002, 11948, 11891, 11828, 11744, 11663, 11588, 11495, 11402, 11288, 11204, 11126, 11039, 10961, 10883, 10787, 10679, 10583, 10481, 10360, 10227, 10113, 9961, 9828, 9717, 9584, 9485, 9324, 9112, 9019, 8908, 8766, 8584, 8426, 8211, 7920, 7663, 7406, 7152, 6904, 6677, 6453, 6265, 6101, 5904, 5716, 5489, 5307, 5056, 4850, 4569, 4284, 3966, 3712, 3518, 3342, 3206, 3048, 2909, 2773, 2668, 2596, 2512, 2370, 2295, 2232, 2166, 2103, 2022, 1956, 1887, 1830, 1803, 1770, 1728, 1674, 1635, 1599, 1557, 1500, 1482, 1434, 1389, 1356, 1317, 1284, 1245, 1200, 1179, 1140, 1110, 1092, 1062, 1044, 1035, 1014, 1008, 993, 981, 954, 936, 912, 894, 876, 864, 849, 828, 816, 801, 792, 777, 756, 732, 690, 660, 642, 615, 597, 576, 555, 522, 489, 459, 435, 411, 405, 396, 387, 375, 360, 354, 345, 344, 329, 314, 293, 278, 251, 236, 230, 224, 215, 214, 208, 199, 193, 184, 178, 169, 154, 127, 100, 94, 73, 37, 36, 35, 34, 33, 32, 31, 30, 29, 28, 27, 26, 20, 19, 18, 17, 16, 15, 9, 0};
|
|
|
|
#define clearblk(comp) memset(H263Info->block[comp], 0, sizeof(int [66]));
|
|
|
|
/* decode one frame or field picture */
|
|
|
|
SvStatus_t sv_H263GetPicture(SvCodecInfo_t *Info)
|
|
{
|
|
int i;
|
|
unsigned char *tmp;
|
|
SvH263DecompressInfo_t *H263Info = Info->h263dcmp;
|
|
ScBitstream_t *BSIn=Info->BSIn;
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "sv_H263GetPicture()\n") );
|
|
for (i=0; i<3; i++) {
|
|
tmp = H263Info->oldrefframe[i];
|
|
H263Info->oldrefframe[i] = H263Info->refframe[i];
|
|
H263Info->refframe[i] = tmp;
|
|
H263Info->newframe[i] = H263Info->refframe[i];
|
|
}
|
|
|
|
if (H263Info->mv_outside_frame && H263Info->framenum > 0) {
|
|
make_edge_image(H263Info->oldrefframe[0],H263Info->edgeframe[0],H263Info->coded_picture_width,
|
|
H263Info->coded_picture_height,32);
|
|
make_edge_image(H263Info->oldrefframe[1],H263Info->edgeframe[1],H263Info->chrom_width, H263Info->chrom_height,16);
|
|
make_edge_image(H263Info->oldrefframe[2],H263Info->edgeframe[2],H263Info->chrom_width, H263Info->chrom_height,16);
|
|
}
|
|
getMBs(H263Info, BSIn);
|
|
|
|
if (H263Info->pb_frame) {
|
|
if (H263Info->expand && H263Info->outtype == H263_T_X11) {
|
|
interpolate_image(H263Info->bframe[0], H263Info->exnewframe[0],
|
|
H263Info->coded_picture_width, H263Info->coded_picture_height);
|
|
interpolate_image(H263Info->bframe[1], H263Info->exnewframe[1], H263Info->chrom_width, H263Info->chrom_height);
|
|
interpolate_image(H263Info->bframe[2], H263Info->exnewframe[2], H263Info->chrom_width, H263Info->chrom_height);
|
|
|
|
/* svH263Storeframe(H263Info->exnewframe, *framenum); */
|
|
}
|
|
else
|
|
/* svH263Storeframe(H263Info->bframe,*framenum); */
|
|
|
|
H263Info->framenum += H263Info->pb_frame;
|
|
}
|
|
|
|
if (H263Info->expand && H263Info->outtype == H263_T_X11) {
|
|
interpolate_image(H263Info->newframe[0], H263Info->exnewframe[0],
|
|
H263Info->coded_picture_width, H263Info->coded_picture_height);
|
|
interpolate_image(H263Info->newframe[1], H263Info->exnewframe[1], H263Info->chrom_width, H263Info->chrom_height);
|
|
interpolate_image(H263Info->newframe[2], H263Info->exnewframe[2], H263Info->chrom_width, H263Info->chrom_height);
|
|
|
|
/* svH263Storeframe(H263Info->exnewframe, *framenum); */
|
|
}
|
|
else {
|
|
/* svH263Storeframe(H263Info->newframe,*framenum); */
|
|
}
|
|
return(SvErrorNone);
|
|
}
|
|
|
|
|
|
/* decode all macroblocks of the current picture */
|
|
|
|
#ifdef CAMERAMOTION
|
|
extern void setwintitle(char *);
|
|
|
|
#define MOVERIGHT (1<<0)
|
|
#define MOVELEFT (1<<1)
|
|
#define MOVEUP (1<<2)
|
|
#define MOVEDOWN (1<<3)
|
|
|
|
static int left_x,left_y;
|
|
static int top_x,top_y;
|
|
static int right_x,right_y;
|
|
static int bottom_x,bottom_y;
|
|
static int center_x,center_y;
|
|
static int strip_width = 3;
|
|
|
|
static int framewidth;
|
|
static int frameheight;
|
|
|
|
static int startoff;
|
|
static int filtermean,havecut;
|
|
static int loopstart = 0;
|
|
static int loopend;
|
|
static int scenenumber = 1;
|
|
|
|
|
|
static void initmotion(int width,int height)
|
|
{
|
|
left_x = left_y = 0;
|
|
right_x = right_y = 0;
|
|
bottom_x = bottom_y = 0;
|
|
top_x = top_y = 0;
|
|
center_x = center_y = 0;
|
|
|
|
framewidth = width;
|
|
frameheight = height;
|
|
startoff = svH263Tellbits();
|
|
}
|
|
|
|
static void analyzemotion(int xpos,int ypos,int mvx,int mvy)
|
|
{
|
|
/* Count Motion Vectors for 4 overlapping strips at the edges */
|
|
/* and the center area between them */
|
|
if (xpos < strip_width) { /* Left strip */
|
|
left_x += mvx;
|
|
left_y += mvy;
|
|
}
|
|
if (xpos >= (framewidth - strip_width)) { /* Right strip */
|
|
right_x += mvx;
|
|
right_y += mvy;
|
|
}
|
|
if (ypos < strip_width) { /* Top strip */
|
|
top_x += mvx;
|
|
top_y += mvy;
|
|
}
|
|
if (ypos >= (frameheight - strip_width)) { /* Bottom strip */
|
|
bottom_x += mvx;
|
|
bottom_y += mvy;
|
|
}
|
|
if (xpos >= strip_width && xpos < (framewidth - strip_width)
|
|
&& ypos >= strip_width && ypos < (frameheight - strip_width)) {
|
|
center_x += mvx;
|
|
center_y += mvy;
|
|
}
|
|
}
|
|
|
|
static int last_movement = 0;
|
|
static int keepdirection = 15;
|
|
static int movestartframe = 0;
|
|
static int moveendframe;
|
|
static int movesumx;
|
|
static int movesumy;
|
|
|
|
static void summarizemotion(int frameno)
|
|
{
|
|
int threshold = 8;
|
|
int striparea;
|
|
int sx,sy;
|
|
char title[128];
|
|
int cur_movement = 0;
|
|
int framesize;
|
|
|
|
|
|
title[0] = 0;
|
|
|
|
framesize = svH263Tellbits() - startoff;
|
|
if (frameno < 1) {
|
|
filtermean = framesize/2;
|
|
havecut = 1;
|
|
} else {
|
|
filtermean = (4*filtermean + framesize)/5;
|
|
}
|
|
if (!havecut && framesize > 1.8 * filtermean && frameno > 5) {
|
|
if (!last_movement) {
|
|
printf("dump %d\n",movestartframe + ((moveendframe-movestartframe)/2));
|
|
}
|
|
printf("mark %d %d %d %d\n",scenenumber,loopstart,frameno-1,filtermean);
|
|
printf("shot %d %d %d\n",scenenumber,loopstart,frameno-1);
|
|
movestartframe = frameno;
|
|
loopstart = frameno;
|
|
havecut = 1;
|
|
scenenumber++;
|
|
} else {
|
|
havecut = 0;
|
|
}
|
|
striparea = strip_width * framewidth;
|
|
top_x = (top_x*10)/striparea;
|
|
top_y = (top_y*10)/striparea;
|
|
bottom_x = (bottom_x*10)/striparea;
|
|
bottom_y = (bottom_y*10)/striparea;
|
|
|
|
striparea = strip_width * frameheight;
|
|
left_x = (left_x*10)/striparea;
|
|
left_y = (left_y*10)/striparea;
|
|
right_x = (right_x*10)/striparea;
|
|
right_y = (right_y*10)/striparea;
|
|
|
|
striparea = (framewidth - 2 * strip_width) * (frameheight - 2 * strip_width);
|
|
center_x = (center_x*10)/striparea;
|
|
center_y = (center_y*10)/striparea;
|
|
|
|
/* Horizontal left motion */
|
|
if (top_x < -threshold && bottom_x < -threshold
|
|
&& right_x < -threshold && left_x < -threshold) {
|
|
strcat(title,"Left ");
|
|
cur_movement |= MOVELEFT;
|
|
}
|
|
/* Horizontal right motion */
|
|
if (top_x > threshold && bottom_x > threshold
|
|
&& left_x > threshold && right_x > threshold) {
|
|
strcat(title,"Right ");
|
|
cur_movement |= MOVERIGHT;
|
|
}
|
|
/* Vertical up motion */
|
|
if (left_y < -threshold && right_y < -threshold
|
|
&& top_y < -threshold && bottom_y < -threshold) {
|
|
strcat(title,"Up");
|
|
cur_movement |= MOVEUP;
|
|
}
|
|
/* Vertical down motion */
|
|
if (left_y > threshold && right_y > threshold
|
|
&& bottom_y > threshold && top_y > threshold) {
|
|
strcat(title,"Down ");
|
|
cur_movement |= MOVEDOWN;
|
|
}
|
|
|
|
if (last_movement) {
|
|
movesumx += (top_x + bottom_x + right_x +left_x)/4;
|
|
movesumy += (top_y + bottom_y + right_y +left_y)/4;
|
|
}
|
|
if (!cur_movement != !last_movement) { /* Possible movement change */
|
|
if (--keepdirection < 0) { /* Seen move several times */
|
|
keepdirection = 10;
|
|
if (last_movement) { /* Have been in a move */
|
|
int movelength;
|
|
int i,images,fno;
|
|
|
|
movesumx = movesumx/(16*framewidth);
|
|
movesumy = movesumy/(16*frameheight);
|
|
movelength = moveendframe - movestartframe;
|
|
if (movelength > 25) {
|
|
images = ((abs(movesumx) + abs(movesumy)) / 5) + 1;
|
|
fno = movestartframe + movelength/(2*images);
|
|
for (i = 0; i < images; i++) {
|
|
printf("dump %d\n",fno);
|
|
fno += movelength/images;
|
|
}
|
|
}
|
|
} else { /* Last part was fixed */
|
|
if (havecut) {
|
|
/* Do nothing, was already handled */
|
|
} else {
|
|
printf("dump %d\n",movestartframe + ((moveendframe-movestartframe)/2));
|
|
movestartframe = frameno;
|
|
}
|
|
}
|
|
movesumx = 0;
|
|
movesumy = 0;
|
|
movestartframe = moveendframe+1;
|
|
last_movement = cur_movement;
|
|
if (H263Info->outtype == H263_T_X11) {
|
|
if (title[0] != 0) {
|
|
setwintitle(title);
|
|
} else {
|
|
setwintitle("fixed");
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
moveendframe = frameno;
|
|
}
|
|
fflush(stdout);
|
|
}
|
|
#endif /* CAMERAMOTION */
|
|
|
|
|
|
|
|
static void getMBs(SvH263DecompressInfo_t *H263Info, ScBitstream_t *BSIn)
|
|
{
|
|
int comp;
|
|
int MBA, MBAmax;
|
|
int bx, by;
|
|
|
|
int COD=0,MCBPC, CBPY, CBP=0, CBPB=0, MODB=0, Mode=0, DQUANT;
|
|
int COD_index, CBPY_index, MODB_index, DQUANT_index, MCBPC_index;
|
|
int INTRADC_index, YCBPB_index, UVCBPB_index, mvdbx_index, mvdby_index;
|
|
int mvx, mvy, mvy_index, mvx_index, pmv0, pmv1, xpos, ypos, gob, i,k;
|
|
int mvdbx=0, mvdby=0, pmvdbx, pmvdby, gfid, YCBPB, UVCBPB, gobheader_read;
|
|
int startmv,stopmv,offset,bsize,last_done=0,pCBP=0,pCBPB=0,pCOD=0;
|
|
int DQ_tab[4] = {-1,-2,1,2};
|
|
|
|
register int *bp;
|
|
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs()\n") );
|
|
/* number of macroblocks per picture */
|
|
MBAmax = H263Info->mb_width*H263Info->mb_height;
|
|
|
|
MBA = 0; /* macroblock address */
|
|
H263Info->newgob = 0;
|
|
|
|
/* mark MV's above the picture */
|
|
for (i = 1; i < H263Info->mb_width+1; i++) {
|
|
for (k = 0; k < 5; k++) {
|
|
H263Info->MV[0][k][0][i] = H263_NO_VEC;
|
|
H263Info->MV[1][k][0][i] = H263_NO_VEC;
|
|
}
|
|
H263Info->modemap[0][i] = H263_MODE_INTRA;
|
|
}
|
|
/* zero MV's on the sides of the picture */
|
|
for (i = 0; i < H263Info->mb_height+1; i++) {
|
|
for (k = 0; k < 5; k++) {
|
|
H263Info->MV[0][k][i][0] = 0;
|
|
H263Info->MV[1][k][i][0] = 0;
|
|
H263Info->MV[0][k][i][H263Info->mb_width+1] = 0;
|
|
H263Info->MV[1][k][i][H263Info->mb_width+1] = 0;
|
|
}
|
|
H263Info->modemap[i][0] = H263_MODE_INTRA;
|
|
H263Info->modemap[i][H263Info->mb_width+1] = H263_MODE_INTRA;
|
|
}
|
|
|
|
H263Info->fault = 0;
|
|
gobheader_read = 0;
|
|
|
|
for (;;) {
|
|
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "frame %d, MB %d, bytepos=%ld\n",
|
|
H263Info->framenum,MBA,ScBSBytePosition(BSIn)) );
|
|
/* if (MBA>15) return; */
|
|
resync:
|
|
|
|
/* This version of the decoder does not resync on every possible
|
|
error, and it does not do all possible error checks. It is not
|
|
difficult to make it much more error robust, but I do not think
|
|
it is necessary to include this in the freely available
|
|
version. */
|
|
|
|
if (H263Info->fault) {
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() fault\n") );
|
|
printf("Warning: A Fault Condition Has Occurred - Resyncing \n");
|
|
/* look for startcode */
|
|
if (sv_H263StartCode(BSIn)!=SvErrorNone) /* sync on new startcode */
|
|
{
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() couldn't get startcode\n") );
|
|
return;
|
|
}
|
|
|
|
H263Info->fault = 0;
|
|
}
|
|
|
|
if (!(ScBSPeekBits(BSIn, 22)>>6)) { /* startcode */
|
|
if (sv_H263StartCode(BSIn)!=SvErrorNone) /* sync on new startcode */
|
|
{
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() couldn't get startcode\n") );
|
|
return;
|
|
}
|
|
|
|
/* in case of byte aligned start code, ie. PSTUF, GSTUF or ESTUF
|
|
is used */
|
|
|
|
if (ScBSPeekBits(BSIn, 22) == (32|H263_SE_CODE)) { /* end of sequence */
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() end of sequence\n") );
|
|
if (!(H263Info->syntax_arith_coding && MBA < MBAmax))
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
else if ((ScBSPeekBits(BSIn, 22) == H263_PSC<<5) ) { /* new picture */
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() new picture code, bytepos=%ld\n",
|
|
ScBSBytePosition(BSIn)) );
|
|
if (!(H263Info->syntax_arith_coding && MBA < MBAmax)) {
|
|
#ifdef CAMERAMOTION
|
|
goto summarize_motion;
|
|
#else
|
|
return;
|
|
#endif
|
|
}
|
|
}
|
|
else {
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() new group code, bytepos=%ld\n",
|
|
ScBSBytePosition(BSIn)) );
|
|
if (!(H263Info->syntax_arith_coding && MBA%H263Info->mb_width)) {
|
|
|
|
if (H263Info->syntax_arith_coding) { /* SAC hack to finish GOBs which */
|
|
gob = (int)(ScBSPeekBits(BSIn, 22) & 31); /* end with MBs coded with no bits */
|
|
if (gob * H263Info->mb_width != MBA)
|
|
{
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() goto finish_gob\n") );
|
|
goto finish_gob;
|
|
}
|
|
}
|
|
|
|
if (sv_H263GetHeader(H263Info, BSIn, &gob)!=SvErrorNone)
|
|
{
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() GetHeader failed\n") );
|
|
return;
|
|
}
|
|
/* gob--; */
|
|
if (gob > H263Info->mb_height) {
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() GN out of range: gob=%d\n", gob) );
|
|
return;
|
|
}
|
|
|
|
/* GFID is not allowed to change unless PTYPE in picture header
|
|
changes */
|
|
gfid = (int)ScBSGetBits(BSIn, 2);
|
|
/* NB: in error-prone environments the decoder can use this
|
|
value to determine whether a picture header where the PTYPE
|
|
has changed, has been lost */
|
|
|
|
H263Info->quant = (int)ScBSGetBits(BSIn, 5);
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() GQUANT=%d gob=%d\n",
|
|
H263Info->quant, gob) );
|
|
xpos = 0;
|
|
ypos = gob;
|
|
MBA = ypos * H263Info->mb_width;
|
|
|
|
H263Info->newgob = 1;
|
|
gobheader_read = 1;
|
|
if (H263Info->syntax_arith_coding)
|
|
sv_H263SACDecoderReset(BSIn); /* init. arithmetic decoder buffer after gob */
|
|
}
|
|
}
|
|
}
|
|
|
|
finish_gob: /* SAC specific label */
|
|
|
|
if (!gobheader_read) {
|
|
xpos = MBA%H263Info->mb_width;
|
|
ypos = MBA/H263Info->mb_width;
|
|
if (xpos == 0 && ypos > 0)
|
|
H263Info->newgob = 0;
|
|
}
|
|
else
|
|
gobheader_read = 0;
|
|
|
|
if (MBA>=MBAmax)
|
|
{
|
|
#ifdef CAMERAMOTION
|
|
goto summarize_motion;
|
|
#else
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() done MBA=%d\n", MBA) );
|
|
return; /* all macroblocks decoded */
|
|
#endif
|
|
}
|
|
read_cod:
|
|
if (H263Info->syntax_arith_coding) {
|
|
if (H263Info->pict_type == H263_PCT_INTER) {
|
|
COD_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_COD);
|
|
COD = H263_codtab[COD_index];
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg,
|
|
"getMBs() Arithmetic Decoding, COD Index: %d COD: %d \n", COD_index, COD) );
|
|
}
|
|
else
|
|
COD = 0; /* COD not used in I-pictures, set to zero */
|
|
}
|
|
else {
|
|
if (H263Info->pict_type == H263_PCT_INTER)
|
|
COD = (int)ScBSPeekBits(BSIn, 1);
|
|
else
|
|
COD = 0; /* Intra picture -> not skipped */
|
|
}
|
|
|
|
if (!COD) { /* COD == 0 --> not skipped */
|
|
if (H263Info->syntax_arith_coding) {
|
|
if (H263Info->pict_type == H263_PCT_INTER) {
|
|
MCBPC_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_MCBPC);
|
|
MCBPC = H263_mcbpctab[MCBPC_index];
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg,
|
|
"getMBs() (arith/inter) MCBPC Index: %ld MCBPC: %d \n",MCBPC_index, MCBPC) );
|
|
}
|
|
else {
|
|
int lastMCBPC=MCBPC;
|
|
MCBPC_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_MCBPC_intra);
|
|
MCBPC = H263_mcbpc_intratab[MCBPC_index];
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg,
|
|
"getMBs() (arith) MCBPC Index: %ld MCBPC: %d \n",MCBPC_index, MCBPC) );
|
|
if (MCBPC==255 && lastMCBPC==255)
|
|
return;
|
|
}
|
|
}
|
|
else {
|
|
if (H263Info->pict_type == H263_PCT_INTER)
|
|
ScBSSkipBit(BSIn); /* flush COD bit */
|
|
if (H263Info->pict_type == H263_PCT_INTRA)
|
|
MCBPC = sv_H263GetMCBPCintra(H263Info, BSIn);
|
|
else
|
|
MCBPC = sv_H263GetMCBPC(H263Info, BSIn);
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg,
|
|
"getMBs() MCBPC: %d \n", MCBPC) );
|
|
}
|
|
|
|
if (H263Info->fault) goto resync;
|
|
|
|
if (MCBPC == 255) { /* stuffing */
|
|
goto read_cod; /* read next COD without advancing MB count */
|
|
}
|
|
|
|
else { /* normal MB data */
|
|
|
|
Mode = MCBPC & 7;
|
|
|
|
/* MODB and CBPB */
|
|
if (H263Info->pb_frame) {
|
|
CBPB = 0;
|
|
if (H263Info->syntax_arith_coding) {
|
|
MODB_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_MODB);
|
|
MODB = H263_modb_tab[MODB_index];
|
|
}
|
|
else
|
|
MODB = sv_H263GetMODB(H263Info, BSIn);
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg, "getMBs() MODB: %d\n", MODB) );
|
|
if (MODB == H263_PBMODE_CBPB_MVDB) {
|
|
if (H263Info->syntax_arith_coding) {
|
|
for(i=0; i<4; i++) {
|
|
YCBPB_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_YCBPB);
|
|
YCBPB = H263_ycbpb_tab[YCBPB_index];
|
|
CBPB |= (YCBPB << (6-1-i));
|
|
}
|
|
|
|
for(i=4; i<6; i++) {
|
|
UVCBPB_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_UVCBPB);
|
|
UVCBPB = H263_uvcbpb_tab[UVCBPB_index];
|
|
CBPB |= (UVCBPB << (6-1-i));
|
|
}
|
|
}
|
|
else
|
|
CBPB = (int)ScBSGetBits(BSIn, 6);
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg, "getMBs() CBPB = %d\n",CBPB) );
|
|
}
|
|
}
|
|
|
|
if (H263Info->syntax_arith_coding) {
|
|
|
|
if (Mode == H263_MODE_INTRA || Mode == H263_MODE_INTRA_Q) { /* Intra */
|
|
CBPY_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_CBPY_intra);
|
|
CBPY = H263_cbpy_intratab[CBPY_index];
|
|
}
|
|
else {
|
|
CBPY_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_CBPY);
|
|
CBPY = H263_cbpytab[CBPY_index];
|
|
|
|
}
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg, "getMBs() CBPY Index: %d CBPY %d \n",CBPY_index, CBPY) );
|
|
}
|
|
else
|
|
{
|
|
CBPY = sv_H263GetCBPY(H263Info, BSIn);
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg, "getMBs() CBPY %d \n",CBPY) );
|
|
}
|
|
|
|
/* Decode Mode and CBP */
|
|
|
|
|
|
if (Mode == H263_MODE_INTRA || Mode == H263_MODE_INTRA_Q)
|
|
{/* Intra */
|
|
if (!H263Info->syntax_arith_coding)
|
|
CBPY = CBPY^15; /* needed in huffman coding only */
|
|
}
|
|
|
|
CBP = (CBPY << 2) | (MCBPC >> 4);
|
|
}
|
|
|
|
if (Mode == H263_MODE_INTER4V && !H263Info->adv_pred_mode)
|
|
{
|
|
_SlibDebug(_VERBOSE_,
|
|
ScDebugPrintf(H263Info->dbg, "getMBs() 8x8 vectors not allowed in normal prediction mode\n") );
|
|
/* Could set fault-flag and resync */
|
|
}
|
|
|
|
|
|
if (Mode == H263_MODE_INTER_Q || Mode == H263_MODE_INTRA_Q) {
|
|
/* Read DQUANT if necessary */
|
|
|
|
if (H263Info->syntax_arith_coding) {
|
|
DQUANT_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_DQUANT);
|
|
DQUANT = H263_dquanttab[DQUANT_index] - 2;
|
|
H263Info->quant +=DQUANT;
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg, "getMBs() DQUANT Index: %d DQUANT %d \n",DQUANT_index, DQUANT) );
|
|
}
|
|
else {
|
|
DQUANT = (int)ScBSGetBits(BSIn, 2);
|
|
H263Info->quant += DQ_tab[DQUANT];
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg, "getMBs() DQUANT: %d\n", DQUANT) );
|
|
#if 0
|
|
if (H263Info->trace) {
|
|
printf("DQUANT (");
|
|
svH263Printbits(DQUANT,2,2);
|
|
printf("): %d = %d\n",DQUANT,DQ_tab[DQUANT]);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
if (H263Info->quant > 31 || H263Info->quant < 1) {
|
|
_SlibDebug(_WARN_, ScDebugPrintf(H263Info->dbg, "GetMBs() Quantizer out of range: clipping\n") );
|
|
H263Info->quant = mmax(1,mmin(31,H263Info->quant));
|
|
/* could set fault-flag and resync here */
|
|
}
|
|
}
|
|
|
|
/* motion vectors */
|
|
if (Mode == H263_MODE_INTER || Mode == H263_MODE_INTER_Q ||
|
|
Mode == H263_MODE_INTER4V || H263Info->pb_frame) {
|
|
|
|
if (Mode == H263_MODE_INTER4V) { startmv = 1; stopmv = 4;}
|
|
else { startmv = 0; stopmv = 0;}
|
|
|
|
for (k = startmv; k <= stopmv; k++) {
|
|
if (H263Info->syntax_arith_coding) {
|
|
mvx_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_MVD);
|
|
mvx = H263_mvdtab[mvx_index];
|
|
mvy_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_MVD);
|
|
mvy = H263_mvdtab[mvy_index];
|
|
_SlibDebug(_WARN_, ScDebugPrintf(H263Info->dbg,
|
|
"GetMBs() mvx_index: %d mvy_index: %d \n", mvx_index, mvy_index) );
|
|
}
|
|
else {
|
|
mvx = sv_H263GetTMNMV(H263Info, BSIn);
|
|
mvy = sv_H263GetTMNMV(H263Info, BSIn);
|
|
}
|
|
|
|
pmv0 = find_pmv(H263Info, xpos,ypos,k,0);
|
|
pmv1 = find_pmv(H263Info, xpos,ypos,k,1);
|
|
mvx = motion_decode(H263Info, mvx, pmv0);
|
|
mvy = motion_decode(H263Info, mvy, pmv1);
|
|
#ifdef CAMERAMOTION
|
|
analyzemotion(xpos,ypos,mvx,mvy);
|
|
#else
|
|
_SlibDebug(_WARN_, ScDebugPrintf(H263Info->dbg,
|
|
"GetMBs() mvx: %d mvy: %d\n", mvx, mvy) );
|
|
#endif
|
|
/* Check mv's to prevent seg.faults when error rate is high */
|
|
if (!H263Info->mv_outside_frame) {
|
|
bsize = k ? 8 : 16;
|
|
offset = k ? (((k-1)&1)<<3) : 0;
|
|
/* checking only integer component */
|
|
if ((xpos<<4) + (mvx/2) + offset < 0 ||
|
|
(xpos<<4) + (mvx/2) + offset > (H263Info->mb_width<<4) - bsize) {
|
|
_SlibDebug(_WARN_, ScDebugPrintf(H263Info->dbg, "GetMBs() mvx out of range: searching for sync\n") );
|
|
H263Info->fault = 1;
|
|
}
|
|
offset = k ? (((k-1)&2)<<2) : 0;
|
|
if ((ypos<<4) + (mvy/2) + offset < 0 ||
|
|
(ypos<<4) + (mvy/2) + offset > (H263Info->mb_height<<4) - bsize) {
|
|
_SlibDebug(_WARN_, ScDebugPrintf(H263Info->dbg, "GetMBs() mvy out of range: searching for sync\n") );
|
|
H263Info->fault = 1;
|
|
}
|
|
}
|
|
H263Info->MV[0][k][ypos+1][xpos+1] = mvx;
|
|
H263Info->MV[1][k][ypos+1][xpos+1] = mvy;
|
|
}
|
|
|
|
/* PB frame delta vectors */
|
|
|
|
if (H263Info->pb_frame) {
|
|
if (MODB == H263_PBMODE_MVDB || MODB == H263_PBMODE_CBPB_MVDB) {
|
|
if (H263Info->syntax_arith_coding) {
|
|
mvdbx_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_MVD);
|
|
mvdbx = H263_mvdtab[mvdbx_index];
|
|
|
|
mvdby_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_MVD);
|
|
mvdby = H263_mvdtab[mvdby_index];
|
|
}
|
|
else {
|
|
mvdbx = sv_H263GetTMNMV(H263Info, BSIn);
|
|
mvdby = sv_H263GetTMNMV(H263Info, BSIn);
|
|
}
|
|
|
|
|
|
mvdbx = motion_decode(H263Info, mvdbx, 0);
|
|
mvdby = motion_decode(H263Info, mvdby, 0);
|
|
/* This will not work if the PB deltas are so large they
|
|
require the second colums of the motion vector VLC
|
|
table to be used. To fix this it is necessary to
|
|
calculate the MV predictor for the PB delta: TRB*MV/TRD
|
|
here, and use this as the second parameter to
|
|
motion_decode(). The B vector itself will then be
|
|
returned from motion_decode(). This will have to be
|
|
changed to the PB delta again, since it is the PB delta
|
|
which is used later in this program. I don't think PB
|
|
deltas outside the range mentioned above is useful, but
|
|
you never know... */
|
|
|
|
}
|
|
else {
|
|
mvdbx = 0;
|
|
mvdby = 0;
|
|
}
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg,
|
|
"GetMBs() MVDB x: %d MVDB y: %d\n", mvdbx, mvdby) );
|
|
}
|
|
}
|
|
|
|
if (H263Info->fault) goto resync;
|
|
|
|
}
|
|
else { /* COD == 1 --> skipped MB */
|
|
if (MBA>=MBAmax)
|
|
{
|
|
#ifdef CAMERAMOTION /* Burkhard */
|
|
goto summarize_motion;
|
|
#else
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() done MBA=%d\n", MBA) );
|
|
return; /* all macroblocks decoded */
|
|
#endif
|
|
}
|
|
if (!H263Info->syntax_arith_coding)
|
|
if (H263Info->pict_type == H263_PCT_INTER)
|
|
ScBSSkipBits(BSIn, 1);
|
|
|
|
Mode = H263_MODE_INTER;
|
|
|
|
/* Reset CBP */
|
|
CBP = CBPB = 0;
|
|
|
|
/* reset motion vectors */
|
|
H263Info->MV[0][0][ypos+1][xpos+1] = 0;
|
|
H263Info->MV[1][0][ypos+1][xpos+1] = 0;
|
|
mvdbx = 0;
|
|
mvdby = 0;
|
|
}
|
|
|
|
/* Store Mode*/
|
|
H263Info->modemap[ypos+1][xpos+1] = Mode;
|
|
|
|
if (Mode == H263_MODE_INTRA || Mode == H263_MODE_INTRA_Q)
|
|
if (!H263Info->pb_frame)
|
|
H263Info->MV[0][0][ypos+1][xpos+1]=H263Info->MV[1][0][ypos+1][xpos+1] = 0;
|
|
|
|
|
|
reconstruct_mb:
|
|
|
|
/* pixel coordinates of top left corner of current macroblock */
|
|
/* one delayed because of OBMC */
|
|
if (xpos > 0) {
|
|
bx = 16*(xpos-1);
|
|
by = 16*ypos;
|
|
}
|
|
else {
|
|
bx = H263Info->coded_picture_width-16;
|
|
by = 16*(ypos-1);
|
|
}
|
|
|
|
if (MBA > 0) {
|
|
|
|
Mode = H263Info->modemap[by/16+1][bx/16+1];
|
|
|
|
/* forward motion compensation for B-frame */
|
|
if (H263Info->pb_frame)
|
|
{
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() sv_H263Reconstruct(B-frame)\n") );
|
|
sv_H263Reconstruct(H263Info,bx,by,0,pmvdbx,pmvdby);
|
|
}
|
|
/* motion compensation for P-frame */
|
|
if (Mode == H263_MODE_INTER || Mode == H263_MODE_INTER_Q || Mode == H263_MODE_INTER4V)
|
|
{
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() sv_H263Reconstruct(P-frame)\n") );
|
|
sv_H263Reconstruct(H263Info,bx,by,1,0,0);
|
|
}
|
|
/* copy or add block data into P-picture */
|
|
for (comp=0; comp<H263Info->blk_cnt; comp++) {
|
|
|
|
bp = H263Info->block[comp];
|
|
|
|
/* inverse DCT */
|
|
if (Mode == H263_MODE_INTRA || Mode == H263_MODE_INTRA_Q) {
|
|
addblock(H263Info, comp,bx,by,0);
|
|
}
|
|
else if ( (pCBP & (1<<(H263Info->blk_cnt-1-comp))) ) {
|
|
/* No need to to do this for blocks with no coeffs */
|
|
addblock(H263Info,comp,bx,by,1);
|
|
}
|
|
}
|
|
|
|
|
|
if (H263Info->pb_frame) {
|
|
/* add block data into B-picture */
|
|
for (comp = 6; comp<H263Info->blk_cnt+6; comp++) {
|
|
if (!pCOD || H263Info->adv_pred_mode)
|
|
reconblock_b(H263Info, comp-6,bx,by,Mode,pmvdbx,pmvdby);
|
|
if ( (pCBPB & (1<<(H263Info->blk_cnt-1-comp%6))) ) {
|
|
bp = H263Info->block[comp];
|
|
addblock(H263Info,comp,bx,by,1);
|
|
}
|
|
}
|
|
}
|
|
|
|
} /* end if (MBA > 0) */
|
|
|
|
if (!COD) {
|
|
|
|
Mode = H263Info->modemap[ypos+1][xpos+1];
|
|
|
|
/* decode blocks */
|
|
for (comp=0; comp<H263Info->blk_cnt; comp++) {
|
|
|
|
clearblock(H263Info, comp);
|
|
if (Mode == H263_MODE_INTRA || Mode == H263_MODE_INTRA_Q) { /* Intra */
|
|
bp = H263Info->block[comp];
|
|
if(H263Info->syntax_arith_coding) {
|
|
INTRADC_index = sv_H263SACDecode_a_symbol(BSIn,H263_cumf_INTRADC);
|
|
bp[0] = H263_intradctab[INTRADC_index];
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg,
|
|
"getMBs() INTRADC Index: %d INTRADC: %d \n", INTRADC_index, bp[0]) );
|
|
}
|
|
else {
|
|
bp[0] = (int)ScBSGetBits(BSIn, 8);
|
|
_SlibDebug(_DEBUG_, ScDebugPrintf(H263Info->dbg, "getMBs() DC: %d \n", (int)bp[0]));
|
|
}
|
|
|
|
_SlibDebug(_WARN_ && bp[0] == 128,
|
|
ScDebugPrintf(H263Info->dbg, "getMBs() Illegal DC-coeff: 1000000\n"));
|
|
if (bp[0] == 255) /* Spec. in H.26P, not in TMN4 */
|
|
bp[0] = 128;
|
|
bp[0] *= 8; /* Iquant */
|
|
|
|
/* DC scale */
|
|
bp[0] *= (int)((float)(1 << 20) * (float)0.125) ;
|
|
|
|
if ( (CBP & (1<<(H263Info->blk_cnt-1-comp))) ) {
|
|
if (!H263Info->syntax_arith_coding)
|
|
sv_H263GetBlock(H263Info,BSIn,comp,0);
|
|
else
|
|
sv_H263GetSACblock(H263Info,BSIn,comp,0);
|
|
}
|
|
}
|
|
else { /* Inter */
|
|
if ( (CBP & (1<<(H263Info->blk_cnt-1-comp))) ) {
|
|
if (!H263Info->syntax_arith_coding)
|
|
sv_H263GetBlock(H263Info,BSIn,comp,1);
|
|
else
|
|
sv_H263GetSACblock(H263Info,BSIn,comp,1);
|
|
}
|
|
|
|
}
|
|
if (H263Info->fault) goto resync;
|
|
}
|
|
|
|
/* Decode B blocks */
|
|
if (H263Info->pb_frame) {
|
|
for (comp=6; comp<H263Info->blk_cnt+6; comp++) {
|
|
clearblock(H263Info, comp);
|
|
if ( (CBPB & (1<<(H263Info->blk_cnt-1-comp%6))) ) {
|
|
if (!H263Info->syntax_arith_coding)
|
|
sv_H263GetBlock(H263Info,BSIn,comp,1);
|
|
else
|
|
sv_H263GetSACblock(H263Info,BSIn,comp,1);
|
|
}
|
|
if (H263Info->fault) goto resync;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* advance to next macroblock */
|
|
MBA++;
|
|
|
|
pCBP = CBP; pCBPB = CBPB; pCOD = COD;
|
|
pmvdbx = mvdbx; pmvdby = mvdby;
|
|
/* fflush(stdout); */
|
|
|
|
if (MBA >= MBAmax && !last_done) {
|
|
COD = 1;
|
|
xpos = 0;
|
|
ypos++;
|
|
last_done = 1;
|
|
goto reconstruct_mb;
|
|
}
|
|
|
|
}
|
|
_SlibDebug(_VERBOSE_, ScDebugPrintf(H263Info->dbg, "getMBs() exiting at MBA=%d\n", MBA) );
|
|
/* Burkhard */
|
|
#ifdef CAMERAMOTION
|
|
summarize_motion:
|
|
summarizemotion(framenum);
|
|
#endif
|
|
}
|
|
|
|
|
|
/* set block to zero */
|
|
static void clearblock(SvH263DecompressInfo_t *H263Info, int comp)
|
|
{
|
|
clearblk(comp) ;
|
|
|
|
#if 0
|
|
qword *bp;
|
|
int i;
|
|
|
|
bp = (qword *)H263Info->block[comp];
|
|
|
|
for (i=0; i<8; i++)
|
|
{
|
|
bp[0] = bp[1] = 0;
|
|
bp += 2;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/* move/add 8x8-Block from block[comp] to refframe or bframe */
|
|
static void addblock(SvH263DecompressInfo_t *H263Info, int comp, int bx, int by, int addflag)
|
|
{
|
|
int cc, iincr, P = 1;
|
|
unsigned char *rfp;
|
|
#ifdef USE_C
|
|
int i ;
|
|
unsigned qword *lp;
|
|
unsigned qword acc,src;
|
|
#endif
|
|
register int *bp;
|
|
|
|
bp = H263Info->block[comp];
|
|
|
|
if (comp >= 6) {
|
|
/* This is a component for B-frame forward prediction */
|
|
P = 0;
|
|
addflag = 1;
|
|
comp -= 6;
|
|
}
|
|
|
|
cc = (comp<4) ? 0 : (comp&1)+1; /* color component index */
|
|
|
|
if (cc==0) {
|
|
/* luminance */
|
|
|
|
/* frame DCT coding */
|
|
if (P)
|
|
rfp = H263Info->newframe[0]
|
|
+ H263Info->coded_picture_width*(by+((comp&2)<<2)) + bx + ((comp&1)<<3);
|
|
else
|
|
rfp = H263Info->bframe[0]
|
|
+ H263Info->coded_picture_width*(by+((comp&2)<<2)) + bx + ((comp&1)<<3);
|
|
iincr = H263Info->coded_picture_width;
|
|
}
|
|
else {
|
|
/* chrominance */
|
|
|
|
/* scale coordinates */
|
|
bx >>= 1;
|
|
by >>= 1;
|
|
/* frame DCT coding */
|
|
if (P)
|
|
rfp = H263Info->newframe[cc] + H263Info->chrom_width*by + bx;
|
|
else
|
|
rfp = H263Info->bframe[cc] + H263Info->chrom_width*by + bx;
|
|
iincr = H263Info->chrom_width;
|
|
}
|
|
|
|
if (addflag) {
|
|
#ifdef USE_C
|
|
ScScaleIDCT8x8i_C(bp, bp);
|
|
for (i=0; i<8; i++) {
|
|
lp = (unsigned qword *) rfp;
|
|
src = lp[0];
|
|
acc = (unsigned qword) H263Info->clp[bp[0]+(src & 0xff)];
|
|
acc |= (unsigned qword) H263Info->clp[bp[1]+((src >> 8) & 0xff)] << 8;
|
|
acc |= (unsigned qword) H263Info->clp[bp[2]+((src >> 16) & 0xff)] << 16;
|
|
acc |= (unsigned qword) H263Info->clp[bp[3]+((src >> 24) & 0xff)] << 24;
|
|
acc |= (unsigned qword) H263Info->clp[bp[4]+((src >> 32) & 0xff)] << 32;
|
|
acc |= (unsigned qword) H263Info->clp[bp[5]+((src >> 40) & 0xff)] << 40;
|
|
acc |= (unsigned qword) H263Info->clp[bp[6]+((src >> 48) & 0xff)] << 48;
|
|
acc |= (unsigned qword) H263Info->clp[bp[7]+((src >> 56) & 0xff)] << 56;
|
|
lp[0] = acc;
|
|
bp += 8;
|
|
rfp+= iincr;
|
|
}
|
|
#else
|
|
sv_H263IDCTAddToFrameP_S(bp,rfp,iincr) ;
|
|
#endif
|
|
}
|
|
else {
|
|
#ifdef USE_C
|
|
ScScaleIDCT8x8i_C(bp, bp);
|
|
for (i=0; i<8; i++) {
|
|
lp = (unsigned qword *) rfp;
|
|
acc = (unsigned qword ) H263Info->clp[bp[0]];
|
|
acc |= (unsigned qword ) H263Info->clp[bp[1]] << 8;
|
|
acc |= (unsigned qword ) H263Info->clp[bp[2]] << 16;
|
|
acc |= (unsigned qword ) H263Info->clp[bp[3]] << 24;
|
|
acc |= (unsigned qword ) H263Info->clp[bp[4]] << 32;
|
|
acc |= (unsigned qword ) H263Info->clp[bp[5]] << 40;
|
|
acc |= (unsigned qword ) H263Info->clp[bp[6]] << 48;
|
|
acc |= (unsigned qword ) H263Info->clp[bp[7]] << 56;
|
|
lp[0] = acc;
|
|
bp += 8;
|
|
rfp += iincr;
|
|
}
|
|
#else
|
|
sv_H263IDCTToFrameP_S(bp,rfp,iincr) ;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
|
|
/* bidirectionally reconstruct 8x8-Block from block[comp] to bframe */
|
|
static void reconblock_b(SvH263DecompressInfo_t *H263Info, int comp, int bx, int by,
|
|
int mode, int bdx, int bdy)
|
|
{
|
|
int cc,i,j,k, ii;
|
|
register unsigned char *bfr, *ffr;
|
|
int BMVx, BMVy;
|
|
int xa,xb,ya,yb,x,y,xvec,yvec,mvx,mvy;
|
|
int xint,xhalf,yint,yhalf,pel;
|
|
|
|
x = bx/16+1;y=by/16+1;
|
|
|
|
if (mode == H263_MODE_INTER4V) {
|
|
if (comp < 4) {
|
|
/* luma */
|
|
mvx = H263Info->MV[0][comp+1][y][x];
|
|
mvy = H263Info->MV[1][comp+1][y][x];
|
|
BMVx = (bdx == 0 ? (H263Info->trb-H263Info->trd)* mvx/H263Info->trd : H263Info->trb * mvx/H263Info->trd + bdx - mvx);
|
|
BMVy = (bdy == 0 ? (H263Info->trb-H263Info->trd)* mvy/H263Info->trd : H263Info->trb * mvy/H263Info->trd + bdy - mvy);
|
|
}
|
|
else {
|
|
/* chroma */
|
|
xvec = yvec = 0;
|
|
for (k = 1; k <= 4; k++) {
|
|
mvx = H263Info->MV[0][k][y][x];
|
|
mvy = H263Info->MV[1][k][y][x];
|
|
xvec += (bdx == 0 ? (H263Info->trb-H263Info->trd)* mvx/H263Info->trd : H263Info->trb * mvx/H263Info->trd + bdx - mvx);
|
|
yvec += (bdy == 0 ? (H263Info->trb-H263Info->trd)* mvy/H263Info->trd : H263Info->trb * mvy/H263Info->trd + bdy - mvy);
|
|
}
|
|
|
|
/* chroma rounding (table 16/H.263) */
|
|
BMVx = sign(xvec)*(H263_roundtab[abs(xvec)%16] + (abs(xvec)/16)*2);
|
|
BMVy = sign(yvec)*(H263_roundtab[abs(yvec)%16] + (abs(yvec)/16)*2);
|
|
}
|
|
}
|
|
else {
|
|
if (comp < 4) {
|
|
/* luma */
|
|
mvx = H263Info->MV[0][0][y][x];
|
|
mvy = H263Info->MV[1][0][y][x];
|
|
BMVx = (bdx == 0 ? (H263Info->trb-H263Info->trd)* mvx/H263Info->trd : H263Info->trb * mvx/H263Info->trd + bdx - mvx);
|
|
BMVy = (bdy == 0 ? (H263Info->trb-H263Info->trd)* mvy/H263Info->trd : H263Info->trb * mvy/H263Info->trd + bdy - mvy);
|
|
}
|
|
else {
|
|
/* chroma */
|
|
mvx = H263Info->MV[0][0][y][x];
|
|
mvy = H263Info->MV[1][0][y][x];
|
|
xvec = (bdx == 0 ? (H263Info->trb-H263Info->trd)* mvx/H263Info->trd : H263Info->trb * mvx/H263Info->trd + bdx - mvx);
|
|
yvec = (bdy == 0 ? (H263Info->trb-H263Info->trd)* mvy/H263Info->trd : H263Info->trb * mvy/H263Info->trd + bdy - mvy);
|
|
xvec *= 4;
|
|
yvec *= 4;
|
|
|
|
/* chroma rounding (table 16/H.263) */
|
|
BMVx = sign(xvec)*(H263_roundtab[abs(xvec)%16] + (abs(xvec)/16)*2);
|
|
BMVy = sign(yvec)*(H263_roundtab[abs(yvec)%16] + (abs(yvec)/16)*2);
|
|
}
|
|
}
|
|
|
|
cc = (comp<4) ? 0 : (comp&1)+1; /* color component index */
|
|
|
|
if (cc==0) {
|
|
/* luminance */
|
|
find_bidir_limits(BMVx,&xa,&xb,comp&1);
|
|
find_bidir_limits(BMVy,&ya,&yb,(comp&2)>>1);
|
|
bfr = H263Info->bframe[0] +
|
|
H263Info->coded_picture_width*(by+((comp&2)<<2)) + bx + ((comp&1)<<3);
|
|
ffr = H263Info->newframe[0] +
|
|
H263Info->coded_picture_width*(by+((comp&2)<<2)) + bx + ((comp&1)<<3);
|
|
ii = H263Info->coded_picture_width;
|
|
}
|
|
else {
|
|
/* chrominance */
|
|
/* scale coordinates and vectors*/
|
|
bx >>= 1;
|
|
by >>= 1;
|
|
|
|
find_bidir_chroma_limits(BMVx,&xa,&xb);
|
|
find_bidir_chroma_limits(BMVy,&ya,&yb);
|
|
|
|
bfr = H263Info->bframe[cc] + H263Info->chrom_width*(by+((comp&2)<<2)) + bx + (comp&8);
|
|
ffr = H263Info->newframe[cc] + H263Info->chrom_width*(by+((comp&2)<<2)) + bx + (comp&8);
|
|
ii = H263Info->chrom_width;
|
|
}
|
|
|
|
xint = BMVx>>1;
|
|
xhalf = BMVx - 2*xint;
|
|
yint = BMVy>>1;
|
|
yhalf = BMVy - 2*yint;
|
|
|
|
ffr += xint + (yint+ya)*ii;
|
|
bfr += ya*ii;
|
|
|
|
if (!xhalf && !yhalf) {
|
|
for (j = ya; j < yb; j++) {
|
|
for (i = xa; i < xb; i++) {
|
|
pel = ffr[i];
|
|
bfr[i] = ((unsigned int)(pel + bfr[i]))>>1;
|
|
}
|
|
bfr += ii;
|
|
ffr += ii;
|
|
}
|
|
}
|
|
else if (xhalf && !yhalf) {
|
|
for (j = ya; j < yb; j++) {
|
|
for (i = xa; i < xb; i++) {
|
|
pel = ((unsigned int)(ffr[i]+ffr[i+1]+1))>>1;
|
|
bfr[i] = ((unsigned int)(pel + bfr[i]))>>1;
|
|
}
|
|
bfr += ii;
|
|
ffr += ii;
|
|
}
|
|
}
|
|
else if (!xhalf && yhalf) {
|
|
for (j = ya; j < yb; j++) {
|
|
for (i = xa; i < xb; i++) {
|
|
pel = ((unsigned int)(ffr[i]+ffr[ii+i]+1))>>1;
|
|
bfr[i] = ((unsigned int)(pel + bfr[i]))>>1;
|
|
}
|
|
bfr += ii;
|
|
ffr += ii;
|
|
}
|
|
}
|
|
else { /* if (xhalf && yhalf) */
|
|
for (j = ya; j < yb; j++) {
|
|
for (i = xa; i < xb; i++) {
|
|
pel = ((unsigned int)(ffr[i]+ffr[i+1]+ffr[ii+i]+ffr[ii+i+1]+2))>>2;
|
|
bfr[i] = ((unsigned int)(pel + bfr[i]))>>1;
|
|
}
|
|
bfr += ii;
|
|
ffr += ii;
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
|
|
static int motion_decode(SvH263DecompressInfo_t *H263Info, int vec, int pmv)
|
|
{
|
|
if (vec > 31) vec -= 64;
|
|
vec += pmv;
|
|
if (!H263Info->long_vectors) {
|
|
if (vec > 31)
|
|
vec -= 64;
|
|
if (vec < -32)
|
|
vec += 64;
|
|
}
|
|
else {
|
|
if (pmv < -31 && vec < -63)
|
|
vec += 64;
|
|
if (pmv > 32 && vec > 63)
|
|
vec -= 64;
|
|
}
|
|
return vec;
|
|
}
|
|
|
|
|
|
|
|
static int find_pmv(SvH263DecompressInfo_t *H263Info, int x, int y, int block, int comp)
|
|
{
|
|
int p1,p2,p3;
|
|
int xin1,xin2,xin3;
|
|
int yin1,yin2,yin3;
|
|
int vec1,vec2,vec3;
|
|
int l8,o8,or8;
|
|
|
|
x++;y++;
|
|
|
|
l8 = (H263Info->modemap[y][x-1] == H263_MODE_INTER4V ? 1 : 0);
|
|
o8 = (H263Info->modemap[y-1][x] == H263_MODE_INTER4V ? 1 : 0);
|
|
or8 = (H263Info->modemap[y-1][x+1] == H263_MODE_INTER4V ? 1 : 0);
|
|
|
|
switch (block) {
|
|
case 0:
|
|
vec1 = (l8 ? 2 : 0) ; yin1 = y ; xin1 = x-1;
|
|
vec2 = (o8 ? 3 : 0) ; yin2 = y-1; xin2 = x;
|
|
vec3 = (or8? 3 : 0) ; yin3 = y-1; xin3 = x+1;
|
|
break;
|
|
case 1:
|
|
vec1 = (l8 ? 2 : 0) ; yin1 = y ; xin1 = x-1;
|
|
vec2 = (o8 ? 3 : 0) ; yin2 = y-1; xin2 = x;
|
|
vec3 = (or8? 3 : 0) ; yin3 = y-1; xin3 = x+1;
|
|
break;
|
|
case 2:
|
|
vec1 = 1 ; yin1 = y ; xin1 = x;
|
|
vec2 = (o8 ? 4 : 0) ; yin2 = y-1; xin2 = x;
|
|
vec3 = (or8? 3 : 0) ; yin3 = y-1; xin3 = x+1;
|
|
break;
|
|
case 3:
|
|
vec1 = (l8 ? 4 : 0) ; yin1 = y ; xin1 = x-1;
|
|
vec2 = 1 ; yin2 = y ; xin2 = x;
|
|
vec3 = 2 ; yin3 = y ; xin3 = x;
|
|
break;
|
|
case 4:
|
|
vec1 = 3 ; yin1 = y ; xin1 = x;
|
|
vec2 = 1 ; yin2 = y ; xin2 = x;
|
|
vec3 = 2 ; yin3 = y ; xin3 = x;
|
|
break;
|
|
default:
|
|
fprintf(stderr,"Illegal block number in find_pmv (getpic.c)\n");
|
|
exit(1);
|
|
break;
|
|
}
|
|
p1 = H263Info->MV[comp][vec1][yin1][xin1];
|
|
p2 = H263Info->MV[comp][vec2][yin2][xin2];
|
|
p3 = H263Info->MV[comp][vec3][yin3][xin3];
|
|
|
|
if (H263Info->newgob && (block == 0 || block == 1 || block == 2))
|
|
p2 = H263_NO_VEC;
|
|
|
|
if (p2 == H263_NO_VEC) { p2 = p3 = p1; }
|
|
|
|
return p1+p2+p3 - mmax(p1,mmax(p2,p3)) - mmin(p1,mmin(p2,p3));
|
|
}
|
|
|
|
|
|
|
|
void find_bidir_limits(int vec, int *start, int *stop, int nhv)
|
|
{
|
|
/* limits taken from C loop in section G5 in H.263 */
|
|
*start = mmax(0,(-vec+1)/2 - nhv*8);
|
|
*stop = mmin(7,15-(vec+1)/2 - nhv*8);
|
|
|
|
(*stop)++; /* I use < and not <= in the loop */
|
|
}
|
|
|
|
|
|
|
|
void find_bidir_chroma_limits(int vec, int *start, int *stop)
|
|
{
|
|
|
|
/* limits taken from C loop in section G5 in H.263 */
|
|
*start = mmax(0,(-vec+1)/2);
|
|
*stop = mmin(7,7-(vec+1)/2);
|
|
|
|
(*stop)++; /* I use < and not <= in the loop */
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
void make_edge_image(unsigned char *src, unsigned char *dst, int width, int height, int edge)
|
|
{
|
|
int i,j,incrp,incro;
|
|
register unsigned char *p1,*p2,*p3,*p4;
|
|
register unsigned char *o1,*o2,*o3,*o4;
|
|
register unsigned int *dp1, *do1 ;
|
|
|
|
/* center image */
|
|
/*
|
|
p1 = dst;
|
|
o1 = src;
|
|
for (j = 0; j < height;j++) {
|
|
for (i = 0; i < width; i++) {
|
|
*(p1 + i) = *(o1 + i);
|
|
}
|
|
p1 += width + (edge<<1);
|
|
o1 += width;
|
|
}
|
|
*/
|
|
dp1 = (unsigned int *)dst ;
|
|
do1 = (unsigned int *)src ;
|
|
incrp = (width + (edge<<1))/4 ;
|
|
incro = width/4 ;
|
|
for (j = 0; j < height;j++) {
|
|
for (i = 0; i < width/4; i++) {
|
|
*(dp1 + i) = *(do1 + i);
|
|
}
|
|
dp1 += incrp;
|
|
do1 += incro;
|
|
}
|
|
|
|
/* left and right edges */
|
|
p1 = dst-1;
|
|
o1 = src;
|
|
incrp = width + (edge<<1) ;
|
|
incro = width ;
|
|
for (j = 0; j < height;j++) {
|
|
for (i = 0; i < edge; i++) {
|
|
*(p1 - i) = *o1;
|
|
*(p1 + width + i + 1) = *(o1 + width - 1);
|
|
}
|
|
p1 += incrp;
|
|
o1 += incro;
|
|
}
|
|
|
|
/* top and bottom edges */
|
|
p1 = dst;
|
|
p2 = dst + (width + (edge<<1))*(height-1);
|
|
o1 = src;
|
|
o2 = src + width*(height-1);
|
|
incrp = (width + (edge<<1)) ;
|
|
for (j = 0; j < edge;j++) {
|
|
p1 -= incrp ;
|
|
p2 += incrp ;
|
|
for (i = 0; i < width; i++) {
|
|
*(p1 + i) = *(o1 + i);
|
|
*(p2 + i) = *(o2 + i);
|
|
}
|
|
}
|
|
|
|
/* corners */
|
|
p1 = dst - (width+(edge<<1)) - 1;
|
|
p2 = p1 + width + 1;
|
|
p3 = dst + (width+(edge<<1))*(height)-1;
|
|
p4 = p3 + width + 1;
|
|
|
|
o1 = src;
|
|
o2 = o1 + width - 1;
|
|
o3 = src + width*(height-1);
|
|
o4 = o3 + width - 1;
|
|
incrp = (width + (edge<<1)) ;
|
|
for (j = 0; j < edge; j++) {
|
|
for (i = 0; i < edge; i++) {
|
|
*(p1 - i) = *o1;
|
|
*(p2 + i) = *o2;
|
|
*(p3 - i) = *o3;
|
|
*(p4 + i) = *o4;
|
|
}
|
|
p1 -= incrp ;
|
|
p2 -= incrp ;
|
|
p3 += incrp ;
|
|
p4 += incrp ;
|
|
}
|
|
}
|
|
|
|
|
|
/* only used for displayed interpolated frames, not reconstructed ones */
|
|
void interpolate_image(unsigned char *in, unsigned char *out, int width, int height)
|
|
{
|
|
register int x,xx,y,w2;
|
|
register unsigned char *pp,*ii;
|
|
|
|
return ;
|
|
|
|
w2 = 2*width;
|
|
|
|
/* Horizontally */
|
|
pp = out;
|
|
ii = in;
|
|
for (y = 0; y < height-1; y++) {
|
|
for (x = 0,xx=0; x < width-1; x++,xx+=2) {
|
|
*(pp + xx) = *(ii + x);
|
|
*(pp + xx+1) = ((unsigned int)(*(ii + x) + *(ii + x + 1)))>>1;
|
|
*(pp + w2 + xx) = ((unsigned int)(*(ii + x) + *(ii + x + width)))>>1;
|
|
*(pp + w2 + xx+1) = ((unsigned int)(*(ii + x) + *(ii + x + 1) +
|
|
*(ii + x + width) + *(ii + x + width + 1)))>>2;
|
|
|
|
}
|
|
*(pp + w2 - 2) = *(ii + width - 1);
|
|
*(pp + w2 - 1) = *(ii + width - 1);
|
|
*(pp + w2 + w2 - 2) = *(ii + width + width - 1);
|
|
*(pp + w2 + w2 - 1) = *(ii + width + width - 1);
|
|
pp += w2<<1;
|
|
ii += width;
|
|
}
|
|
|
|
/* last lines */
|
|
for (x = 0,xx=0; x < width-1; x++,xx+=2) {
|
|
*(pp+ xx) = *(ii + x);
|
|
*(pp+ xx+1) = ((unsigned int)(*(ii + x) + *(ii + x + 1) + 1))>>1;
|
|
*(pp+ w2+ xx) = *(ii + x);
|
|
*(pp+ w2+ xx+1) = ((unsigned int)(*(ii + x) + *(ii + x + 1) + 1))>>1;
|
|
}
|
|
|
|
/* bottom right corner pels */
|
|
*(pp + (width<<1) - 2) = *(ii + width -1);
|
|
*(pp + (width<<1) - 1) = *(ii + width -1);
|
|
*(pp + (width<<2) - 2) = *(ii + width -1);
|
|
*(pp + (width<<2) - 1) = *(ii + width -1);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|