/* * * 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 #include #include #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, };