/*
* Copyright (c) 2011 Intel Corporation. All Rights Reserved.
* Copyright (c) Imagination Technologies Limited, UK
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sub license, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice (including the
* next paragraph) shall be included in all copies or substantial portions
* of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
* IN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR
* ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*
* Authors:
* Elaine Wang <elaine.wang@intel.com>
* Zeng Li <zeng.li@intel.com>
*
*/
#include <errno.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdint.h>
#include <string.h>
#include "psb_def.h"
#include "psb_drv_debug.h"
#include "psb_surface.h"
#include "psb_cmdbuf.h"
#include "pnw_hostcode.h"
#include "pnw_H264ES.h"
#include "pnw_hostheader.h"
#include "va/va_enc_h264.h"
#define TOPAZ_H264_MAX_BITRATE 50000000
#define INIT_CONTEXT_H264ES context_ENC_p ctx = (context_ENC_p) obj_context->format_data
#define SURFACE(id) ((object_surface_p) object_heap_lookup( &ctx->obj_context->driver_data->surface_heap, id ))
#define BUFFER(id) ((object_buffer_p) object_heap_lookup( &ctx->obj_context->driver_data->buffer_heap, id ))
static void pnw_H264ES_QueryConfigAttributes(
VAProfile __maybe_unused profile,
VAEntrypoint __maybe_unused entrypoint,
VAConfigAttrib *attrib_list,
int num_attribs)
{
int i;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264ES_QueryConfigAttributes\n");
/* RateControl attributes */
for (i = 0; i < num_attribs; i++) {
switch (attrib_list[i].type) {
case VAConfigAttribRTFormat:
break;
case VAConfigAttribRateControl:
attrib_list[i].value = VA_RC_NONE | VA_RC_CBR | VA_RC_VBR | VA_RC_VCM;
break;
case VAConfigAttribEncMaxRefFrames:
attrib_list[i].value = 1;
break;
default:
attrib_list[i].value = VA_ATTRIB_NOT_SUPPORTED;
break;
}
}
}
static VAStatus pnw_H264ES_ValidateConfig(
object_config_p obj_config)
{
int i;
/* Check all attributes */
for (i = 0; i < obj_config->attrib_count; i++) {
switch (obj_config->attrib_list[i].type) {
case VAConfigAttribRTFormat:
/* Ignore */
break;
case VAConfigAttribRateControl:
break;
default:
return VA_STATUS_ERROR_ATTR_NOT_SUPPORTED;
}
}
return VA_STATUS_SUCCESS;
}
static VAStatus pnw_H264ES_CreateContext(
object_context_p obj_context,
object_config_p obj_config)
{
VAStatus vaStatus = VA_STATUS_SUCCESS;
context_ENC_p ctx;
int i;
unsigned int eRCmode;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264ES_CreateContext\n");
vaStatus = pnw_CreateContext(obj_context, obj_config, 0);
if (VA_STATUS_SUCCESS != vaStatus)
return VA_STATUS_ERROR_ALLOCATION_FAILED;
ctx = (context_ENC_p) obj_context->format_data;
for (i = 0; i < obj_config->attrib_count; i++) {
if (obj_config->attrib_list[i].type == VAConfigAttribRateControl)
break;
}
if (i >= obj_config->attrib_count)
eRCmode = VA_RC_NONE;
else
eRCmode = obj_config->attrib_list[i].value;
if (eRCmode == VA_RC_VBR) {
ctx->eCodec = IMG_CODEC_H264_VBR;
ctx->sRCParams.RCEnable = IMG_TRUE;
ctx->sRCParams.bDisableBitStuffing = IMG_FALSE;
} else if (eRCmode == VA_RC_CBR) {
ctx->eCodec = IMG_CODEC_H264_CBR;
ctx->sRCParams.RCEnable = IMG_TRUE;
ctx->sRCParams.bDisableBitStuffing = IMG_TRUE;
} else if (eRCmode == VA_RC_NONE) {
ctx->eCodec = IMG_CODEC_H264_NO_RC;
ctx->sRCParams.RCEnable = IMG_FALSE;
ctx->sRCParams.bDisableBitStuffing = IMG_FALSE;
} else if (eRCmode == VA_RC_VCM) {
ctx->eCodec = IMG_CODEC_H264_VCM;
ctx->sRCParams.RCEnable = IMG_TRUE;
ctx->sRCParams.bDisableBitStuffing = IMG_FALSE;
} else
return VA_STATUS_ERROR_UNSUPPORTED_RT_FORMAT;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "eCodec is %d\n", ctx->eCodec);
ctx->eFormat = IMG_CODEC_PL12; /* use default */
ctx->Slices = 1;
ctx->idr_pic_id = 1;
ctx->buffer_size = 0;
ctx->initial_buffer_fullness = 0;
//initialize the frame_rate and qp
ctx->sRCParams.FrameRate = 30;
if (getenv("PSB_VIDEO_SIG_CORE") == NULL) {
ctx->Slices = 2;
ctx->NumCores = 2;
}
ctx->ParallelCores = min(ctx->NumCores, ctx->Slices);
ctx->IPEControl = pnw__get_ipe_control(ctx->eCodec);
switch (obj_config->profile) {
case VAProfileH264Baseline:
ctx->profile_idc = 5;
break;
case VAProfileH264Main:
ctx->profile_idc = 6;
break;
default:
ctx->profile_idc = 6;
break;
}
return vaStatus;
}
static void pnw_H264ES_DestroyContext(
object_context_p obj_context)
{
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264ES_DestroyPicture\n");
pnw_DestroyContext(obj_context);
}
static VAStatus pnw_H264ES_BeginPicture(
object_context_p obj_context)
{
INIT_CONTEXT_H264ES;
VAStatus vaStatus = VA_STATUS_SUCCESS;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264ES_BeginPicture\n");
vaStatus = pnw_BeginPicture(ctx);
return vaStatus;
}
static VAStatus pnw__H264ES_process_sequence_param(context_ENC_p ctx, object_buffer_p obj_buffer)
{
VAEncSequenceParameterBufferH264 *pSequenceParams;
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
H264_VUI_PARAMS *pVUI_Params = &(ctx->VUI_Params);
H264_CROP_PARAMS sCrop;
int i;
unsigned int frame_size;
unsigned int max_bps;
ASSERT(obj_buffer->type == VAEncSequenceParameterBufferType);
ASSERT(obj_buffer->num_elements == 1);
ASSERT(obj_buffer->size == sizeof(VAEncSequenceParameterBufferH264));
if ((obj_buffer->num_elements != 1) ||
(obj_buffer->size != sizeof(VAEncSequenceParameterBufferH264))) {
return VA_STATUS_ERROR_UNKNOWN;
}
if(ctx->sRCParams.FrameRate == 0)
ctx->sRCParams.FrameRate = 30;
ctx->obj_context->frame_count = 0;
pSequenceParams = (VAEncSequenceParameterBufferH264 *) obj_buffer->buffer_data;
obj_buffer->buffer_data = NULL;
obj_buffer->size = 0;
if (!pSequenceParams->bits_per_second) {
pSequenceParams->bits_per_second = ctx->Height * ctx->Width * 30 * 12;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "bits_per_second is 0, set to %d\n",
pSequenceParams->bits_per_second);
}
ctx->sRCParams.bBitrateChanged =
(pSequenceParams->bits_per_second == ctx->sRCParams.BitsPerSecond ?
IMG_FALSE : IMG_TRUE);
if (pSequenceParams->bits_per_second > TOPAZ_H264_MAX_BITRATE) {
ctx->sRCParams.BitsPerSecond = TOPAZ_H264_MAX_BITRATE;
drv_debug_msg(VIDEO_DEBUG_GENERAL, " bits_per_second(%d) exceeds \
the maximum bitrate, set it with %d\n",
pSequenceParams->bits_per_second,
TOPAZ_H264_MAX_BITRATE);
}
/* According to Table A-1 Level limits, if resolution is bigger than 625SD,
min compression ratio is 4, otherwise min compression ratio is 2 */
max_bps = (ctx->Width * ctx->Height * 3 / 2 ) * 8 * ctx->sRCParams.FrameRate;
if (ctx->Width > 720)
max_bps /= 4;
else
max_bps /= 2;
drv_debug_msg(VIDEO_DEBUG_GENERAL, " width %d height %d, frame rate %d\n",
ctx->Width, ctx->Height, ctx->sRCParams.FrameRate);
if (pSequenceParams->bits_per_second > max_bps) {
drv_debug_msg(VIDEO_DEBUG_ERROR,
"Invalid bitrate %d, violate ITU-T Rec. H.264 (03/2005) A.3.1"
"\n clip to %d bps\n", pSequenceParams->bits_per_second, max_bps);
ctx->sRCParams.BitsPerSecond = max_bps;
} else {
/* See 110% target bitrate for VCM. Otherwise, the resulted bitrate is much lower
than target bitrate */
if (ctx->eCodec == IMG_CODEC_H264_VCM)
pSequenceParams->bits_per_second =
pSequenceParams->bits_per_second / 100 * 110;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Bitrate is set to %d\n",
pSequenceParams->bits_per_second);
ctx->sRCParams.BitsPerSecond = pSequenceParams->bits_per_second;
}
/*if (ctx->sRCParams.IntraFreq != pSequenceParams->intra_period)
ctx->sRCParams.bBitrateChanged = IMG_TRUE;*/
ctx->sRCParams.IDRFreq = pSequenceParams->intra_idr_period;
ctx->sRCParams.Slices = ctx->Slices;
ctx->sRCParams.QCPOffset = 0;
if (ctx->sRCParams.IntraFreq != pSequenceParams->intra_period
&& ctx->raw_frame_count != 0
&& ctx->sRCParams.IntraFreq != 0
&& ((ctx->obj_context->frame_count + 1) % ctx->sRCParams.IntraFreq) != 0
&& (!ctx->sRCParams.bDisableFrameSkipping)) {
drv_debug_msg(VIDEO_DEBUG_ERROR,
"Changing intra period value in the middle of a GOP is\n"
"not allowed if frame skip isn't disabled.\n"
"it can cause I frame been skipped\n");
free(pSequenceParams);
return VA_STATUS_ERROR_INVALID_PARAMETER;
}
else
ctx->sRCParams.IntraFreq = pSequenceParams->intra_period;
frame_size = ctx->sRCParams.BitsPerSecond / ctx->sRCParams.FrameRate;
if (ctx->bInserHRDParams &&
ctx->buffer_size != 0 && ctx->initial_buffer_fullness != 0) {
ctx->sRCParams.BufferSize = ctx->buffer_size;
ctx->sRCParams.InitialLevel = ctx->buffer_size - ctx->initial_buffer_fullness;
ctx->sRCParams.InitialDelay = ctx->initial_buffer_fullness;
}
else {
ctx->buffer_size = ctx->sRCParams.BitsPerSecond;
ctx->initial_buffer_fullness = ctx->sRCParams.BitsPerSecond;
ctx->sRCParams.BufferSize = ctx->buffer_size;
ctx->sRCParams.InitialLevel = (3 * ctx->sRCParams.BufferSize) >> 4;
/* Aligned with target frame size */
ctx->sRCParams.InitialLevel += (frame_size / 2);
ctx->sRCParams.InitialLevel /= frame_size;
ctx->sRCParams.InitialLevel *= frame_size;
ctx->sRCParams.InitialDelay = ctx->buffer_size - ctx->sRCParams.InitialLevel;
}
if (ctx->raw_frame_count == 0) {
for (i = (ctx->ParallelCores - 1); i >= 0; i--)
pnw_set_bias(ctx, i);
}
pVUI_Params->bit_rate_value_minus1 = ctx->sRCParams.BitsPerSecond / 64 - 1;
pVUI_Params->cbp_size_value_minus1 = ctx->sRCParams.BufferSize / 64 - 1;
if (IMG_CODEC_H264_CBR != ctx->eCodec ||
ctx->sRCParams.bDisableBitStuffing ||
ctx->sRCParams.bDisableFrameSkipping)
pVUI_Params->CBR = 0;
else
pVUI_Params->CBR = 1;
pVUI_Params->initial_cpb_removal_delay_length_minus1 = BPH_SEI_NAL_INITIAL_CPB_REMOVAL_DELAY_SIZE - 1;
pVUI_Params->cpb_removal_delay_length_minus1 = PTH_SEI_NAL_CPB_REMOVAL_DELAY_SIZE - 1;
pVUI_Params->dpb_output_delay_length_minus1 = PTH_SEI_NAL_DPB_OUTPUT_DELAY_SIZE - 1;
pVUI_Params->time_offset_length = 24;
ctx->bInsertVUI = pSequenceParams->vui_parameters_present_flag ? IMG_TRUE: IMG_FALSE;
if (ctx->bInsertVUI) {
if (pSequenceParams->num_units_in_tick !=0 && pSequenceParams->time_scale !=0
&& (pSequenceParams->time_scale > pSequenceParams->num_units_in_tick) ) {
pVUI_Params->Time_Scale = pSequenceParams->time_scale;
pVUI_Params->num_units_in_tick = pSequenceParams->num_units_in_tick;
}
else {
pVUI_Params->num_units_in_tick = 1;
pVUI_Params->Time_Scale = ctx->sRCParams.FrameRate * 2;
}
}
if (ctx->bInsertVUI && pSequenceParams->vui_fields.bits.aspect_ratio_info_present_flag &&
(pSequenceParams->aspect_ratio_idc == 0xff /* Extended_SAR */)) {
pVUI_Params->aspect_ratio_info_present_flag = IMG_TRUE;
pVUI_Params->aspect_ratio_idc = 0xff;
pVUI_Params->sar_width = pSequenceParams->sar_width;
pVUI_Params->sar_height = pSequenceParams->sar_height;
}
sCrop.bClip = pSequenceParams->frame_cropping_flag;
sCrop.LeftCropOffset = 0;
sCrop.RightCropOffset = 0;
sCrop.TopCropOffset = 0;
sCrop.BottomCropOffset = 0;
if (!sCrop.bClip) {
if (ctx->RawHeight & 0xf) {
sCrop.bClip = IMG_TRUE;
sCrop.BottomCropOffset = (((ctx->RawHeight + 0xf) & (~0xf)) - ctx->RawHeight) / 2;
}
if (ctx->RawWidth & 0xf) {
sCrop.bClip = IMG_TRUE;
sCrop.RightCropOffset = (((ctx->RawWidth + 0xf) & (~0xf)) - ctx->RawWidth) / 2;
}
} else {
sCrop.LeftCropOffset = pSequenceParams->frame_crop_left_offset;
sCrop.RightCropOffset = pSequenceParams->frame_crop_right_offset;
sCrop.TopCropOffset = pSequenceParams->frame_crop_top_offset;
sCrop.BottomCropOffset = pSequenceParams->frame_crop_bottom_offset;
}
/* sequence header is always inserted */
memset(cmdbuf->header_mem_p + ctx->seq_header_ofs,
0,
HEADER_SIZE);
/*
if (ctx->bInserHRDParams) {
memset(cmdbuf->header_mem_p + ctx->aud_header_ofs,
0,
HEADER_SIZE);
pnw__H264_prepare_AUD_header(cmdbuf->header_mem_p + ctx->aud_header_ofs);
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->ParallelCores - 1,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->aud_header_ofs);
}
*/
if (ctx->eCodec == IMG_CODEC_H264_NO_RC)
pnw__H264_prepare_sequence_header(cmdbuf->header_mem_p + ctx->seq_header_ofs,
pSequenceParams->picture_width_in_mbs,
pSequenceParams->picture_height_in_mbs,
pSequenceParams->vui_parameters_present_flag,
pSequenceParams->vui_parameters_present_flag ? (pVUI_Params) : NULL,
&sCrop,
pSequenceParams->level_idc, ctx->profile_idc);
else
pnw__H264_prepare_sequence_header(cmdbuf->header_mem_p + ctx->seq_header_ofs,
pSequenceParams->picture_width_in_mbs,
pSequenceParams->picture_height_in_mbs,
pSequenceParams->vui_parameters_present_flag,
pSequenceParams->vui_parameters_present_flag ? (pVUI_Params) : NULL,
&sCrop,
pSequenceParams->level_idc, ctx->profile_idc);
/*Periodic IDR need SPS. We save the sequence header here*/
if (ctx->sRCParams.IDRFreq != 0) {
if (NULL == ctx->save_seq_header_p) {
ctx->save_seq_header_p = malloc(HEADER_SIZE);
if (NULL == ctx->save_seq_header_p) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "Ran out of memory!\n");
free(pSequenceParams);
return VA_STATUS_ERROR_ALLOCATION_FAILED;
}
memcpy((unsigned char *)ctx->save_seq_header_p,
(unsigned char *)(cmdbuf->header_mem_p + ctx->seq_header_ofs),
HEADER_SIZE);
}
}
ctx->none_vcl_nal++;
cmdbuf->cmd_idx_saved[PNW_CMDBUF_SEQ_HEADER_IDX] = cmdbuf->cmd_idx;
/* Send to the last core as this will complete first */
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->ParallelCores - 1,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->seq_header_ofs);
free(pSequenceParams);
return VA_STATUS_SUCCESS;
}
static VAStatus pnw__H264ES_insert_SEI_buffer_period(context_ENC_p ctx)
{
unsigned int ui32nal_initial_cpb_removal_delay;
unsigned int ui32nal_initial_cpb_removal_delay_offset;
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
ui32nal_initial_cpb_removal_delay =
90000 * (1.0 * ctx->sRCParams.InitialDelay / ctx->sRCParams.BitsPerSecond);
ui32nal_initial_cpb_removal_delay_offset =
90000 * (1.0 * ctx->buffer_size / ctx->sRCParams.BitsPerSecond)
- ui32nal_initial_cpb_removal_delay;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Insert SEI buffer period message with "
"ui32nal_initial_cpb_removal_delay(%d) and "
"ui32nal_initial_cpb_removal_delay_offset(%d)\n",
ui32nal_initial_cpb_removal_delay,
ui32nal_initial_cpb_removal_delay_offset);
memset(cmdbuf->header_mem_p + ctx->sei_buf_prd_ofs,
0,
HEADER_SIZE);
pnw__H264_prepare_SEI_buffering_period_header(
(MTX_HEADER_PARAMS *)(cmdbuf->header_mem_p + ctx->sei_buf_prd_ofs),
1, //ui8NalHrdBpPresentFlag,
0, //ui8nal_cpb_cnt_minus1,
1 + ctx->VUI_Params.initial_cpb_removal_delay_length_minus1, //ui8nal_initial_cpb_removal_delay_length,
ui32nal_initial_cpb_removal_delay, //ui32nal_initial_cpb_removal_delay,
ui32nal_initial_cpb_removal_delay_offset, //ui32nal_initial_cpb_removal_delay_offset,
0, //ui8VclHrdBpPresentFlag,
NOT_USED_BY_TOPAZ, //ui8vcl_cpb_cnt_minus1,
0, //ui32vcl_initial_cpb_removal_delay,
0 //ui32vcl_initial_cpb_removal_delay_offset
);
cmdbuf->cmd_idx_saved[PNW_CMDBUF_SEI_BUF_PERIOD_IDX] = cmdbuf->cmd_idx;
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->ParallelCores - 1,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->sei_buf_prd_ofs);
ctx->none_vcl_nal++;
return VA_STATUS_SUCCESS;
}
static VAStatus pnw__H264ES_insert_SEI_pic_timing(context_ENC_p ctx)
{
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
uint32_t ui32cpb_removal_delay;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Insert SEI picture timing message. \n");
memset(cmdbuf->header_mem_p + ctx->sei_pic_tm_ofs,
0,
HEADER_SIZE);
/* ui32cpb_removal_delay is zero for 1st frame and will be reset
* after a IDR frame */
if (ctx->obj_context->frame_count == 0) {
if (ctx->raw_frame_count == 0)
ui32cpb_removal_delay = 0;
else
ui32cpb_removal_delay =
ctx->sRCParams.IDRFreq * ctx->sRCParams.IntraFreq * 2;
} else
ui32cpb_removal_delay = 2 * ctx->obj_context->frame_count;
pnw__H264_prepare_SEI_picture_timing_header(
(MTX_HEADER_PARAMS *)(cmdbuf->header_mem_p + ctx->sei_pic_tm_ofs),
1,
ctx->VUI_Params.cpb_removal_delay_length_minus1,
ctx->VUI_Params.dpb_output_delay_length_minus1,
ui32cpb_removal_delay, //ui32cpb_removal_delay,
2, //ui32dpb_output_delay,
0, //ui8pic_struct_present_flag,
0, //ui8pic_struct,
0, //ui8NumClockTS,
0, //*aui8clock_timestamp_flag,
0, //ui8full_timestamp_flag,
0, //ui8seconds_flag,
0, //ui8minutes_flag,
0, //ui8hours_flag,
0, //ui8seconds_value,
0, //ui8minutes_value,
0, //ui8hours_value,
0, //ui8ct_type,
0, //ui8nuit_field_based_flag,
0, //ui8counting_type,
0, //ui8discontinuity_flag,
0, //ui8cnt_dropped_flag,
0, //ui8n_frames,
0, //ui8time_offset_length,
0 //i32time_offset)
);
cmdbuf->cmd_idx_saved[PNW_CMDBUF_SEI_PIC_TIMING_IDX] = cmdbuf->cmd_idx;
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->ParallelCores - 1,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->sei_pic_tm_ofs);
ctx->none_vcl_nal++;
return VA_STATUS_SUCCESS;
}
#if PSB_MFLD_DUMMY_CODE
static VAStatus pnw__H264ES_insert_SEI_FPA_param(context_ENC_p ctx, object_buffer_p obj_buffer)
{
VAStatus vaStatus = VA_STATUS_SUCCESS;
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
VAEncPackedHeaderParameterBuffer *sei_param_buf = (VAEncPackedHeaderParameterBuffer *)obj_buffer->buffer_data;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Insert SEI frame packing arrangement message. \n");
ctx->sei_pic_data_size = sei_param_buf->bit_length/8;
return VA_STATUS_SUCCESS;
}
static VAStatus pnw__H264ES_insert_SEI_FPA_data(context_ENC_p ctx, object_buffer_p obj_buffer)
{
VAStatus vaStatus = VA_STATUS_SUCCESS;
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
char *sei_data_buf;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Insert SEI frame packing arrangement message. \n");
memset(cmdbuf->header_mem_p + ctx->sei_pic_fpa_ofs,
0,
HEADER_SIZE);
sei_data_buf = (char *)obj_buffer->buffer_data;
pnw__H264_prepare_SEI_FPA_header((MTX_HEADER_PARAMS *)(cmdbuf->header_mem_p + ctx->sei_pic_fpa_ofs), sei_data_buf, ctx->sei_pic_data_size);
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->ParallelCores - 1,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->sei_pic_fpa_ofs);
return VA_STATUS_SUCCESS;
}
#endif
static VAStatus pnw__H264ES_process_picture_param(context_ENC_p ctx, object_buffer_p obj_buffer)
{
VAStatus vaStatus = VA_STATUS_SUCCESS;
int i;
VAEncPictureParameterBufferH264 *pBuffer;
int need_sps = 0;
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
ASSERT(obj_buffer->type == VAEncPictureParameterBufferType);
if ((obj_buffer->num_elements != 1) ||
(obj_buffer->size != sizeof(VAEncPictureParameterBufferH264))) {
return VA_STATUS_ERROR_UNKNOWN;
}
/* Transfer ownership of VAEncPictureParameterBufferH264 data */
pBuffer = (VAEncPictureParameterBufferH264 *) obj_buffer->buffer_data;
obj_buffer->buffer_data = NULL;
obj_buffer->size = 0;
ctx->ref_surface = SURFACE(pBuffer->ReferenceFrames[0].picture_id);
ctx->dest_surface = SURFACE(pBuffer->CurrPic.picture_id);
ctx->coded_buf = BUFFER(pBuffer->coded_buf);
//ASSERT(ctx->Width == pBuffer->picture_width);
//ASSERT(ctx->Height == pBuffer->picture_height);
if (NULL == ctx->coded_buf) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "%s L%d Invalid coded buffer handle\n", __FUNCTION__, __LINE__);
free(pBuffer);
return VA_STATUS_ERROR_INVALID_BUFFER;
}
if ((ctx->sRCParams.IntraFreq != 0) && (ctx->sRCParams.IDRFreq != 0)) { /* period IDR is desired */
unsigned int is_intra = 0;
unsigned int intra_cnt = 0;
ctx->force_idr_h264 = 0;
if ((ctx->obj_context->frame_count % ctx->sRCParams.IntraFreq) == 0) {
is_intra = 1; /* suppose current frame is I frame */
intra_cnt = ctx->obj_context->frame_count / ctx->sRCParams.IntraFreq;
}
/* current frame is I frame (suppose), and an IDR frame is desired*/
if ((is_intra) && ((intra_cnt % ctx->sRCParams.IDRFreq) == 0)) {
ctx->force_idr_h264 = 1;
/*When two consecutive access units in decoding order are both IDR access
* units, the value of idr_pic_id in the slices of the first such IDR
* access unit shall differ from the idr_pic_id in the second such IDR
* access unit. We set it with 1 or 0 alternately.*/
ctx->idr_pic_id = 1 - ctx->idr_pic_id;
/* it is periodic IDR in the middle of one sequence encoding, need SPS */
if (ctx->obj_context->frame_count > 0)
need_sps = 1;
ctx->obj_context->frame_count = 0;
}
}
/* If VUI header isn't enabled, we'll igore the request for HRD header insertion */
if (ctx->bInserHRDParams)
ctx->bInserHRDParams = ctx->bInsertVUI;
/* For H264, PicHeader only needed in the first picture*/
if (!(ctx->obj_context->frame_count)) {
cmdbuf = ctx->obj_context->pnw_cmdbuf;
if (need_sps) {
drv_debug_msg(VIDEO_DEBUG_GENERAL, "TOPAZ: insert a SPS before IDR frame\n");
/* reuse the previous SPS */
memcpy((unsigned char *)(cmdbuf->header_mem_p + ctx->seq_header_ofs),
(unsigned char *)ctx->save_seq_header_p,
HEADER_SIZE);
cmdbuf->cmd_idx_saved[PNW_CMDBUF_SEQ_HEADER_IDX] = cmdbuf->cmd_idx;
/* Send to the last core as this will complete first */
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->ParallelCores - 1,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->seq_header_ofs);
ctx->none_vcl_nal++;
}
if (ctx->bInserHRDParams) {
pnw__H264ES_insert_SEI_buffer_period(ctx);
pnw__H264ES_insert_SEI_pic_timing(ctx);
}
pnw__H264_prepare_picture_header(cmdbuf->header_mem_p + ctx->pic_header_ofs, IMG_FALSE, ctx->sRCParams.QCPOffset);
cmdbuf->cmd_idx_saved[PNW_CMDBUF_PIC_HEADER_IDX] = cmdbuf->cmd_idx;
/* Send to the last core as this will complete first */
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->ParallelCores - 1,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->pic_header_ofs);
ctx->none_vcl_nal++;
}
else if (ctx->bInserHRDParams)
pnw__H264ES_insert_SEI_pic_timing(ctx);
if (ctx->ParallelCores == 1) {
ctx->coded_buf_per_slice = 0;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "TOPAZ: won't splite coded buffer(%d) since only one slice being encoded\n",
ctx->coded_buf->size);
} else {
/*Make sure DMA start is 128bits alignment*/
ctx->coded_buf_per_slice = (ctx->coded_buf->size / ctx->ParallelCores) & (~0xf) ;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "TOPAZ: the size of coded_buf per slice %d( Total %d) \n", ctx->coded_buf_per_slice,
ctx->coded_buf->size);
}
/* Prepare START_PICTURE params */
/* FIXME is really need multiple picParams? Need multiple calculate for each? */
for (i = (ctx->ParallelCores - 1); i >= 0; i--)
vaStatus = pnw_RenderPictureParameter(ctx, i);
free(pBuffer);
return vaStatus;
}
static VAStatus pnw__H264ES_encode_one_slice(context_ENC_p ctx,
VAEncSliceParameterBuffer *pBuffer)
{
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
unsigned int MBSkipRun, FirstMBAddress;
unsigned char deblock_idc;
unsigned char is_intra = 0;
int slice_param_idx;
PIC_PARAMS *psPicParams = (PIC_PARAMS *)(cmdbuf->pic_params_p);
VAStatus vaStatus = VA_STATUS_SUCCESS;
/*Slice encoding Order:
*1.Insert Do header command
*2.setup InRowParams
*3.setup Slice params
*4.Insert Do slice command
* */
if (pBuffer->slice_height > (ctx->Height / 16) ||
pBuffer->start_row_number > (ctx->Height / 16) ||
(pBuffer->slice_height + pBuffer->start_row_number) > (ctx->Height / 16)) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "slice height %d or start row number %d is too large",
pBuffer->slice_height, pBuffer->start_row_number);
return VA_STATUS_ERROR_INVALID_PARAMETER;
}
MBSkipRun = (pBuffer->slice_height * ctx->Width) / 16;
deblock_idc = pBuffer->slice_flags.bits.disable_deblocking_filter_idc;
/*If the frame is skipped, it shouldn't be a I frame*/
if (ctx->force_idr_h264 || (ctx->obj_context->frame_count == 0)) {
is_intra = 1;
} else
is_intra = (ctx->sRCParams.RCEnable && ctx->sRCParams.FrameSkip) ? 0 : pBuffer->slice_flags.bits.is_intra;
FirstMBAddress = (pBuffer->start_row_number * ctx->Width) / 16;
memset(cmdbuf->header_mem_p + ctx->slice_header_ofs
+ ctx->obj_context->slice_count * HEADER_SIZE,
0,
HEADER_SIZE);
/* Insert Do Header command, relocation is needed */
pnw__H264_prepare_slice_header(cmdbuf->header_mem_p + ctx->slice_header_ofs
+ ctx->obj_context->slice_count * HEADER_SIZE,
is_intra,
pBuffer->slice_flags.bits.disable_deblocking_filter_idc,
ctx->obj_context->frame_count,
FirstMBAddress,
MBSkipRun,
0,
ctx->force_idr_h264,
IMG_FALSE,
IMG_FALSE,
ctx->idr_pic_id);
/* ensure that this slice is consequtive to that last processed by the target core */
/*
ASSERT( -1 == ctx->LastSliceNum[ctx->SliceToCore]
|| ctx->obj_context->slice_count == 1 + ctx->LastSliceNum[ctx->SliceToCore] );
*/
/* note the slice number the target core is now processing */
ctx->LastSliceNum[ctx->SliceToCore] = ctx->obj_context->slice_count;
pnw_cmdbuf_insert_command_package(ctx->obj_context,
ctx->SliceToCore,
MTX_CMDID_DO_HEADER,
&cmdbuf->header_mem,
ctx->slice_header_ofs + ctx->obj_context->slice_count * HEADER_SIZE);
if (!(ctx->sRCParams.RCEnable && ctx->sRCParams.FrameSkip)) {
/*Only reset on the first frame. It's more effective than DDK. Have confirmed with IMG*/
if (ctx->obj_context->frame_count == 0)
pnw_reset_encoder_params(ctx);
if ((pBuffer->start_row_number == 0) && pBuffer->slice_flags.bits.is_intra) {
ctx->BelowParamsBufIdx = (ctx->BelowParamsBufIdx + 1) & 0x1;
}
slice_param_idx = (pBuffer->slice_flags.bits.is_intra ? 0 : 1) * ctx->slice_param_num
+ ctx->obj_context->slice_count;
if (VAEncSliceParameter_Equal(&ctx->slice_param_cache[slice_param_idx], pBuffer) == 0) {
/* cache current param parameters */
memcpy(&ctx->slice_param_cache[slice_param_idx],
pBuffer, sizeof(VAEncSliceParameterBuffer));
/* Setup InParams value*/
pnw_setup_slice_params(ctx,
pBuffer->start_row_number * 16,
pBuffer->slice_height * 16,
pBuffer->slice_flags.bits.is_intra,
ctx->obj_context->frame_count > 0,
psPicParams->sInParams.SeInitQP);
}
/* Insert do slice command and setup related buffer value */
pnw__send_encode_slice_params(ctx,
pBuffer->slice_flags.bits.is_intra,
pBuffer->start_row_number * 16,
deblock_idc,
ctx->obj_context->frame_count,
pBuffer->slice_height * 16,
ctx->obj_context->slice_count);
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Now frame_count/slice_count is %d/%d\n",
ctx->obj_context->frame_count, ctx->obj_context->slice_count);
}
ctx->obj_context->slice_count++;
return vaStatus;
}
/* convert from VAEncSliceParameterBufferH264 to VAEncSliceParameterBuffer */
static VAStatus pnw__convert_sliceparameter_buffer(VAEncSliceParameterBufferH264 *pBufferH264,
VAEncSliceParameterBuffer *pBuffer,
int picture_width_in_mbs,
unsigned int num_elemenent)
{
unsigned int i;
for (i = 0; i < num_elemenent; i++) {
pBuffer->start_row_number = pBufferH264->macroblock_address / picture_width_in_mbs;
pBuffer->slice_height = pBufferH264->num_macroblocks / picture_width_in_mbs;
pBuffer->slice_flags.bits.is_intra =
(((pBufferH264->slice_type == 2) || (pBufferH264->slice_type == 7)) ? 1 : 0);
pBuffer->slice_flags.bits.disable_deblocking_filter_idc = pBufferH264->disable_deblocking_filter_idc;
/* next conversion */
pBuffer++;
pBufferH264++;
}
return 0;
}
static VAStatus pnw__H264ES_process_slice_param(context_ENC_p ctx, object_buffer_p obj_buffer)
{
/* Prepare InParams for macros of current slice, insert slice header, insert do slice command */
VAEncSliceParameterBuffer *pBuf_per_core, *pBuffer;
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
PIC_PARAMS *psPicParams = (PIC_PARAMS *)(cmdbuf->pic_params_p);
unsigned int i, j, slice_per_core;
VAStatus vaStatus = VA_STATUS_SUCCESS;
ASSERT(obj_buffer->type == VAEncSliceParameterBufferType);
if (obj_buffer->num_elements > (ctx->Height / 16)) {
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
goto out2;
}
cmdbuf = ctx->obj_context->pnw_cmdbuf;
psPicParams = (PIC_PARAMS *)cmdbuf->pic_params_p;
/* Transfer ownership of VAEncPictureParameterBuffer data */
if (obj_buffer->size == sizeof(VAEncSliceParameterBufferH264)) {
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Receive VAEncSliceParameterBufferH264 buffer");
pBuffer = calloc(obj_buffer->num_elements, sizeof(VAEncSliceParameterBuffer));
if (pBuffer == NULL) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "Run out of memory!\n");
vaStatus = VA_STATUS_ERROR_ALLOCATION_FAILED;
goto out2;
}
pnw__convert_sliceparameter_buffer((VAEncSliceParameterBufferH264 *)obj_buffer->buffer_data,
pBuffer,
ctx->Width / 16,
obj_buffer->num_elements);
} else if (obj_buffer->size == sizeof(VAEncSliceParameterBuffer)) {
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Receive VAEncSliceParameterBuffer buffer");
pBuffer = (VAEncSliceParameterBuffer *) obj_buffer->buffer_data;
} else {
drv_debug_msg(VIDEO_DEBUG_ERROR, "Buffer size(%d) is wrong. It should be %d or %d\n",
obj_buffer->size, sizeof(VAEncSliceParameterBuffer),
sizeof(VAEncSliceParameterBufferH264));
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
goto out2;
}
obj_buffer->size = 0;
/*In case the slice number changes*/
if ((ctx->slice_param_cache != NULL) && (obj_buffer->num_elements != ctx->slice_param_num)) {
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Slice number changes. Previous value is %d. Now it's %d\n",
ctx->slice_param_num, obj_buffer->num_elements);
free(ctx->slice_param_cache);
ctx->slice_param_cache = NULL;
ctx->slice_param_num = 0;
}
if (NULL == ctx->slice_param_cache) {
ctx->slice_param_num = obj_buffer->num_elements;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Allocate %d VAEncSliceParameterBuffer cache buffers\n", 2 * ctx->slice_param_num);
ctx->slice_param_cache = calloc(2 * ctx->slice_param_num, sizeof(VAEncSliceParameterBuffer));
if (NULL == ctx->slice_param_cache) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "Run out of memory!\n");
/* free the converted VAEncSliceParameterBuffer */
if (obj_buffer->size == sizeof(VAEncSliceParameterBufferH264))
free(pBuffer);
free(obj_buffer->buffer_data);
return VA_STATUS_ERROR_ALLOCATION_FAILED;
}
}
ctx->sRCParams.Slices = obj_buffer->num_elements;
if (getenv("PSB_VIDEO_SIG_CORE") == NULL) {
if ((ctx->ParallelCores == 2) && (obj_buffer->num_elements == 1)) {
/*Need to replace unneccesary MTX_CMDID_STARTPICs with MTX_CMDID_PAD*/
for (i = 0; i < (ctx->ParallelCores - 1); i++) {
*(cmdbuf->cmd_idx_saved[PNW_CMDBUF_START_PIC_IDX] + i * 4) &= (~MTX_CMDWORD_ID_MASK);
*(cmdbuf->cmd_idx_saved[PNW_CMDBUF_START_PIC_IDX] + i * 4) |= MTX_CMDID_PAD;
}
drv_debug_msg(VIDEO_DEBUG_GENERAL, " Remove unneccesary %d MTX_CMDID_STARTPIC commands from cmdbuf\n",
ctx->ParallelCores - obj_buffer->num_elements);
ctx->ParallelCores = obj_buffer->num_elements;
/* All header generation commands should be send to core 0*/
for (i = PNW_CMDBUF_SEQ_HEADER_IDX; i < PNW_CMDBUF_SAVING_MAX; i++) {
if (cmdbuf->cmd_idx_saved[i] != 0)
*(cmdbuf->cmd_idx_saved[i]) &=
~(MTX_CMDWORD_CORE_MASK << MTX_CMDWORD_CORE_SHIFT);
}
ctx->SliceToCore = ctx->ParallelCores - 1;
}
}
slice_per_core = obj_buffer->num_elements / ctx->ParallelCores;
pBuf_per_core = pBuffer;
for (i = 0; i < slice_per_core; i++) {
pBuffer = pBuf_per_core;
for (j = 0; j < ctx->ParallelCores; j++) {
vaStatus = pnw__H264ES_encode_one_slice(ctx, pBuffer);
if (vaStatus != VA_STATUS_SUCCESS)
goto out1;
if (0 == ctx->SliceToCore) {
ctx->SliceToCore = ctx->ParallelCores;
}
ctx->SliceToCore--;
ASSERT(ctx->obj_context->slice_count < MAX_SLICES_PER_PICTURE);
/*Move to the next buffer which will be sent to core j*/
pBuffer += slice_per_core;
}
pBuf_per_core++; /* Move to the next buffer */
}
/*Cope with last slice when slice number is odd and parallelCores is even*/
if (obj_buffer->num_elements > (slice_per_core * ctx->ParallelCores)) {
ctx->SliceToCore = 0;
pBuffer -= slice_per_core;
pBuffer ++;
vaStatus = pnw__H264ES_encode_one_slice(ctx, pBuffer);
}
out1:
/* free the converted VAEncSliceParameterBuffer */
if (obj_buffer->size == sizeof(VAEncSliceParameterBufferH264))
free(pBuffer);
out2:
free(obj_buffer->buffer_data);
obj_buffer->buffer_data = NULL;
return vaStatus;
}
static VAStatus pnw__H264ES_process_misc_param(context_ENC_p ctx, object_buffer_p obj_buffer)
{
/* Prepare InParams for macros of current slice, insert slice header, insert do slice command */
VAEncMiscParameterBuffer *pBuffer;
VAEncMiscParameterRateControl *rate_control_param;
VAEncMiscParameterAIR *air_param;
VAEncMiscParameterMaxSliceSize *max_slice_size_param;
VAEncMiscParameterFrameRate *frame_rate_param;
VAEncMiscParameterHRD *hrd_param;
VAStatus vaStatus = VA_STATUS_SUCCESS;
unsigned int max_bps;
unsigned int frame_size;
ASSERT(obj_buffer->type == VAEncMiscParameterBufferType);
/* Transfer ownership of VAEncMiscParameterBuffer data */
pBuffer = (VAEncMiscParameterBuffer *) obj_buffer->buffer_data;
obj_buffer->size = 0;
if (ctx->eCodec != IMG_CODEC_H264_VCM
&& (pBuffer->type != VAEncMiscParameterTypeHRD
&& pBuffer->type != VAEncMiscParameterTypeRateControl
&& pBuffer->type != VAEncMiscParameterTypeFrameRate)) {
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Buffer type %d isn't supported in none VCM mode.\n",
pBuffer->type);
free(obj_buffer->buffer_data);
obj_buffer->buffer_data = NULL;
return VA_STATUS_SUCCESS;
}
switch (pBuffer->type) {
case VAEncMiscParameterTypeFrameRate:
frame_rate_param = (VAEncMiscParameterFrameRate *)pBuffer->data;
if (frame_rate_param->framerate < 1 || frame_rate_param->framerate > 65535) {
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
break;
}
if (ctx->sRCParams.FrameRate == frame_rate_param->framerate)
break;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "frame rate changed from %d to %d\n",
ctx->sRCParams.FrameRate,
frame_rate_param->framerate);
ctx->sRCParams.FrameRate = frame_rate_param->framerate;
ctx->sRCParams.bBitrateChanged = IMG_TRUE;
ctx->sRCParams.FrameRate = (frame_rate_param->framerate < 1) ? 1 :
((65535 < frame_rate_param->framerate) ? 65535 : frame_rate_param->framerate);
break;
case VAEncMiscParameterTypeRateControl:
rate_control_param = (VAEncMiscParameterRateControl *)pBuffer->data;
/* Currently, none VCM mode only supports frame skip and bit stuffing
* disable flag and doesn't accept other parameters in
* buffer of VAEncMiscParameterTypeRateControl type */
if (rate_control_param->rc_flags.value != 0 || ctx->raw_frame_count == 0) {
if (rate_control_param->rc_flags.bits.disable_frame_skip)
ctx->sRCParams.bDisableFrameSkipping = IMG_TRUE;
if (rate_control_param->rc_flags.bits.disable_bit_stuffing)
ctx->sRCParams.bDisableBitStuffing = IMG_TRUE;
drv_debug_msg(VIDEO_DEBUG_GENERAL,
"bDisableFrameSkipping is %d and bDisableBitStuffing is %d\n",
ctx->sRCParams.bDisableFrameSkipping, ctx->sRCParams.bDisableBitStuffing);
}
if (rate_control_param->initial_qp > 51 ||
rate_control_param->min_qp > 51) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "Initial_qp(%d) and min_qpinitial_qp(%d) "
"are invalid.\nQP shouldn't be larger than 51 for H264\n",
rate_control_param->initial_qp, rate_control_param->min_qp);
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
break;
}
if (rate_control_param->window_size > 2000) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "window_size is too much!\n");
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
break;
}
/* Check if any none-zero RC parameter is changed*/
if ((rate_control_param->bits_per_second == 0 ||
rate_control_param->bits_per_second == ctx->sRCParams.BitsPerSecond) &&
(rate_control_param->window_size == 0 ||
ctx->sRCParams.BufferSize == ctx->sRCParams.BitsPerSecond / 1000 * rate_control_param->window_size) &&
(ctx->sRCParams.MinQP == rate_control_param->min_qp) &&
(ctx->sRCParams.InitialQp == rate_control_param->initial_qp) &&
(rate_control_param->basic_unit_size == 0 ||
ctx->sRCParams.BUSize == rate_control_param->basic_unit_size)) {
drv_debug_msg(VIDEO_DEBUG_GENERAL, "%s No RC parameter is changed\n",
__FUNCTION__);
break;
}
else if (ctx->raw_frame_count != 0 || ctx->eCodec == IMG_CODEC_H264_VCM)
ctx->sRCParams.bBitrateChanged = IMG_TRUE;
/* The initial target bitrate is set by Sequence parameter buffer.
Here is for changed bitrate only */
if (rate_control_param->bits_per_second > TOPAZ_H264_MAX_BITRATE) {
drv_debug_msg(VIDEO_DEBUG_ERROR, " bits_per_second(%d) exceeds \
the maximum bitrate, set it with %d\n",
rate_control_param->bits_per_second,
TOPAZ_H264_MAX_BITRATE);
break;
}
/* The initial target bitrate is set by Sequence parameter buffer.
Here is for changed bitrate only */
if (rate_control_param->bits_per_second != 0 &&
ctx->raw_frame_count != 0) {
drv_debug_msg(VIDEO_DEBUG_GENERAL,
"bitrate is changed from %d to %d on frame %d\n",
ctx->sRCParams.BitsPerSecond,
rate_control_param->bits_per_second,
ctx->raw_frame_count);
max_bps = (ctx->Width * ctx->Height * 3 / 2 ) * 8 * ctx->sRCParams.FrameRate;
if (ctx->Width > 720)
max_bps /= 4;
else
max_bps /= 2;
drv_debug_msg(VIDEO_DEBUG_GENERAL, " width %d height %d, frame rate %d\n",
ctx->Width, ctx->Height, ctx->sRCParams.FrameRate);
if (rate_control_param->bits_per_second > max_bps) {
drv_debug_msg(VIDEO_DEBUG_ERROR,
"Invalid bitrate %d, violate ITU-T Rec. H.264 (03/2005) A.3.1"
"\n clip to %d bps\n", rate_control_param->bits_per_second, max_bps);
ctx->sRCParams.BitsPerSecond = max_bps;
} else {
/* See 110% target bitrate for VCM. Otherwise, the resulted bitrate is much lower
than target bitrate */
if (ctx->eCodec == IMG_CODEC_H264_VCM)
rate_control_param->bits_per_second =
rate_control_param->bits_per_second / 100 * 110;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Bitrate is set to %d\n",
rate_control_param->bits_per_second);
ctx->sRCParams.BitsPerSecond = rate_control_param->bits_per_second;
}
}
if (rate_control_param->min_qp != 0)
ctx->sRCParams.MinQP = rate_control_param->min_qp;
if (rate_control_param->window_size != 0) {
ctx->sRCParams.BufferSize =
ctx->sRCParams.BitsPerSecond / 1000 * rate_control_param->window_size;
if (ctx->sRCParams.FrameRate == 0) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "frame rate can't be zero. Set it to 30");
ctx->sRCParams.FrameRate = 30;
}
frame_size = ctx->sRCParams.BitsPerSecond / ctx->sRCParams.FrameRate;
if (frame_size == 0) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "Bitrate is too low %d\n",
ctx->sRCParams.BitsPerSecond);
break;
}
ctx->sRCParams.InitialLevel = (3 * ctx->sRCParams.BufferSize) >> 4;
ctx->sRCParams.InitialLevel += (frame_size / 2);
ctx->sRCParams.InitialLevel /= frame_size;
ctx->sRCParams.InitialLevel *= frame_size;
ctx->sRCParams.InitialDelay =
ctx->sRCParams.BufferSize - ctx->sRCParams.InitialLevel;
}
if (rate_control_param->initial_qp != 0)
ctx->sRCParams.InitialQp = rate_control_param->initial_qp;
if (rate_control_param->basic_unit_size != 0)
ctx->sRCParams.BUSize = rate_control_param->basic_unit_size;
drv_debug_msg(VIDEO_DEBUG_GENERAL,
"Set Misc parameters(frame %d): window_size %d, initial qp %d\n" \
"\tmin qp %d, bunit size %d\n",
ctx->raw_frame_count,
rate_control_param->window_size,
rate_control_param->initial_qp,
rate_control_param->min_qp,
rate_control_param->basic_unit_size);
break;
case VAEncMiscParameterTypeMaxSliceSize:
max_slice_size_param = (VAEncMiscParameterMaxSliceSize *)pBuffer->data;
/*The max slice size should not be bigger than 1920x1080x1.5x8 */
if (max_slice_size_param->max_slice_size > 24883200) {
drv_debug_msg(VIDEO_DEBUG_ERROR,"Invalid max_slice_size. It should be 1~ 24883200.\n");
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
break;
}
if (ctx->max_slice_size == max_slice_size_param->max_slice_size)
break;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "max slice size changed to %d\n",
max_slice_size_param->max_slice_size);
ctx->max_slice_size = max_slice_size_param->max_slice_size;
break;
case VAEncMiscParameterTypeAIR:
air_param = (VAEncMiscParameterAIR *)pBuffer->data;
if (air_param->air_num_mbs > 65535 ||
air_param->air_threshold > 65535) {
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
break;
}
drv_debug_msg(VIDEO_DEBUG_GENERAL,"air slice size changed to num_air_mbs %d "
"air_threshold %d, air_auto %d\n",
air_param->air_num_mbs, air_param->air_threshold,
air_param->air_auto);
if (((ctx->Height * ctx->Width) >> 8) < (int)air_param->air_num_mbs)
air_param->air_num_mbs = ((ctx->Height * ctx->Width) >> 8);
if (air_param->air_threshold == 0)
drv_debug_msg(VIDEO_DEBUG_GENERAL,"%s: air threshold is set to zero\n",
__func__);
ctx->num_air_mbs = air_param->air_num_mbs;
ctx->air_threshold = air_param->air_threshold;
//ctx->autotune_air_flag = air_param->air_auto;
break;
case VAEncMiscParameterTypeHRD:
hrd_param = (VAEncMiscParameterHRD *)pBuffer->data;
if (hrd_param->buffer_size == 0
|| hrd_param->initial_buffer_fullness == 0)
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Find zero value for buffer_size "
"and initial_buffer_fullness.\n"
"Will assign default value to them later \n");
if (ctx->initial_buffer_fullness > ctx->buffer_size) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "initial_buffer_fullnessi(%d) shouldn't be"
" larger that buffer_size(%d)!\n",
hrd_param->initial_buffer_fullness,
hrd_param->buffer_size);
vaStatus = VA_STATUS_ERROR_INVALID_PARAMETER;
break;
}
if (!ctx->sRCParams.RCEnable) {
drv_debug_msg(VIDEO_DEBUG_ERROR, "Only when rate control is enabled,"
" VAEncMiscParameterTypeHRD will take effect.\n");
break;
}
ctx->buffer_size = hrd_param->buffer_size;
ctx->initial_buffer_fullness = hrd_param->initial_buffer_fullness;
ctx->bInserHRDParams = IMG_TRUE;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "hrd param buffer_size set to %d "
"initial buffer fullness set to %d\n",
ctx->buffer_size, ctx->initial_buffer_fullness);
break;
default:
vaStatus = VA_STATUS_ERROR_UNKNOWN;
DEBUG_FAILURE;
break;
}
free(obj_buffer->buffer_data);
obj_buffer->buffer_data = NULL;
return vaStatus;
}
static VAStatus pnw_H264ES_RenderPicture(
object_context_p obj_context,
object_buffer_p *buffers,
int num_buffers)
{
INIT_CONTEXT_H264ES;
VAStatus vaStatus = VA_STATUS_SUCCESS;
int i;
drv_debug_msg(VIDEO_DEBUG_GENERAL,"pnw_H264ES_RenderPicture\n");
for (i = 0; i < num_buffers; i++) {
object_buffer_p obj_buffer = buffers[i];
switch (obj_buffer->type) {
case VAEncSequenceParameterBufferType:
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264_RenderPicture got VAEncSequenceParameterBufferType\n");
vaStatus = pnw__H264ES_process_sequence_param(ctx, obj_buffer);
DEBUG_FAILURE;
break;
case VAEncPictureParameterBufferType:
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264_RenderPicture got VAEncPictureParameterBuffer\n");
vaStatus = pnw__H264ES_process_picture_param(ctx, obj_buffer);
DEBUG_FAILURE;
break;
case VAEncSliceParameterBufferType:
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264_RenderPicture got VAEncSliceParameterBufferType\n");
vaStatus = pnw__H264ES_process_slice_param(ctx, obj_buffer);
DEBUG_FAILURE;
break;
case VAEncMiscParameterBufferType:
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264_RenderPicture got VAEncMiscParameterBufferType\n");
vaStatus = pnw__H264ES_process_misc_param(ctx, obj_buffer);
DEBUG_FAILURE;
break;
#if PSB_MFLD_DUMMY_CODE
case VAEncPackedHeaderParameterBufferType:
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264_RenderPicture got VAEncPackedHeaderParameterBufferType\n");
vaStatus = pnw__H264ES_insert_SEI_FPA_param(ctx, obj_buffer);
DEBUG_FAILURE;
break;
case VAEncPackedHeaderDataBufferType:
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264_RenderPicture got VAEncPackedHeaderDataBufferType\n");
vaStatus = pnw__H264ES_insert_SEI_FPA_data(ctx, obj_buffer);
DEBUG_FAILURE;
break;
#endif
default:
vaStatus = VA_STATUS_ERROR_UNKNOWN;
DEBUG_FAILURE;
}
if (vaStatus != VA_STATUS_SUCCESS) {
break;
}
}
return vaStatus;
}
static VAStatus pnw_H264ES_EndPicture(
object_context_p obj_context)
{
INIT_CONTEXT_H264ES;
pnw_cmdbuf_p cmdbuf = ctx->obj_context->pnw_cmdbuf;
PIC_PARAMS *psPicParams = (PIC_PARAMS *)cmdbuf->pic_params_p;
VAStatus vaStatus = VA_STATUS_SUCCESS;
unsigned char core = 0;
drv_debug_msg(VIDEO_DEBUG_GENERAL, "pnw_H264ES_EndPicture\n");
/* Unlike MPEG4 and H263, slices number is defined by user */
for (core = 0; core < ctx->ParallelCores; core++) {
psPicParams = (PIC_PARAMS *)
(cmdbuf->pic_params_p + ctx->pic_params_size * core);
psPicParams->NumSlices = ctx->sRCParams.Slices;
}
vaStatus = pnw_EndPicture(ctx);
return vaStatus;
}
struct format_vtable_s pnw_H264ES_vtable = {
queryConfigAttributes:
pnw_H264ES_QueryConfigAttributes,
validateConfig:
pnw_H264ES_ValidateConfig,
createContext:
pnw_H264ES_CreateContext,
destroyContext:
pnw_H264ES_DestroyContext,
beginPicture:
pnw_H264ES_BeginPicture,
renderPicture:
pnw_H264ES_RenderPicture,
endPicture:
pnw_H264ES_EndPicture
};
VAStatus pnw_set_frame_skip_flag(
object_context_p obj_context)
{
INIT_CONTEXT_H264ES;
VAStatus vaStatus = VA_STATUS_SUCCESS;
if (ctx && ctx->previous_src_surface) {
SET_SURFACE_INFO_skipped_flag(ctx->previous_src_surface->psb_surface, 1);
drv_debug_msg(VIDEO_DEBUG_GENERAL, "Detected a skipped frame for surface 0x%08x.\n",
ctx->previous_src_surface->psb_surface);
}
return vaStatus;
}