/* * Copyright 2020 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. */ #include "iep2_api.h" #include #include #include #include #include #include #include #include "mpp_env.h" #include "mpp_debug.h" #include "mpp_buffer.h" #include "iep2_ff.h" #include "iep2_pd.h" #include "iep2_gmv.h" #include "iep2_osd.h" #include "iep2_roi.h" #include "iep2.h" #include "mpp_service.h" #include "mpp_platform.h" #define IEP2_TILE_W_MAX 120 #define IEP2_TILE_H_MAX 480 RK_U32 iep_debug = 0; static MPP_RET iep2_init(IepCtx *ictx) { MPP_RET ret; struct iep2_api_ctx *ctx = *ictx; MppReqV1 mpp_req; RK_U32 client_data = IEP_CLIENT_TYPE; mpp_env_get_u32("iep_debug", &iep_debug, 0); ctx->fd = open("/dev/mpp_service", O_RDWR | O_CLOEXEC); if (ctx->fd < 0) { mpp_err("can NOT find device /dev/iep2\n"); return MPP_NOK; } mpp_req.cmd = MPP_CMD_INIT_CLIENT_TYPE; mpp_req.flag = 0; mpp_req.size = sizeof(client_data); mpp_req.data_ptr = REQ_DATA_PTR(&client_data); ret = (RK_S32)ioctl(ctx->fd, MPP_IOC_CFG_V1, &mpp_req); if (ret) { mpp_err("ioctl set_client failed\n"); return MPP_NOK; } // set default parameters. ctx->params.src_fmt = IEP2_FMT_YUV420; ctx->params.src_yuv_swap = IEP2_YUV_SWAP_SP_UV; ctx->params.dst_fmt = IEP2_FMT_YUV420; ctx->params.dst_yuv_swap = IEP2_YUV_SWAP_SP_UV; ctx->params.src_y_stride = 720 / 4; ctx->params.src_uv_stride = 720 / 4; ctx->params.dst_y_stride = 720 / 4; ctx->params.tile_cols = 720 / 16; ctx->params.tile_rows = 480 / 4; ctx->params.dil_mode = IEP2_DIL_MODE_I1O1T; ctx->params.dil_out_mode = IEP2_OUT_MODE_LINE; ctx->params.dil_field_order = IEP2_FIELD_ORDER_TFF; ctx->params.md_theta = 1; ctx->params.md_r = 4; ctx->params.md_lambda = 4; ctx->params.dect_resi_thr = 30; ctx->params.osd_area_num = 0; ctx->params.osd_gradh_thr = 60; ctx->params.osd_gradv_thr = 60; ctx->params.osd_pos_limit_en = 0; ctx->params.osd_pos_limit_num = 0; ctx->params.osd_pec_thr = 20; ctx->params.osd_line_num = 2; ctx->params.me_pena = 4; ctx->params.mv_similar_thr = 4; ctx->params.mv_similar_num_thr0 = 4; ctx->params.mv_bonus = 10; ctx->params.me_thr_offset = 20; ctx->params.mv_left_limit = 28; ctx->params.mv_right_limit = 27; ctx->params.eedi_thr0 = 12; memset(ctx->params.comb_osd_vld, 1, sizeof(ctx->params.comb_osd_vld)); ctx->params.comb_t_thr = 4; ctx->params.comb_feature_thr = 16; ctx->params.comb_cnt_thr = 0; ctx->params.ble_backtoma_num = 1; ctx->params.mtn_en = 0; ctx->params.roi_en = 0; ctx->params.roi_layer_num = 0; ctx->params.dil_mode = IEP2_DIL_MODE_I1O1T; ctx->params.src_fmt = IEP2_FMT_YUV420; ctx->params.src_yuv_swap = IEP2_YUV_SWAP_SP_UV; ctx->params.dst_fmt = IEP2_FMT_YUV420; ctx->params.dst_yuv_swap = IEP2_YUV_SWAP_SP_UV; ctx->params.src_y_stride = 720 / 4; ctx->params.src_uv_stride = 720 / 4; ctx->params.dst_y_stride = 720 / 4; ctx->params.tile_cols = 720 / 16; ctx->params.tile_rows = 480 / 4; memset(&ctx->ff_inf, 0, sizeof(ctx->ff_inf)); memset(&ctx->pd_inf, 0, sizeof(ctx->pd_inf)); ctx->pd_inf.pdtype = PD_TYPES_UNKNOWN; ctx->pd_inf.step = -1; ret = mpp_buffer_group_get_internal(&ctx->memGroup, MPP_BUFFER_TYPE_DRM); if (MPP_OK != ret) { close(ctx->fd); mpp_err("memGroup mpp_buffer_group_get failed\n"); return ret; } ret = mpp_buffer_get(ctx->memGroup, &ctx->mv_buf, IEP2_TILE_W_MAX * IEP2_TILE_H_MAX); if (ret) { close(ctx->fd); mpp_buffer_group_put(ctx->memGroup); mpp_err_f("allocate mv buffer failed\n"); return MPP_NOK; } ret = mpp_buffer_get(ctx->memGroup, &ctx->md_buf, 1920 * 1088); if (ret) { close(ctx->fd); mpp_buffer_group_put(ctx->memGroup); mpp_buffer_put(ctx->mv_buf); mpp_err_f("allocate md buffer failed\n"); return MPP_NOK; } ctx->params.mv_addr = mpp_buffer_get_fd(ctx->mv_buf); ctx->params.md_addr = mpp_buffer_get_fd(ctx->md_buf); return MPP_OK; } static MPP_RET iep2_deinit(IepCtx ictx) { struct iep2_api_ctx *ctx = ictx; close(ctx->fd); mpp_buffer_put(ctx->mv_buf); mpp_buffer_put(ctx->md_buf); if (ctx->memGroup) { mpp_buffer_group_put(ctx->memGroup); ctx->memGroup = NULL; } return MPP_OK; } static MPP_RET iep2_done(struct iep2_api_ctx *ctx) { if (ctx->params.dil_mode == IEP2_DIL_MODE_I5O2 || ctx->params.dil_mode == IEP2_DIL_MODE_I5O1T || ctx->params.dil_mode == IEP2_DIL_MODE_I5O1B) { struct mv_list ls; iep2_set_osd(ctx, &ls); iep2_update_gmv(ctx, &ls); iep2_check_ffo(ctx); iep2_check_pd(ctx); #if 0 if (ctx->params.roi_en && ctx->params.osd_area_num > 0) { struct iep2_rect r; ctx->params.roi_layer_num = 0; r.x = 0; r.y = 0; r.w = ctx->params.tile_cols; r.h = ctx->params.tile_rows; iep2_set_roi(ctx, &r, ROI_MODE_MA); r.x = ctx->params.osd_x_sta[0]; r.y = ctx->params.osd_y_sta[0]; r.w = ctx->params.osd_x_end[0] - ctx->params.osd_x_sta[0]; r.h = ctx->params.osd_y_end[0] - ctx->params.osd_y_sta[0]; iep2_set_roi(ctx, &r, ROI_MODE_MA_MC); } #endif } if (ctx->params.dil_mode == IEP2_DIL_MODE_DECT || ctx->params.dil_mode == IEP2_DIL_MODE_PD) { iep2_check_ffo(ctx); iep2_check_pd(ctx); } if (ctx->pd_inf.pdtype != PD_TYPES_UNKNOWN) { ctx->params.dil_mode = IEP2_DIL_MODE_PD; ctx->params.pd_mode = iep2_pd_get_output(&ctx->pd_inf); } else { // TODO, revise others mode replace by I5O2 //ctx->params.dil_mode = IEP2_DIL_MODE_I5O2; } return 0; } static void iep2_set_param(struct iep2_api_ctx *ctx, union iep2_api_content *param, enum IEP2_PARAM_TYPE type) { switch (type) { case IEP2_PARAM_TYPE_COM: ctx->params.src_fmt = param->com.sfmt; ctx->params.src_yuv_swap = param->com.sswap; ctx->params.dst_fmt = param->com.dfmt; ctx->params.dst_yuv_swap = param->com.dswap; ctx->params.src_y_stride = param->com.width; ctx->params.src_y_stride /= 4; ctx->params.src_uv_stride = param->com.sswap == IEP2_YUV_SWAP_P ? (param->com.width / 2 + 15) / 16 * 16 : param->com.width; ctx->params.src_uv_stride /= 4; ctx->params.dst_y_stride = param->com.width; ctx->params.dst_y_stride /= 4; ctx->params.tile_cols = (param->com.width + 15) / 16; ctx->params.tile_rows = (param->com.height + 3) / 4; iep_dbg_trace("set tile size (%d, %d)\n", param->com.width, param->com.height); ctx->params.osd_pec_thr = (param->com.width * 26) >> 7; break; case IEP2_PARAM_TYPE_MODE: ctx->params.dil_mode = param->mode.dil_mode; ctx->params.dil_out_mode = param->mode.out_mode; if (!ctx->ff_inf.fo_detected) { ctx->params.dil_field_order = param->mode.dil_order; } if (param->mode.dil_order == 0) { ctx->ff_inf.tff_offset = 5; ctx->ff_inf.bff_offset = 0; } else { ctx->ff_inf.tff_offset = 0; ctx->ff_inf.bff_offset = 5; } break; case IEP2_PARAM_TYPE_MD: ctx->params.md_theta = param->md.md_theta; ctx->params.md_r = param->md.md_r; ctx->params.md_lambda = param->md.md_lambda; break; case IEP2_PARAM_TYPE_DECT: case IEP2_PARAM_TYPE_OSD: case IEP2_PARAM_TYPE_ME: case IEP2_PARAM_TYPE_EEDI: case IEP2_PARAM_TYPE_BLE: case IEP2_PARAM_TYPE_COMB: break; case IEP2_PARAM_TYPE_ROI: ctx->params.roi_en = param->roi.roi_en; break; default: ; } } static MPP_RET iep2_param_check(struct iep2_api_ctx *ctx) { if (ctx->params.tile_cols <= 0 || ctx->params.tile_cols > IEP2_TILE_W_MAX || ctx->params.tile_rows <= 0 || ctx->params.tile_rows > IEP2_TILE_H_MAX) { mpp_err("invalidate size (%u, %u)\n", ctx->params.tile_cols, ctx->params.tile_rows); return MPP_NOK; } return MPP_OK; } static MPP_RET iep2_start(struct iep2_api_ctx *ctx) { MPP_RET ret; MppReqV1 mpp_req[2]; mpp_assert(ctx); mpp_req[0].cmd = MPP_CMD_SET_REG_WRITE; mpp_req[0].flag = MPP_FLAGS_MULTI_MSG; mpp_req[0].size = sizeof(ctx->params); mpp_req[0].offset = 0; mpp_req[0].data_ptr = REQ_DATA_PTR(&ctx->params); mpp_req[1].cmd = MPP_CMD_SET_REG_READ; mpp_req[1].flag = MPP_FLAGS_MULTI_MSG | MPP_FLAGS_LAST_MSG; mpp_req[1].size = sizeof(ctx->output); mpp_req[1].offset = 0; mpp_req[1].data_ptr = REQ_DATA_PTR(&ctx->output); iep_dbg_func("in\n"); ret = (RK_S32)ioctl(ctx->fd, MPP_IOC_CFG_V1, &mpp_req[0]); if (ret) { mpp_err_f("ioctl SET_REG failed ret %d errno %d %s\n", ret, errno, strerror(errno)); ret = errno; } return ret; } static MPP_RET iep2_wait(struct iep2_api_ctx *ctx) { MppReqV1 mpp_req; MPP_RET ret; memset(&mpp_req, 0, sizeof(mpp_req)); mpp_req.cmd = MPP_CMD_POLL_HW_FINISH; mpp_req.flag |= MPP_FLAGS_LAST_MSG; ret = (RK_S32)ioctl(ctx->fd, MPP_IOC_CFG_V1, &mpp_req); return ret; } static inline void set_addr(struct iep2_addr *addr, IepImg *img) { addr->y = img->mem_addr; addr->cbcr = img->uv_addr; addr->cr = img->v_addr; } static MPP_RET iep2_control(IepCtx ictx, IepCmd cmd, void *iparam) { struct iep2_api_ctx *ctx = ictx; switch (cmd) { case IEP_CMD_SET_DEI_CFG: { struct iep2_api_params *param = (struct iep2_api_params *)iparam; iep2_set_param(ctx, ¶m->param, param->ptype); } break; case IEP_CMD_SET_SRC: set_addr(&ctx->params.src[0], (IepImg *)iparam); break; case IEP_CMD_SET_DEI_SRC1: set_addr(&ctx->params.src[1], (IepImg *)iparam); break; case IEP_CMD_SET_DEI_SRC2: set_addr(&ctx->params.src[2], (IepImg *)iparam); break; case IEP_CMD_SET_DST: set_addr(&ctx->params.dst[0], (IepImg *)iparam); break; case IEP_CMD_SET_DEI_DST1: set_addr(&ctx->params.dst[1], (IepImg *)iparam); break; case IEP_CMD_RUN_SYNC: { struct iep2_api_info *inf = (struct iep2_api_info*)iparam; if (0 > iep2_param_check(ctx)) break; if (0 > iep2_start(ctx)) return MPP_NOK; iep2_wait(ctx); if (inf) inf->pd_flag = ctx->params.pd_mode; iep2_done(ctx); if (inf) { inf->dil_order = ctx->params.dil_field_order; inf->frm_mode = ctx->ff_inf.is_frm; inf->pd_types = ctx->pd_inf.pdtype; } } break; default: ; } return MPP_OK; } static iep_com_ops iep2_ops = { .init = iep2_init, .deinit = iep2_deinit, .control = iep2_control, .release = NULL, }; iep_com_ctx* rockchip_iep2_api_alloc_ctx(void) { iep_com_ctx *com_ctx = calloc(sizeof(*com_ctx), 1); struct iep2_api_ctx *iep2_ctx = calloc(sizeof(*iep2_ctx), 1); mpp_assert(com_ctx && iep2_ctx); com_ctx->ops = &iep2_ops; com_ctx->priv = iep2_ctx; return com_ctx; } void rockchip_iep2_api_release_ctx(iep_com_ctx *com_ctx) { if (com_ctx->priv) { free(com_ctx->priv); com_ctx->priv = NULL; } free(com_ctx); }