archived-mpp/mpp/hal/rkdec/avsd/hal_avsd_api.c

370 lines
9.4 KiB
C
Raw Normal View History

/*
*
* 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, &param);
}
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,
};