2013-04-24 07:35:27 +00:00
|
|
|
/**
|
|
|
|
* HEVC Encoder
|
|
|
|
* - Marko Viitanen ( fador at iki.fi ), Tampere University of Technology, Department of Pervasive Computing.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*! \file inter.c
|
|
|
|
\brief Inter functions
|
|
|
|
\author Marko Viitanen
|
|
|
|
\date 2013-04
|
|
|
|
|
|
|
|
Inter functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "global.h"
|
|
|
|
#include "config.h"
|
|
|
|
#include "encoder.h"
|
|
|
|
#include "picture.h"
|
|
|
|
#include "inter.h"
|
|
|
|
|
|
|
|
/*!
|
|
|
|
\brief Set block mode (and init typedata)
|
|
|
|
\param pic picture to use
|
|
|
|
\param xCtb x CU position (smallest CU)
|
|
|
|
\param yCtb y CU position (smallest CU)
|
|
|
|
\param depth current CU depth
|
|
|
|
\param mode mode to set
|
|
|
|
\returns Void
|
|
|
|
*/
|
2013-09-12 13:28:40 +00:00
|
|
|
void inter_setBlockMode(picture* pic,uint32_t xCtb, uint32_t yCtb, uint8_t depth, CU_info* cur_cu)
|
2013-04-24 07:35:27 +00:00
|
|
|
{
|
|
|
|
uint32_t x,y,d;
|
|
|
|
/* Width in smallest CU */
|
2013-06-12 12:41:57 +00:00
|
|
|
int width_in_SCU = pic->width_in_LCU<<MAX_DEPTH;
|
2013-04-24 07:35:27 +00:00
|
|
|
int block_SCU_width = (LCU_WIDTH>>depth)/(LCU_WIDTH>>MAX_DEPTH);
|
|
|
|
for(y = yCtb; y < yCtb+block_SCU_width; y++)
|
|
|
|
{
|
|
|
|
int CUpos = y*width_in_SCU;
|
|
|
|
for(x = xCtb; x < xCtb+block_SCU_width; x++)
|
2013-09-05 12:02:53 +00:00
|
|
|
{
|
2013-04-24 07:35:27 +00:00
|
|
|
for(d = 0; d < MAX_DEPTH+1; d++)
|
|
|
|
{
|
|
|
|
pic->CU[d][CUpos+x].depth = depth;
|
2013-09-05 12:02:53 +00:00
|
|
|
pic->CU[d][CUpos+x].type = CU_INTER;
|
2013-09-12 13:28:40 +00:00
|
|
|
pic->CU[d][CUpos+x].inter.mode = cur_cu->inter.mode;
|
|
|
|
pic->CU[d][CUpos+x].inter.mv[0] = cur_cu->inter.mv[0];
|
|
|
|
pic->CU[d][CUpos+x].inter.mv[1] = cur_cu->inter.mv[1];
|
|
|
|
pic->CU[d][CUpos+x].inter.mv_dir = cur_cu->inter.mv_dir;
|
2013-04-24 07:35:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-12 15:50:11 +00:00
|
|
|
/*!
|
|
|
|
\brief Reconstruct inter block
|
|
|
|
\param ref picture to copy the data from
|
|
|
|
\param xpos block x position
|
|
|
|
\param ypos block y position
|
|
|
|
\param width block width
|
|
|
|
\param mv[2] motion vector
|
|
|
|
\param dst destination picture
|
|
|
|
\returns Void
|
|
|
|
*/
|
|
|
|
void inter_recon(picture* ref,int32_t xpos, int32_t ypos,int32_t width, int16_t mv[2], picture* dst)
|
|
|
|
{
|
|
|
|
int x,y,coord_x,coord_y;
|
|
|
|
|
|
|
|
/* negative overflow present */
|
|
|
|
int8_t overflow_neg_x = (xpos+mv[0] < 0)?1:0;
|
|
|
|
int8_t overflow_neg_y = (ypos+mv[1] < 0)?1:0;
|
|
|
|
|
|
|
|
/* positive overflow present */
|
|
|
|
int8_t overflow_pos_x = (xpos+mv[0]+width > ref->width )?1:0;
|
|
|
|
int8_t overflow_pos_y = (ypos+mv[1]+width > ref->height)?1:0;
|
|
|
|
|
|
|
|
/* TODO: Fractional pixel support */
|
|
|
|
mv[0] = mv[0]>>2;
|
|
|
|
mv[1] = mv[1]>>2;
|
|
|
|
|
|
|
|
/* With overflow present, more checking */
|
2013-09-16 14:15:54 +00:00
|
|
|
if (overflow_neg_x || overflow_neg_y || overflow_pos_x || overflow_pos_y) {
|
2013-09-12 15:50:11 +00:00
|
|
|
/* Copy Luma with boundary checking */
|
2013-09-16 14:15:54 +00:00
|
|
|
for (y = ypos; y < ypos+width; y++) {
|
|
|
|
for (x = xpos; x < xpos+width; x++) {
|
2013-09-12 15:50:11 +00:00
|
|
|
coord_x = x;
|
|
|
|
coord_y = y;
|
2013-09-16 14:15:54 +00:00
|
|
|
overflow_neg_x = (x+mv[0] < 0)?1:0;
|
|
|
|
overflow_neg_y = (y+mv[1] < 0)?1:0;
|
2013-09-12 15:50:11 +00:00
|
|
|
|
2013-09-16 14:15:54 +00:00
|
|
|
overflow_pos_x = (x+mv[0] >= ref->width )?1:0;
|
|
|
|
overflow_pos_y = (y+mv[1] >= ref->height)?1:0;
|
2013-09-12 15:50:11 +00:00
|
|
|
|
|
|
|
if(overflow_neg_x) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_x = -mv[0];
|
2013-09-12 15:50:11 +00:00
|
|
|
} else if(overflow_pos_x) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_x = ref->width-1-mv[0];
|
2013-09-12 15:50:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(overflow_neg_y) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_y = -mv[1];
|
2013-09-12 15:50:11 +00:00
|
|
|
} else if(overflow_pos_y) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_y = ref->height-1-mv[1];
|
2013-09-12 15:50:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
dst->yRecData[y*dst->width+x] = ref->yRecData[(coord_y+mv[1])*ref->width+(coord_x+mv[0])];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy Chroma with boundary checking */
|
2013-09-16 14:15:54 +00:00
|
|
|
for (y = ypos>>1; y < (ypos+width)>>1; y++) {
|
|
|
|
for (x = xpos>>1; x < (xpos+width)>>1; x++) {
|
2013-09-12 15:50:11 +00:00
|
|
|
coord_x = x;
|
|
|
|
coord_y = y;
|
2013-09-16 14:15:54 +00:00
|
|
|
overflow_neg_x = (x+(mv[0]>>1) < 0)?1:0;
|
|
|
|
overflow_neg_y = (y+(mv[1]>>1) < 0)?1:0;
|
2013-09-12 15:50:11 +00:00
|
|
|
|
2013-09-16 14:15:54 +00:00
|
|
|
overflow_pos_x = (x+(mv[0]>>1) >= ref->width>>1 )?1:0;
|
|
|
|
overflow_pos_y = (y+(mv[1]>>1) >= ref->height>>1)?1:0;
|
2013-09-12 15:50:11 +00:00
|
|
|
|
|
|
|
if(overflow_neg_x) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_x = -(mv[0]>>1);
|
2013-09-12 15:50:11 +00:00
|
|
|
} else if(overflow_pos_x) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_x = ((ref->width-mv[0])>>1)-1;
|
2013-09-12 15:50:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(overflow_neg_y) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_y = -(mv[1]>>1);
|
2013-09-12 15:50:11 +00:00
|
|
|
} else if(overflow_pos_y) {
|
2013-09-16 14:15:54 +00:00
|
|
|
coord_y = ((ref->height-mv[1])>>1)-1;
|
2013-09-12 15:50:11 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 14:15:54 +00:00
|
|
|
dst->uRecData[y*(dst->width>>1)+x] = ref->uRecData[(coord_y+(mv[1]>>1))*(ref->width>>1)+(coord_x+(mv[0]>>1))];
|
|
|
|
dst->vRecData[y*(dst->width>>1)+x] = ref->vRecData[(coord_y+(mv[1]>>1))*(ref->width>>1)+(coord_x+(mv[0]>>1))];
|
2013-09-12 15:50:11 +00:00
|
|
|
}
|
|
|
|
}
|
2013-09-16 14:15:54 +00:00
|
|
|
} else {
|
2013-09-12 15:50:11 +00:00
|
|
|
/* Copy Luma */
|
2013-09-16 14:15:54 +00:00
|
|
|
for (y = ypos; y < ypos+width; y++) {
|
|
|
|
for (x = xpos; x < xpos+width; x++) {
|
2013-09-12 15:50:11 +00:00
|
|
|
dst->yRecData[y*dst->width+x] = ref->yRecData[(y+mv[1])*ref->width+x+mv[0]];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy Chroma */
|
2013-09-16 14:15:54 +00:00
|
|
|
for (y = ypos>>1; y < (ypos+width)>>1; y++) {
|
|
|
|
for (x = xpos>>1; x < (xpos+width)>>1; x++) {
|
2013-09-12 15:50:11 +00:00
|
|
|
dst->uRecData[y*(dst->width>>1)+x] = ref->uRecData[(y+(mv[1]>>1))*(ref->width>>1)+x+(mv[0]>>1)];
|
|
|
|
dst->vRecData[y*(dst->width>>1)+x] = ref->vRecData[(y+(mv[1]>>1))*(ref->width>>1)+x+(mv[0]>>1)];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-09-16 13:37:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
\brief Get MV prediction for current block
|
|
|
|
\param encoder encoder control struct to use
|
|
|
|
\param xCtb block x position in SCU
|
|
|
|
\param yCtb block y position in SCU
|
|
|
|
\param depth current block depth
|
|
|
|
\param mv_pred[2][2] 2x motion vector prediction
|
|
|
|
\returns Void
|
|
|
|
*/
|
|
|
|
void inter_get_mv_cand(encoder_control *encoder,int32_t xCtb, int32_t yCtb,int8_t depth, int16_t mv_cand[2][2])
|
|
|
|
{
|
|
|
|
uint8_t cur_block_in_scu = (LCU_WIDTH>>depth) / CU_MIN_SIZE_PIXELS;
|
|
|
|
uint8_t candidates = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
Predictor block locations
|
2013-09-18 07:15:05 +00:00
|
|
|
____ _______
|
2013-09-16 13:37:24 +00:00
|
|
|
|B2|______|B1|B0|
|
|
|
|
| |
|
|
|
|
| Cur CU |
|
|
|
|
__| |
|
|
|
|
|A1|_________|
|
|
|
|
|A0|
|
|
|
|
*/
|
|
|
|
CU_info *b0, *b1, *b2, *a0, *a1;
|
|
|
|
|
|
|
|
b0 = b1 = b2 = a0 = a1 = NULL;
|
|
|
|
|
|
|
|
if (xCtb != 0) {
|
|
|
|
a1 = &encoder->in.cur_pic->CU[depth][xCtb-1+(yCtb+cur_block_in_scu-1)*(encoder->in.width_in_LCU<<MAX_DEPTH)];
|
|
|
|
if(!a1->coded) a1 = NULL;
|
|
|
|
|
|
|
|
if (yCtb+cur_block_in_scu < encoder->in.height_in_LCU<<MAX_DEPTH) {
|
|
|
|
a0 = &encoder->in.cur_pic->CU[depth][xCtb-1+(yCtb+cur_block_in_scu)*(encoder->in.width_in_LCU<<MAX_DEPTH)];
|
|
|
|
if(!a0->coded) a0 = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (yCtb != 0) {
|
|
|
|
b0 = &encoder->in.cur_pic->CU[depth][xCtb+cur_block_in_scu+(yCtb-1)*(encoder->in.width_in_LCU<<MAX_DEPTH)];
|
|
|
|
if (!b0->coded) b0 = NULL;
|
|
|
|
b1 = &encoder->in.cur_pic->CU[depth][xCtb+cur_block_in_scu-1+(yCtb-1)*(encoder->in.width_in_LCU<<MAX_DEPTH)];
|
|
|
|
if (!b1->coded) b1 = NULL;
|
|
|
|
|
|
|
|
if (xCtb != 0) {
|
|
|
|
b2 = &encoder->in.cur_pic->CU[depth][xCtb-1+(yCtb-1)*(encoder->in.width_in_LCU<<MAX_DEPTH)];
|
|
|
|
if(!b2->coded) b2 = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Left predictors */
|
|
|
|
if (a0 && a0->type == CU_INTER) {
|
|
|
|
mv_cand[candidates][0] = a0->inter.mv[0];
|
|
|
|
mv_cand[candidates][1] = a0->inter.mv[1];
|
|
|
|
candidates++;
|
|
|
|
} else if (a1 && a1->type == CU_INTER) {
|
|
|
|
mv_cand[candidates][0] = a1->inter.mv[0];
|
|
|
|
mv_cand[candidates][1] = a1->inter.mv[1];
|
|
|
|
candidates++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Top predictors */
|
|
|
|
if (b0 && b0->type == CU_INTER) {
|
|
|
|
mv_cand[candidates][0] = b0->inter.mv[0];
|
|
|
|
mv_cand[candidates][1] = b0->inter.mv[1];
|
|
|
|
candidates++;
|
|
|
|
} else if (b1 && b1->type == CU_INTER) {
|
|
|
|
mv_cand[candidates][0] = b1->inter.mv[0];
|
|
|
|
mv_cand[candidates][1] = b1->inter.mv[1];
|
|
|
|
candidates++;
|
|
|
|
} else if(b2 && b2->type == CU_INTER) {
|
|
|
|
mv_cand[candidates][0] = b2->inter.mv[0];
|
|
|
|
mv_cand[candidates][1] = b2->inter.mv[1];
|
|
|
|
candidates++;
|
|
|
|
}
|
|
|
|
|
2013-09-18 07:15:05 +00:00
|
|
|
/* Remove identical candidate */
|
|
|
|
if(candidates == 2 && mv_cand[0][0] == mv_cand[1][0] && mv_cand[0][1] == mv_cand[1][1]) {
|
|
|
|
candidates = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ENABLE_TEMPORAL_MVP
|
2013-09-16 13:37:24 +00:00
|
|
|
if(candidates < 2) {
|
|
|
|
//TODO: add temporal mv predictor
|
|
|
|
}
|
2013-09-18 07:15:05 +00:00
|
|
|
#endif
|
2013-09-16 13:37:24 +00:00
|
|
|
|
2013-09-18 07:15:05 +00:00
|
|
|
while (candidates < 2) {
|
2013-09-16 13:37:24 +00:00
|
|
|
mv_cand[candidates][0] = 0;
|
|
|
|
mv_cand[candidates][1] = 0;
|
|
|
|
candidates++;
|
|
|
|
}
|
|
|
|
|
2013-09-12 15:50:11 +00:00
|
|
|
}
|