/** * HEVC Encoder * - Marko Viitanen ( fador at iki.fi ), Tampere University of Technology, Department of Pervasive Computing. */ /*! \file search.c \brief searching \author Marko Viitanen \date 2013-04 Search related functions */ #include #include #include #include "global.h" #include "config.h" #include "bitstream.h" #include "picture.h" #include "intra.h" #include "encoder.h" #include "filter.h" #include "search.h" void search_tree(encoder_control* encoder,uint16_t xCtb,uint16_t yCtb, uint8_t depth) { uint8_t border_x = ((encoder->in.width)<( xCtb*(LCU_WIDTH>>MAX_DEPTH) + (LCU_WIDTH>>depth) ))?1:0; uint8_t border_y = ((encoder->in.height)<( yCtb*(LCU_WIDTH>>MAX_DEPTH) + (LCU_WIDTH>>depth) ))?1:0; uint8_t border = border_x | border_y; /*!< are we in any border CU */ CU_info *cur_CU = &encoder->in.cur_pic.CU[depth][(xCtb>>(MAX_DEPTH-depth))+(yCtb>>(MAX_DEPTH-depth))*(encoder->in.width_in_LCU<intra.cost = 0xffffffff; /* Force split on border */ if(depth != MAX_DEPTH) { if(border) { /* Split blocks and remember to change x and y block positions */ uint8_t change = 1<<(MAX_DEPTH-1-depth); SET_SPLITDATA(cur_CU,1); search_tree(encoder,xCtb,yCtb,depth+1); if(!border_x) { search_tree(encoder,xCtb+change,yCtb,depth+1); } if(!border_y) { search_tree(encoder,xCtb,yCtb+change,depth+1); } if(!border) { search_tree(encoder,xCtb+change,yCtb+change,depth+1); } /* We don't need to do anything else here */ return; } } if(encoder->in.cur_pic.slicetype != SLICE_I) { } /* INTRA SEARCH */ if(depth >= MIN_SEARCH_DEPTH) { int x,y; uint8_t *base = &encoder->in.cur_pic.yData[xCtb*(LCU_WIDTH>>(MAX_DEPTH)) + (yCtb*(LCU_WIDTH>>(MAX_DEPTH))) *encoder->in.width]; uint32_t width = LCU_WIDTH>>depth; /* INTRAPREDICTION */ /* ToDo: split to a function */ int16_t pred[LCU_WIDTH*LCU_WIDTH]; int16_t rec[(LCU_WIDTH*2+8)*(LCU_WIDTH*2+8)]; int16_t *recShift = &rec[(LCU_WIDTH>>(depth))*2+8+1]; /* Use original pic as reference border for prediction */ for(x = 0; x < LCU_WIDTH*2 && x < encoder->in.width-xCtb*(LCU_WIDTH>>(MAX_DEPTH)); x++) { rec[x] = base[x]; } for(;x < LCU_WIDTH*2; x++) { rec[x] = 1<<(g_bitDepth-1); } for(y = 1; y < LCU_WIDTH*2 && y < encoder->in.height-yCtb*(LCU_WIDTH>>(MAX_DEPTH)); y++) { rec[y*((LCU_WIDTH>>(depth))*2+8)] = base[y*encoder->in.width]; } for(;y < LCU_WIDTH*2; y++) { rec[y*((LCU_WIDTH>>(depth))*2+8)] = 1<<(g_bitDepth-1); } /* Build reconstructed block to use in prediction with extrapolated borders */ //intra_buildReferenceBorder(&encoder->in.cur_pic, xCtb, yCtb,(LCU_WIDTH>>(depth))*2+8, rec, (LCU_WIDTH>>(depth))*2+8, 0); cur_CU->intra.mode = (uint8_t)intra_prediction(encoder->in.cur_pic.yData,encoder->in.width,recShift,(LCU_WIDTH>>(depth))*2+8,xCtb*(LCU_WIDTH>>(MAX_DEPTH)),yCtb*(LCU_WIDTH>>(MAX_DEPTH)),width,pred,width,&cur_CU->intra.cost); } /* Split and search to max_depth */ if(depth != MAX_SEARCH_DEPTH) { /* Split blocks and remember to change x and y block positions */ uint8_t change = 1<<(MAX_DEPTH-1-depth); search_tree(encoder,xCtb,yCtb,depth+1); search_tree(encoder,xCtb+change,yCtb,depth+1); search_tree(encoder,xCtb,yCtb+change,depth+1); search_tree(encoder,xCtb+change,yCtb+change,depth+1); } } uint32_t search_best_mode(encoder_control* encoder,uint16_t xCtb,uint16_t yCtb, uint8_t depth) { CU_info *cur_CU = &encoder->in.cur_pic.CU[depth][(xCtb>>(MAX_DEPTH-depth))+(yCtb>>(MAX_DEPTH-depth))*(encoder->in.width_in_LCU<intra.cost; int8_t bestMode = cur_CU->type; uint32_t cost = 0; uint32_t lambdaCost = g_lambda_cost[encoder->QP]<<9;//<<16; /* Split and search to max_depth */ if(depth != MAX_SEARCH_DEPTH) { /* Split blocks and remember to change x and y block positions */ uint8_t change = 1<<(MAX_DEPTH-1-depth); cost = search_best_mode(encoder,xCtb,yCtb,depth+1); cost += search_best_mode(encoder,xCtb+change,yCtb,depth+1); cost += search_best_mode(encoder,xCtb,yCtb+change,depth+1); cost += search_best_mode(encoder,xCtb+change,yCtb+change,depth+1); /* We split if the cost is better */ if(cost != 0 && cost+lambdaCost < bestCost) { cur_CU->split = 1; bestCost = cost+lambdaCost; } /* Else, dont split and recursively set block mode */ else { cur_CU->split = 0; intra_setBlockMode(&encoder->in.cur_pic,xCtb,yCtb,depth,cur_CU->intra.mode); } } else { cur_CU->split = 0; intra_setBlockMode(&encoder->in.cur_pic,xCtb,yCtb,depth,cur_CU->intra.mode); } return bestCost; } void search_slice_data(encoder_control* encoder) { uint16_t xCtb,yCtb; /* Loop through every LCU in the slice */ for(yCtb = 0; yCtb < encoder->in.height_in_LCU; yCtb++) { uint8_t lastCUy = (yCtb == (encoder->in.height_in_LCU-1))?1:0; for(xCtb = 0; xCtb < encoder->in.width_in_LCU; xCtb++) { uint8_t lastCUx = (xCtb == (encoder->in.width_in_LCU-1))?1:0; uint8_t depth = 0; /* Recursive function for looping through all the sub-blocks */ search_tree(encoder, xCtb<