root/libde265/encoder/algo/cb-intrapartmode.cc

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. analyze
  2. analyze

/*
 * H.265 video codec.
 * Copyright (c) 2013-2014 struktur AG, Dirk Farin <farin@struktur.de>
 *
 * Authors: struktur AG, Dirk Farin <farin@struktur.de>
 *
 * This file is part of libde265.
 *
 * libde265 is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation, either version 3 of
 * the License, or (at your option) any later version.
 *
 * libde265 is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with libde265.  If not, see <http://www.gnu.org/licenses/>.
 */


#include "libde265/encoder/algo/cb-intrapartmode.h"
#include "libde265/encoder/algo/coding-options.h"
#include "libde265/encoder/encoder-context.h"
#include <assert.h>
#include <limits>
#include <math.h>
#include <iostream>


#define ENCODER_DEVELOPMENT 1



enc_cb* Algo_CB_IntraPartMode_BruteForce::analyze(encoder_context* ectx,
                                                  context_model_table& ctxModel,
                                                  enc_cb* cb_in)
{
  const int log2CbSize = cb_in->log2Size;
  const int x = cb_in->x;
  const int y = cb_in->y;

  const bool can_use_NxN = ((log2CbSize == ectx->get_sps().Log2MinCbSizeY) &&
                            (log2CbSize >  ectx->get_sps().Log2MinTrafoSize));


  // test all modes

  assert(cb_in->pcm_flag==0);


  // 0: 2Nx2N  (always checked)
  // 1:  NxN   (only checked at MinCbSize)

  CodingOptions<enc_cb> options(ectx,cb_in,ctxModel);
  CodingOption<enc_cb> option[2];
  option[0] = options.new_option(true);
  option[1] = options.new_option(can_use_NxN);

  options.start();

  for (int p=0;p<2;p++)
    if (option[p]) {
      option[p].begin();

      enc_cb* cb = option[p].get_node();
      *(cb_in->downPtr) = cb;

      // --- set intra prediction mode ---

      cb->PartMode = (p==0 ? PART_2Nx2N : PART_NxN);

      ectx->img->set_pred_mode(x,y, log2CbSize, cb->PredMode);  // TODO: probably unnecessary
      ectx->img->set_PartMode (x,y, cb->PartMode);


      // encode transform tree

      int IntraSplitFlag= (cb->PredMode == MODE_INTRA && cb->PartMode == PART_NxN);
      int MaxTrafoDepth = ectx->get_sps().max_transform_hierarchy_depth_intra + IntraSplitFlag;

      descend(cb,p==0 ? "2Nx2N" : "NxN");

      enc_tb* tb = new enc_tb(x,y,log2CbSize,cb);
      tb->downPtr = &cb->transform_tree;

      cb->transform_tree = mTBIntraPredModeAlgo->analyze(ectx, option[p].get_context(),
                                                         ectx->imgdata->input, tb,
                                                         0, MaxTrafoDepth, IntraSplitFlag);

      ascend();

      cb->distortion = cb->transform_tree->distortion;
      cb->rate       = cb->transform_tree->rate;


      // rate for cu syntax

      logtrace(LogSymbols,"$1 part_mode=%d\n",cb->PartMode);
      if (log2CbSize == ectx->get_sps().Log2MinCbSizeY) {
        int bin = (cb->PartMode==PART_2Nx2N);
        option[p].get_cabac()->reset();
        option[p].get_cabac()->write_CABAC_bit(CONTEXT_MODEL_PART_MODE+0, bin);
        cb->rate += option[p].get_cabac()->getRDBits();
      }

      option[p].end();
    }

  options.compute_rdo_costs();
  enc_cb* bestCB = options.return_best_rdo_node();

  return bestCB;
}


enc_cb* Algo_CB_IntraPartMode_Fixed::analyze(encoder_context* ectx,
                                             context_model_table& ctxModel,
                                             enc_cb* cb)
{
  enum PartMode PartMode = mParams.partMode();


  const int log2CbSize = cb->log2Size;
  const int x = cb->x;
  const int y = cb->y;


  // NxN can only be applied at minimum CB size.
  // If we are not at the minimum size, we have to use 2Nx2N.

  if (PartMode==PART_NxN && log2CbSize != ectx->get_sps().Log2MinCbSizeY) {
    PartMode = PART_2Nx2N;
  }


  // --- set intra prediction mode ---

  cb->PartMode = PartMode;
  ectx->img->set_PartMode(x,y, PartMode);


  // encode transform tree

  int IntraSplitFlag= (cb->PredMode == MODE_INTRA && cb->PartMode == PART_NxN);
  int MaxTrafoDepth = ectx->get_sps().max_transform_hierarchy_depth_intra + IntraSplitFlag;

  enc_tb* tb = new enc_tb(x,y,log2CbSize,cb);
  tb->blkIdx = 0;
  tb->downPtr = &cb->transform_tree;

  descend(cb,"fixed:%s", (PartMode==PART_2Nx2N ? "2Nx2N":"NxN"));
  cb->transform_tree = mTBIntraPredModeAlgo->analyze(ectx, ctxModel,
                                                     ectx->imgdata->input, tb,
                                                     0, MaxTrafoDepth, IntraSplitFlag);
  ascend();


  // rate and distortion for this CB

  cb->distortion = cb->transform_tree->distortion;
  cb->rate       = cb->transform_tree->rate;


  // rate for cu syntax

  CABAC_encoder_estim estim;
  estim.set_context_models(&ctxModel);

  //encode_coding_unit(ectx,&estim,cb,x,y,log2CbSize, false);

  //encode_part_mode(ectx,&estim, MODE_INTRA, PartMode, 0);

  logtrace(LogSymbols,"$1 part_mode=%d\n",PartMode);
  if (log2CbSize == ectx->get_sps().Log2MinCbSizeY) {
    int bin = (PartMode==PART_2Nx2N);
    estim.write_CABAC_bit(CONTEXT_MODEL_PART_MODE+0, bin);
  }

  cb->rate += estim.getRDBits();

  return cb;
}

/* [<][>][^][v][top][bottom][index][help] */