/*
|
*
|
* Copyright 2015 Rockchip Electronics Co. LTD
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
|
|
#define MODULE_TAG "hal_avsd_api"
|
|
#include <stdio.h>
|
#include <stdlib.h>
|
#include <string.h>
|
|
#include "mpp_mem.h"
|
#include "mpp_env.h"
|
#include "mpp_common.h"
|
|
#include "hal_avsd_api.h"
|
#include "hal_avsd_reg.h"
|
#include "mpp_dec_cb_param.h"
|
|
RK_U32 avsd_hal_debug = 0;
|
|
MPP_RET hal_avsd_start(void *decoder, HalTaskInfo *task);
|
MPP_RET hal_avsd_wait(void *decoder, HalTaskInfo *task);
|
|
static RK_U32 avsd_ver_align(RK_U32 val)
|
{
|
return MPP_ALIGN(val, 16);
|
}
|
|
static RK_U32 avsd_hor_align(RK_U32 val)
|
{
|
return MPP_ALIGN(val, 16);
|
}
|
|
static RK_U32 avsd_len_align(RK_U32 val)
|
{
|
return (2 * MPP_ALIGN(val, 16));
|
}
|
|
static void explain_input_buffer(AvsdHalCtx_t *p_hal, HalDecTask *task)
|
{
|
memcpy(&p_hal->syn, task->syntax.data, sizeof(AvsdSyntax_t));
|
}
|
|
static MPP_RET repeat_other_field(AvsdHalCtx_t *p_hal, HalTaskInfo *task)
|
{
|
RK_U8 i = 0;
|
RK_U8 *pdata = NULL;
|
MppBuffer mbuffer = NULL;
|
MPP_RET ret = MPP_ERR_UNKNOW;
|
AvsdRegs_t *p_regs = (AvsdRegs_t *)p_hal->p_regs;
|
|
//!< re-find start code and calculate offset
|
p_hal->data_offset = p_regs->sw12.rlc_vlc_base >> 10;
|
mpp_buf_slot_get_prop(p_hal->packet_slots, task->dec.input, SLOT_BUFFER, &mbuffer);
|
pdata = (RK_U8 *)mpp_buffer_get_ptr(mbuffer) + p_hal->data_offset;
|
while (i < 8) {
|
if (pdata[i] == 0 && pdata[i + 1] == 0 && pdata[i + 2] == 1) {
|
p_hal->data_offset += i;
|
p_hal->syn.bitstream_size -= i;
|
break;
|
}
|
i++;
|
}
|
AVSD_HAL_DBG(AVSD_HAL_DBG_OFFSET, "frame_no=%d, offset=%d\n", p_hal->frame_no, p_hal->data_offset);
|
//!< re-generate register
|
memset(p_hal->p_regs, 0, sizeof(AvsdRegs_t));
|
FUN_CHECK(ret = set_defalut_parameters(p_hal));
|
FUN_CHECK(ret = set_regs_parameters(p_hal, &task->dec));
|
hal_avsd_start((void *)p_hal, task);
|
hal_avsd_wait((void *)p_hal, task);
|
|
return ret = MPP_OK;
|
__FAILED:
|
return ret;
|
|
}
|
/*!
|
***********************************************************************
|
* \brief
|
* init
|
***********************************************************************
|
*/
|
//extern "C"
|
MPP_RET hal_avsd_init(void *decoder, MppHalCfg *cfg)
|
{
|
MPP_RET ret = MPP_ERR_UNKNOW;
|
AvsdHalCtx_t *p_hal = NULL;
|
|
AVSD_HAL_TRACE("In.");
|
INP_CHECK(ret, NULL == decoder);
|
|
mpp_env_get_u32("avsd_debug", &avsd_hal_debug, 0);
|
|
p_hal = (AvsdHalCtx_t *)decoder;
|
memset(p_hal, 0, sizeof(AvsdHalCtx_t));
|
|
p_hal->frame_slots = cfg->frame_slots;
|
p_hal->packet_slots = cfg->packet_slots;
|
//!< callback function to parser module
|
p_hal->dec_cb = cfg->dec_cb;
|
//!< mpp_device_init
|
|
ret = mpp_dev_init(&p_hal->dev, VPU_CLIENT_AVSPLUS_DEC);
|
if (ret) {
|
mpp_err("mpp_device_init failed. ret: %d\n", ret);
|
return ret;
|
}
|
|
//< get buffer group
|
if (p_hal->buf_group == NULL) {
|
RK_U32 buf_size = (1920 * 1088) / 16;
|
|
FUN_CHECK(ret = mpp_buffer_group_get_internal(&p_hal->buf_group, MPP_BUFFER_TYPE_ION));
|
FUN_CHECK(ret = mpp_buffer_get(p_hal->buf_group, &p_hal->mv_buf, buf_size));
|
}
|
|
mpp_slots_set_prop(p_hal->frame_slots, SLOTS_HOR_ALIGN, avsd_hor_align);
|
mpp_slots_set_prop(p_hal->frame_slots, SLOTS_VER_ALIGN, avsd_ver_align);
|
mpp_slots_set_prop(p_hal->frame_slots, SLOTS_LEN_ALIGN, avsd_len_align);
|
|
p_hal->p_regs = mpp_calloc_size(RK_U32 , sizeof(AvsdRegs_t));
|
MEM_CHECK(ret, p_hal->p_regs);
|
|
//!< initial for control
|
p_hal->first_field = 1;
|
p_hal->prev_pic_structure = 0; //!< field
|
|
memset(p_hal->pic, 0, sizeof(p_hal->pic));
|
p_hal->work_out = -1;
|
p_hal->work0 = -1;
|
p_hal->work1 = -1;
|
|
__RETURN:
|
AVSD_HAL_TRACE("Out.");
|
|
return ret = MPP_OK;
|
__FAILED:
|
return ret;
|
}
|
/*!
|
***********************************************************************
|
* \brief
|
* deinit
|
***********************************************************************
|
*/
|
//extern "C"
|
MPP_RET hal_avsd_deinit(void *decoder)
|
{
|
MPP_RET ret = MPP_ERR_UNKNOW;
|
AvsdHalCtx_t *p_hal = (AvsdHalCtx_t *)decoder;
|
|
AVSD_HAL_TRACE("In.");
|
INP_CHECK(ret, NULL == decoder);
|
|
//!< mpp_device_init
|
if (p_hal->dev) {
|
ret = mpp_dev_deinit(p_hal->dev);
|
if (ret)
|
mpp_err("mpp_dev_deinit failed. ret: %d\n", ret);
|
}
|
if (p_hal->mv_buf) {
|
FUN_CHECK(ret = mpp_buffer_put(p_hal->mv_buf));
|
}
|
if (p_hal->buf_group) {
|
FUN_CHECK(ret = mpp_buffer_group_put(p_hal->buf_group));
|
}
|
MPP_FREE(p_hal->p_regs);
|
|
__RETURN:
|
AVSD_HAL_TRACE("Out.");
|
|
return ret = MPP_OK;
|
__FAILED:
|
return ret;
|
}
|
/*!
|
***********************************************************************
|
* \brief
|
* generate register
|
***********************************************************************
|
*/
|
//extern "C"
|
MPP_RET hal_avsd_gen_regs(void *decoder, HalTaskInfo *task)
|
{
|
MPP_RET ret = MPP_ERR_UNKNOW;
|
|
AvsdHalCtx_t *p_hal = (AvsdHalCtx_t *)decoder;
|
AVSD_HAL_TRACE("In.");
|
INP_CHECK(ret, NULL == decoder);
|
if (task->dec.flags.parse_err ||
|
task->dec.flags.ref_err) {
|
goto __RETURN;
|
}
|
memset(p_hal->p_regs, 0, sizeof(AvsdRegs_t));
|
explain_input_buffer(p_hal, &task->dec);
|
p_hal->data_offset = 0;
|
|
FUN_CHECK(ret = set_defalut_parameters(p_hal));
|
FUN_CHECK(ret = set_regs_parameters(p_hal, &task->dec));
|
|
__RETURN:
|
AVSD_HAL_TRACE("Out.");
|
|
return ret = MPP_OK;
|
__FAILED:
|
return ret;
|
}
|
/*!
|
***********************************************************************
|
* \brief h
|
* start hard
|
***********************************************************************
|
*/
|
//extern "C"
|
MPP_RET hal_avsd_start(void *decoder, HalTaskInfo *task)
|
{
|
MPP_RET ret = MPP_ERR_UNKNOW;
|
AvsdHalCtx_t *p_hal = (AvsdHalCtx_t *)decoder;
|
|
AVSD_HAL_TRACE("In.");
|
INP_CHECK(ret, NULL == decoder);
|
|
if (task->dec.flags.parse_err || task->dec.flags.ref_err) {
|
goto __RETURN;
|
}
|
|
do {
|
MppDevRegWrCfg wr_cfg;
|
MppDevRegRdCfg rd_cfg;
|
|
wr_cfg.reg = p_hal->p_regs;
|
wr_cfg.size = AVSD_REGISTERS * sizeof(RK_U32);
|
wr_cfg.offset = 0;
|
|
ret = mpp_dev_ioctl(p_hal->dev, MPP_DEV_REG_WR, &wr_cfg);
|
if (ret) {
|
mpp_err_f("set register write failed %d\n", ret);
|
break;
|
}
|
|
rd_cfg.reg = p_hal->p_regs;
|
rd_cfg.size = AVSD_REGISTERS * sizeof(RK_U32);
|
rd_cfg.offset = 0;
|
|
ret = mpp_dev_ioctl(p_hal->dev, MPP_DEV_REG_RD, &rd_cfg);
|
if (ret) {
|
mpp_err_f("set register read failed %d\n", ret);
|
break;
|
}
|
|
ret = mpp_dev_ioctl(p_hal->dev, MPP_DEV_CMD_SEND, NULL);
|
if (ret) {
|
mpp_err_f("send cmd failed %d\n", ret);
|
break;
|
}
|
} while (0);
|
|
p_hal->frame_no++;
|
|
__RETURN:
|
AVSD_HAL_TRACE("Out.");
|
|
return ret;
|
}
|
/*!
|
***********************************************************************
|
* \brief
|
* wait hard
|
***********************************************************************
|
*/
|
//extern "C"
|
MPP_RET hal_avsd_wait(void *decoder, HalTaskInfo *task)
|
{
|
MPP_RET ret = MPP_ERR_UNKNOW;
|
AvsdHalCtx_t *p_hal = (AvsdHalCtx_t *)decoder;
|
|
AVSD_HAL_TRACE("In.");
|
INP_CHECK(ret, NULL == decoder);
|
|
if (task->dec.flags.parse_err ||
|
task->dec.flags.ref_err) {
|
goto __SKIP_HARD;
|
}
|
|
ret = mpp_dev_ioctl(p_hal->dev, MPP_DEV_CMD_POLL, NULL);
|
if (ret)
|
mpp_err_f("poll cmd failed %d\n", ret);
|
|
__SKIP_HARD:
|
if (p_hal->dec_cb) {
|
DecCbHalDone param;
|
|
param.task = (void *)&task->dec;
|
param.regs = (RK_U32 *)p_hal->p_regs;
|
param.hard_err = (!((AvsdRegs_t *)p_hal->p_regs)->sw01.dec_rdy_int);
|
|
mpp_callback(p_hal->dec_cb, ¶m);
|
}
|
update_parameters(p_hal);
|
memset(&p_hal->p_regs[1], 0, sizeof(RK_U32));
|
if (!p_hal->first_field
|
&& p_hal->syn.pp.pictureStructure == FIELDPICTURE) {
|
repeat_other_field(p_hal, task);
|
}
|
|
__RETURN:
|
AVSD_HAL_TRACE("Out.");
|
|
return ret;
|
}
|
/*!
|
***********************************************************************
|
* \brief
|
* reset
|
***********************************************************************
|
*/
|
//extern "C"
|
MPP_RET hal_avsd_reset(void *decoder)
|
{
|
MPP_RET ret = MPP_ERR_UNKNOW;
|
AvsdHalCtx_t *p_hal = (AvsdHalCtx_t *)decoder;
|
|
AVSD_HAL_TRACE("In.");
|
|
memset(p_hal->p_regs, 0, sizeof(AvsdRegs_t));
|
|
p_hal->first_field = 1;
|
p_hal->prev_pic_structure = 0; //!< field
|
|
memset(p_hal->pic, 0, sizeof(p_hal->pic));
|
p_hal->work_out = -1;
|
p_hal->work0 = -1;
|
p_hal->work1 = -1;
|
|
|
AVSD_HAL_TRACE("Out.");
|
|
return ret = MPP_OK;
|
}
|
|
const MppHalApi hal_api_avsd = {
|
.name = "avsd_rkdec",
|
.type = MPP_CTX_DEC,
|
.coding = MPP_VIDEO_CodingAVSPLUS,
|
.ctx_size = sizeof(AvsdHalCtx_t),
|
.flag = 0,
|
.init = hal_avsd_init,
|
.deinit = hal_avsd_deinit,
|
.reg_gen = hal_avsd_gen_regs,
|
.start = hal_avsd_start,
|
.wait = hal_avsd_wait,
|
.reset = hal_avsd_reset,
|
.flush = NULL,
|
.control = NULL,
|
};
|