18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * Copyright (c) 2012-2016, The Linux Foundation. All rights reserved.
48c2ecf20Sopenharmony_ci * Copyright (C) 2017 Linaro Ltd.
58c2ecf20Sopenharmony_ci */
68c2ecf20Sopenharmony_ci#include <linux/slab.h>
78c2ecf20Sopenharmony_ci#include <linux/mutex.h>
88c2ecf20Sopenharmony_ci#include <linux/list.h>
98c2ecf20Sopenharmony_ci#include <linux/completion.h>
108c2ecf20Sopenharmony_ci#include <linux/platform_device.h>
118c2ecf20Sopenharmony_ci#include <linux/videodev2.h>
128c2ecf20Sopenharmony_ci
138c2ecf20Sopenharmony_ci#include "core.h"
148c2ecf20Sopenharmony_ci#include "hfi.h"
158c2ecf20Sopenharmony_ci#include "hfi_cmds.h"
168c2ecf20Sopenharmony_ci#include "hfi_venus.h"
178c2ecf20Sopenharmony_ci
188c2ecf20Sopenharmony_ci#define TIMEOUT		msecs_to_jiffies(1000)
198c2ecf20Sopenharmony_ci
208c2ecf20Sopenharmony_cistatic u32 to_codec_type(u32 pixfmt)
218c2ecf20Sopenharmony_ci{
228c2ecf20Sopenharmony_ci	switch (pixfmt) {
238c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_H264:
248c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_H264_NO_SC:
258c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_H264;
268c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_H263:
278c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_H263;
288c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_MPEG1:
298c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_MPEG1;
308c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_MPEG2:
318c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_MPEG2;
328c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_MPEG4:
338c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_MPEG4;
348c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_VC1_ANNEX_G:
358c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_VC1_ANNEX_L:
368c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_VC1;
378c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_VP8:
388c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_VP8;
398c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_VP9:
408c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_VP9;
418c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_XVID:
428c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_DIVX;
438c2ecf20Sopenharmony_ci	case V4L2_PIX_FMT_HEVC:
448c2ecf20Sopenharmony_ci		return HFI_VIDEO_CODEC_HEVC;
458c2ecf20Sopenharmony_ci	default:
468c2ecf20Sopenharmony_ci		return 0;
478c2ecf20Sopenharmony_ci	}
488c2ecf20Sopenharmony_ci}
498c2ecf20Sopenharmony_ci
508c2ecf20Sopenharmony_ciint hfi_core_init(struct venus_core *core)
518c2ecf20Sopenharmony_ci{
528c2ecf20Sopenharmony_ci	int ret = 0;
538c2ecf20Sopenharmony_ci
548c2ecf20Sopenharmony_ci	mutex_lock(&core->lock);
558c2ecf20Sopenharmony_ci
568c2ecf20Sopenharmony_ci	if (core->state >= CORE_INIT)
578c2ecf20Sopenharmony_ci		goto unlock;
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_ci	reinit_completion(&core->done);
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_ci	ret = core->ops->core_init(core);
628c2ecf20Sopenharmony_ci	if (ret)
638c2ecf20Sopenharmony_ci		goto unlock;
648c2ecf20Sopenharmony_ci
658c2ecf20Sopenharmony_ci	ret = wait_for_completion_timeout(&core->done, TIMEOUT);
668c2ecf20Sopenharmony_ci	if (!ret) {
678c2ecf20Sopenharmony_ci		ret = -ETIMEDOUT;
688c2ecf20Sopenharmony_ci		goto unlock;
698c2ecf20Sopenharmony_ci	}
708c2ecf20Sopenharmony_ci
718c2ecf20Sopenharmony_ci	ret = 0;
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_ci	if (core->error != HFI_ERR_NONE) {
748c2ecf20Sopenharmony_ci		ret = -EIO;
758c2ecf20Sopenharmony_ci		goto unlock;
768c2ecf20Sopenharmony_ci	}
778c2ecf20Sopenharmony_ci
788c2ecf20Sopenharmony_ci	core->state = CORE_INIT;
798c2ecf20Sopenharmony_ciunlock:
808c2ecf20Sopenharmony_ci	mutex_unlock(&core->lock);
818c2ecf20Sopenharmony_ci	return ret;
828c2ecf20Sopenharmony_ci}
838c2ecf20Sopenharmony_ci
848c2ecf20Sopenharmony_ciint hfi_core_deinit(struct venus_core *core, bool blocking)
858c2ecf20Sopenharmony_ci{
868c2ecf20Sopenharmony_ci	int ret = 0, empty;
878c2ecf20Sopenharmony_ci
888c2ecf20Sopenharmony_ci	mutex_lock(&core->lock);
898c2ecf20Sopenharmony_ci
908c2ecf20Sopenharmony_ci	if (core->state == CORE_UNINIT)
918c2ecf20Sopenharmony_ci		goto unlock;
928c2ecf20Sopenharmony_ci
938c2ecf20Sopenharmony_ci	empty = list_empty(&core->instances);
948c2ecf20Sopenharmony_ci
958c2ecf20Sopenharmony_ci	if (!empty && !blocking) {
968c2ecf20Sopenharmony_ci		ret = -EBUSY;
978c2ecf20Sopenharmony_ci		goto unlock;
988c2ecf20Sopenharmony_ci	}
998c2ecf20Sopenharmony_ci
1008c2ecf20Sopenharmony_ci	if (!empty) {
1018c2ecf20Sopenharmony_ci		mutex_unlock(&core->lock);
1028c2ecf20Sopenharmony_ci		wait_var_event(&core->insts_count,
1038c2ecf20Sopenharmony_ci			       !atomic_read(&core->insts_count));
1048c2ecf20Sopenharmony_ci		mutex_lock(&core->lock);
1058c2ecf20Sopenharmony_ci	}
1068c2ecf20Sopenharmony_ci
1078c2ecf20Sopenharmony_ci	if (!core->ops)
1088c2ecf20Sopenharmony_ci		goto unlock;
1098c2ecf20Sopenharmony_ci
1108c2ecf20Sopenharmony_ci	ret = core->ops->core_deinit(core);
1118c2ecf20Sopenharmony_ci
1128c2ecf20Sopenharmony_ci	if (!ret)
1138c2ecf20Sopenharmony_ci		core->state = CORE_UNINIT;
1148c2ecf20Sopenharmony_ci
1158c2ecf20Sopenharmony_ciunlock:
1168c2ecf20Sopenharmony_ci	mutex_unlock(&core->lock);
1178c2ecf20Sopenharmony_ci	return ret;
1188c2ecf20Sopenharmony_ci}
1198c2ecf20Sopenharmony_ci
1208c2ecf20Sopenharmony_ciint hfi_core_suspend(struct venus_core *core)
1218c2ecf20Sopenharmony_ci{
1228c2ecf20Sopenharmony_ci	if (core->state != CORE_INIT)
1238c2ecf20Sopenharmony_ci		return 0;
1248c2ecf20Sopenharmony_ci
1258c2ecf20Sopenharmony_ci	return core->ops->suspend(core);
1268c2ecf20Sopenharmony_ci}
1278c2ecf20Sopenharmony_ci
1288c2ecf20Sopenharmony_ciint hfi_core_resume(struct venus_core *core, bool force)
1298c2ecf20Sopenharmony_ci{
1308c2ecf20Sopenharmony_ci	if (!force && core->state != CORE_INIT)
1318c2ecf20Sopenharmony_ci		return 0;
1328c2ecf20Sopenharmony_ci
1338c2ecf20Sopenharmony_ci	return core->ops->resume(core);
1348c2ecf20Sopenharmony_ci}
1358c2ecf20Sopenharmony_ci
1368c2ecf20Sopenharmony_ciint hfi_core_trigger_ssr(struct venus_core *core, u32 type)
1378c2ecf20Sopenharmony_ci{
1388c2ecf20Sopenharmony_ci	return core->ops->core_trigger_ssr(core, type);
1398c2ecf20Sopenharmony_ci}
1408c2ecf20Sopenharmony_ci
1418c2ecf20Sopenharmony_ciint hfi_core_ping(struct venus_core *core)
1428c2ecf20Sopenharmony_ci{
1438c2ecf20Sopenharmony_ci	int ret;
1448c2ecf20Sopenharmony_ci
1458c2ecf20Sopenharmony_ci	mutex_lock(&core->lock);
1468c2ecf20Sopenharmony_ci
1478c2ecf20Sopenharmony_ci	ret = core->ops->core_ping(core, 0xbeef);
1488c2ecf20Sopenharmony_ci	if (ret)
1498c2ecf20Sopenharmony_ci		goto unlock;
1508c2ecf20Sopenharmony_ci
1518c2ecf20Sopenharmony_ci	ret = wait_for_completion_timeout(&core->done, TIMEOUT);
1528c2ecf20Sopenharmony_ci	if (!ret) {
1538c2ecf20Sopenharmony_ci		ret = -ETIMEDOUT;
1548c2ecf20Sopenharmony_ci		goto unlock;
1558c2ecf20Sopenharmony_ci	}
1568c2ecf20Sopenharmony_ci	ret = 0;
1578c2ecf20Sopenharmony_ci	if (core->error != HFI_ERR_NONE)
1588c2ecf20Sopenharmony_ci		ret = -ENODEV;
1598c2ecf20Sopenharmony_ciunlock:
1608c2ecf20Sopenharmony_ci	mutex_unlock(&core->lock);
1618c2ecf20Sopenharmony_ci	return ret;
1628c2ecf20Sopenharmony_ci}
1638c2ecf20Sopenharmony_ci
1648c2ecf20Sopenharmony_cistatic int wait_session_msg(struct venus_inst *inst)
1658c2ecf20Sopenharmony_ci{
1668c2ecf20Sopenharmony_ci	int ret;
1678c2ecf20Sopenharmony_ci
1688c2ecf20Sopenharmony_ci	ret = wait_for_completion_timeout(&inst->done, TIMEOUT);
1698c2ecf20Sopenharmony_ci	if (!ret)
1708c2ecf20Sopenharmony_ci		return -ETIMEDOUT;
1718c2ecf20Sopenharmony_ci
1728c2ecf20Sopenharmony_ci	if (inst->error != HFI_ERR_NONE)
1738c2ecf20Sopenharmony_ci		return -EIO;
1748c2ecf20Sopenharmony_ci
1758c2ecf20Sopenharmony_ci	return 0;
1768c2ecf20Sopenharmony_ci}
1778c2ecf20Sopenharmony_ci
1788c2ecf20Sopenharmony_ciint hfi_session_create(struct venus_inst *inst, const struct hfi_inst_ops *ops)
1798c2ecf20Sopenharmony_ci{
1808c2ecf20Sopenharmony_ci	struct venus_core *core = inst->core;
1818c2ecf20Sopenharmony_ci
1828c2ecf20Sopenharmony_ci	if (!ops)
1838c2ecf20Sopenharmony_ci		return -EINVAL;
1848c2ecf20Sopenharmony_ci
1858c2ecf20Sopenharmony_ci	inst->state = INST_UNINIT;
1868c2ecf20Sopenharmony_ci	init_completion(&inst->done);
1878c2ecf20Sopenharmony_ci	inst->ops = ops;
1888c2ecf20Sopenharmony_ci
1898c2ecf20Sopenharmony_ci	mutex_lock(&core->lock);
1908c2ecf20Sopenharmony_ci	list_add_tail(&inst->list, &core->instances);
1918c2ecf20Sopenharmony_ci	atomic_inc(&core->insts_count);
1928c2ecf20Sopenharmony_ci	mutex_unlock(&core->lock);
1938c2ecf20Sopenharmony_ci
1948c2ecf20Sopenharmony_ci	return 0;
1958c2ecf20Sopenharmony_ci}
1968c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_create);
1978c2ecf20Sopenharmony_ci
1988c2ecf20Sopenharmony_ciint hfi_session_init(struct venus_inst *inst, u32 pixfmt)
1998c2ecf20Sopenharmony_ci{
2008c2ecf20Sopenharmony_ci	struct venus_core *core = inst->core;
2018c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = core->ops;
2028c2ecf20Sopenharmony_ci	int ret;
2038c2ecf20Sopenharmony_ci
2048c2ecf20Sopenharmony_ci	if (inst->state != INST_UNINIT)
2058c2ecf20Sopenharmony_ci		return -EINVAL;
2068c2ecf20Sopenharmony_ci
2078c2ecf20Sopenharmony_ci	inst->hfi_codec = to_codec_type(pixfmt);
2088c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
2098c2ecf20Sopenharmony_ci
2108c2ecf20Sopenharmony_ci	ret = ops->session_init(inst, inst->session_type, inst->hfi_codec);
2118c2ecf20Sopenharmony_ci	if (ret)
2128c2ecf20Sopenharmony_ci		return ret;
2138c2ecf20Sopenharmony_ci
2148c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
2158c2ecf20Sopenharmony_ci	if (ret)
2168c2ecf20Sopenharmony_ci		return ret;
2178c2ecf20Sopenharmony_ci
2188c2ecf20Sopenharmony_ci	inst->state = INST_INIT;
2198c2ecf20Sopenharmony_ci
2208c2ecf20Sopenharmony_ci	return 0;
2218c2ecf20Sopenharmony_ci}
2228c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_init);
2238c2ecf20Sopenharmony_ci
2248c2ecf20Sopenharmony_civoid hfi_session_destroy(struct venus_inst *inst)
2258c2ecf20Sopenharmony_ci{
2268c2ecf20Sopenharmony_ci	struct venus_core *core = inst->core;
2278c2ecf20Sopenharmony_ci
2288c2ecf20Sopenharmony_ci	mutex_lock(&core->lock);
2298c2ecf20Sopenharmony_ci	list_del_init(&inst->list);
2308c2ecf20Sopenharmony_ci	if (atomic_dec_and_test(&core->insts_count))
2318c2ecf20Sopenharmony_ci		wake_up_var(&core->insts_count);
2328c2ecf20Sopenharmony_ci	mutex_unlock(&core->lock);
2338c2ecf20Sopenharmony_ci}
2348c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_destroy);
2358c2ecf20Sopenharmony_ci
2368c2ecf20Sopenharmony_ciint hfi_session_deinit(struct venus_inst *inst)
2378c2ecf20Sopenharmony_ci{
2388c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
2398c2ecf20Sopenharmony_ci	int ret;
2408c2ecf20Sopenharmony_ci
2418c2ecf20Sopenharmony_ci	if (inst->state == INST_UNINIT)
2428c2ecf20Sopenharmony_ci		return 0;
2438c2ecf20Sopenharmony_ci
2448c2ecf20Sopenharmony_ci	if (inst->state < INST_INIT)
2458c2ecf20Sopenharmony_ci		return -EINVAL;
2468c2ecf20Sopenharmony_ci
2478c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
2488c2ecf20Sopenharmony_ci
2498c2ecf20Sopenharmony_ci	ret = ops->session_end(inst);
2508c2ecf20Sopenharmony_ci	if (ret)
2518c2ecf20Sopenharmony_ci		return ret;
2528c2ecf20Sopenharmony_ci
2538c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
2548c2ecf20Sopenharmony_ci	if (ret)
2558c2ecf20Sopenharmony_ci		return ret;
2568c2ecf20Sopenharmony_ci
2578c2ecf20Sopenharmony_ci	inst->state = INST_UNINIT;
2588c2ecf20Sopenharmony_ci
2598c2ecf20Sopenharmony_ci	return 0;
2608c2ecf20Sopenharmony_ci}
2618c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_deinit);
2628c2ecf20Sopenharmony_ci
2638c2ecf20Sopenharmony_ciint hfi_session_start(struct venus_inst *inst)
2648c2ecf20Sopenharmony_ci{
2658c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
2668c2ecf20Sopenharmony_ci	int ret;
2678c2ecf20Sopenharmony_ci
2688c2ecf20Sopenharmony_ci	if (inst->state != INST_LOAD_RESOURCES)
2698c2ecf20Sopenharmony_ci		return -EINVAL;
2708c2ecf20Sopenharmony_ci
2718c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
2728c2ecf20Sopenharmony_ci
2738c2ecf20Sopenharmony_ci	ret = ops->session_start(inst);
2748c2ecf20Sopenharmony_ci	if (ret)
2758c2ecf20Sopenharmony_ci		return ret;
2768c2ecf20Sopenharmony_ci
2778c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
2788c2ecf20Sopenharmony_ci	if (ret)
2798c2ecf20Sopenharmony_ci		return ret;
2808c2ecf20Sopenharmony_ci
2818c2ecf20Sopenharmony_ci	inst->state = INST_START;
2828c2ecf20Sopenharmony_ci
2838c2ecf20Sopenharmony_ci	return 0;
2848c2ecf20Sopenharmony_ci}
2858c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_start);
2868c2ecf20Sopenharmony_ci
2878c2ecf20Sopenharmony_ciint hfi_session_stop(struct venus_inst *inst)
2888c2ecf20Sopenharmony_ci{
2898c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
2908c2ecf20Sopenharmony_ci	int ret;
2918c2ecf20Sopenharmony_ci
2928c2ecf20Sopenharmony_ci	if (inst->state != INST_START)
2938c2ecf20Sopenharmony_ci		return -EINVAL;
2948c2ecf20Sopenharmony_ci
2958c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
2968c2ecf20Sopenharmony_ci
2978c2ecf20Sopenharmony_ci	ret = ops->session_stop(inst);
2988c2ecf20Sopenharmony_ci	if (ret)
2998c2ecf20Sopenharmony_ci		return ret;
3008c2ecf20Sopenharmony_ci
3018c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
3028c2ecf20Sopenharmony_ci	if (ret)
3038c2ecf20Sopenharmony_ci		return ret;
3048c2ecf20Sopenharmony_ci
3058c2ecf20Sopenharmony_ci	inst->state = INST_STOP;
3068c2ecf20Sopenharmony_ci
3078c2ecf20Sopenharmony_ci	return 0;
3088c2ecf20Sopenharmony_ci}
3098c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_stop);
3108c2ecf20Sopenharmony_ci
3118c2ecf20Sopenharmony_ciint hfi_session_continue(struct venus_inst *inst)
3128c2ecf20Sopenharmony_ci{
3138c2ecf20Sopenharmony_ci	struct venus_core *core = inst->core;
3148c2ecf20Sopenharmony_ci
3158c2ecf20Sopenharmony_ci	if (core->res->hfi_version == HFI_VERSION_1XX)
3168c2ecf20Sopenharmony_ci		return 0;
3178c2ecf20Sopenharmony_ci
3188c2ecf20Sopenharmony_ci	return core->ops->session_continue(inst);
3198c2ecf20Sopenharmony_ci}
3208c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_continue);
3218c2ecf20Sopenharmony_ci
3228c2ecf20Sopenharmony_ciint hfi_session_abort(struct venus_inst *inst)
3238c2ecf20Sopenharmony_ci{
3248c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
3258c2ecf20Sopenharmony_ci	int ret;
3268c2ecf20Sopenharmony_ci
3278c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
3288c2ecf20Sopenharmony_ci
3298c2ecf20Sopenharmony_ci	ret = ops->session_abort(inst);
3308c2ecf20Sopenharmony_ci	if (ret)
3318c2ecf20Sopenharmony_ci		return ret;
3328c2ecf20Sopenharmony_ci
3338c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
3348c2ecf20Sopenharmony_ci	if (ret)
3358c2ecf20Sopenharmony_ci		return ret;
3368c2ecf20Sopenharmony_ci
3378c2ecf20Sopenharmony_ci	return 0;
3388c2ecf20Sopenharmony_ci}
3398c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_abort);
3408c2ecf20Sopenharmony_ci
3418c2ecf20Sopenharmony_ciint hfi_session_load_res(struct venus_inst *inst)
3428c2ecf20Sopenharmony_ci{
3438c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
3448c2ecf20Sopenharmony_ci	int ret;
3458c2ecf20Sopenharmony_ci
3468c2ecf20Sopenharmony_ci	if (inst->state != INST_INIT)
3478c2ecf20Sopenharmony_ci		return -EINVAL;
3488c2ecf20Sopenharmony_ci
3498c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
3508c2ecf20Sopenharmony_ci
3518c2ecf20Sopenharmony_ci	ret = ops->session_load_res(inst);
3528c2ecf20Sopenharmony_ci	if (ret)
3538c2ecf20Sopenharmony_ci		return ret;
3548c2ecf20Sopenharmony_ci
3558c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
3568c2ecf20Sopenharmony_ci	if (ret)
3578c2ecf20Sopenharmony_ci		return ret;
3588c2ecf20Sopenharmony_ci
3598c2ecf20Sopenharmony_ci	inst->state = INST_LOAD_RESOURCES;
3608c2ecf20Sopenharmony_ci
3618c2ecf20Sopenharmony_ci	return 0;
3628c2ecf20Sopenharmony_ci}
3638c2ecf20Sopenharmony_ci
3648c2ecf20Sopenharmony_ciint hfi_session_unload_res(struct venus_inst *inst)
3658c2ecf20Sopenharmony_ci{
3668c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
3678c2ecf20Sopenharmony_ci	int ret;
3688c2ecf20Sopenharmony_ci
3698c2ecf20Sopenharmony_ci	if (inst->state != INST_STOP)
3708c2ecf20Sopenharmony_ci		return -EINVAL;
3718c2ecf20Sopenharmony_ci
3728c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
3738c2ecf20Sopenharmony_ci
3748c2ecf20Sopenharmony_ci	ret = ops->session_release_res(inst);
3758c2ecf20Sopenharmony_ci	if (ret)
3768c2ecf20Sopenharmony_ci		return ret;
3778c2ecf20Sopenharmony_ci
3788c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
3798c2ecf20Sopenharmony_ci	if (ret)
3808c2ecf20Sopenharmony_ci		return ret;
3818c2ecf20Sopenharmony_ci
3828c2ecf20Sopenharmony_ci	inst->state = INST_RELEASE_RESOURCES;
3838c2ecf20Sopenharmony_ci
3848c2ecf20Sopenharmony_ci	return 0;
3858c2ecf20Sopenharmony_ci}
3868c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_unload_res);
3878c2ecf20Sopenharmony_ci
3888c2ecf20Sopenharmony_ciint hfi_session_flush(struct venus_inst *inst, u32 type, bool block)
3898c2ecf20Sopenharmony_ci{
3908c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
3918c2ecf20Sopenharmony_ci	int ret;
3928c2ecf20Sopenharmony_ci
3938c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
3948c2ecf20Sopenharmony_ci
3958c2ecf20Sopenharmony_ci	ret = ops->session_flush(inst, type);
3968c2ecf20Sopenharmony_ci	if (ret)
3978c2ecf20Sopenharmony_ci		return ret;
3988c2ecf20Sopenharmony_ci
3998c2ecf20Sopenharmony_ci	if (block) {
4008c2ecf20Sopenharmony_ci		ret = wait_session_msg(inst);
4018c2ecf20Sopenharmony_ci		if (ret)
4028c2ecf20Sopenharmony_ci			return ret;
4038c2ecf20Sopenharmony_ci	}
4048c2ecf20Sopenharmony_ci
4058c2ecf20Sopenharmony_ci	return 0;
4068c2ecf20Sopenharmony_ci}
4078c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_flush);
4088c2ecf20Sopenharmony_ci
4098c2ecf20Sopenharmony_ciint hfi_session_set_buffers(struct venus_inst *inst, struct hfi_buffer_desc *bd)
4108c2ecf20Sopenharmony_ci{
4118c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
4128c2ecf20Sopenharmony_ci
4138c2ecf20Sopenharmony_ci	return ops->session_set_buffers(inst, bd);
4148c2ecf20Sopenharmony_ci}
4158c2ecf20Sopenharmony_ci
4168c2ecf20Sopenharmony_ciint hfi_session_unset_buffers(struct venus_inst *inst,
4178c2ecf20Sopenharmony_ci			      struct hfi_buffer_desc *bd)
4188c2ecf20Sopenharmony_ci{
4198c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
4208c2ecf20Sopenharmony_ci	int ret;
4218c2ecf20Sopenharmony_ci
4228c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
4238c2ecf20Sopenharmony_ci
4248c2ecf20Sopenharmony_ci	ret = ops->session_unset_buffers(inst, bd);
4258c2ecf20Sopenharmony_ci	if (ret)
4268c2ecf20Sopenharmony_ci		return ret;
4278c2ecf20Sopenharmony_ci
4288c2ecf20Sopenharmony_ci	if (!bd->response_required)
4298c2ecf20Sopenharmony_ci		return 0;
4308c2ecf20Sopenharmony_ci
4318c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
4328c2ecf20Sopenharmony_ci	if (ret)
4338c2ecf20Sopenharmony_ci		return ret;
4348c2ecf20Sopenharmony_ci
4358c2ecf20Sopenharmony_ci	return 0;
4368c2ecf20Sopenharmony_ci}
4378c2ecf20Sopenharmony_ci
4388c2ecf20Sopenharmony_ciint hfi_session_get_property(struct venus_inst *inst, u32 ptype,
4398c2ecf20Sopenharmony_ci			     union hfi_get_property *hprop)
4408c2ecf20Sopenharmony_ci{
4418c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
4428c2ecf20Sopenharmony_ci	int ret;
4438c2ecf20Sopenharmony_ci
4448c2ecf20Sopenharmony_ci	if (inst->state < INST_INIT || inst->state >= INST_STOP)
4458c2ecf20Sopenharmony_ci		return -EINVAL;
4468c2ecf20Sopenharmony_ci
4478c2ecf20Sopenharmony_ci	reinit_completion(&inst->done);
4488c2ecf20Sopenharmony_ci
4498c2ecf20Sopenharmony_ci	ret = ops->session_get_property(inst, ptype);
4508c2ecf20Sopenharmony_ci	if (ret)
4518c2ecf20Sopenharmony_ci		return ret;
4528c2ecf20Sopenharmony_ci
4538c2ecf20Sopenharmony_ci	ret = wait_session_msg(inst);
4548c2ecf20Sopenharmony_ci	if (ret)
4558c2ecf20Sopenharmony_ci		return ret;
4568c2ecf20Sopenharmony_ci
4578c2ecf20Sopenharmony_ci	*hprop = inst->hprop;
4588c2ecf20Sopenharmony_ci
4598c2ecf20Sopenharmony_ci	return 0;
4608c2ecf20Sopenharmony_ci}
4618c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_get_property);
4628c2ecf20Sopenharmony_ci
4638c2ecf20Sopenharmony_ciint hfi_session_set_property(struct venus_inst *inst, u32 ptype, void *pdata)
4648c2ecf20Sopenharmony_ci{
4658c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
4668c2ecf20Sopenharmony_ci
4678c2ecf20Sopenharmony_ci	if (inst->state < INST_INIT || inst->state >= INST_STOP)
4688c2ecf20Sopenharmony_ci		return -EINVAL;
4698c2ecf20Sopenharmony_ci
4708c2ecf20Sopenharmony_ci	return ops->session_set_property(inst, ptype, pdata);
4718c2ecf20Sopenharmony_ci}
4728c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_set_property);
4738c2ecf20Sopenharmony_ci
4748c2ecf20Sopenharmony_ciint hfi_session_process_buf(struct venus_inst *inst, struct hfi_frame_data *fd)
4758c2ecf20Sopenharmony_ci{
4768c2ecf20Sopenharmony_ci	const struct hfi_ops *ops = inst->core->ops;
4778c2ecf20Sopenharmony_ci
4788c2ecf20Sopenharmony_ci	if (fd->buffer_type == HFI_BUFFER_INPUT)
4798c2ecf20Sopenharmony_ci		return ops->session_etb(inst, fd);
4808c2ecf20Sopenharmony_ci	else if (fd->buffer_type == HFI_BUFFER_OUTPUT ||
4818c2ecf20Sopenharmony_ci		 fd->buffer_type == HFI_BUFFER_OUTPUT2)
4828c2ecf20Sopenharmony_ci		return ops->session_ftb(inst, fd);
4838c2ecf20Sopenharmony_ci
4848c2ecf20Sopenharmony_ci	return -EINVAL;
4858c2ecf20Sopenharmony_ci}
4868c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(hfi_session_process_buf);
4878c2ecf20Sopenharmony_ci
4888c2ecf20Sopenharmony_ciirqreturn_t hfi_isr_thread(int irq, void *dev_id)
4898c2ecf20Sopenharmony_ci{
4908c2ecf20Sopenharmony_ci	struct venus_core *core = dev_id;
4918c2ecf20Sopenharmony_ci
4928c2ecf20Sopenharmony_ci	return core->ops->isr_thread(core);
4938c2ecf20Sopenharmony_ci}
4948c2ecf20Sopenharmony_ci
4958c2ecf20Sopenharmony_ciirqreturn_t hfi_isr(int irq, void *dev)
4968c2ecf20Sopenharmony_ci{
4978c2ecf20Sopenharmony_ci	struct venus_core *core = dev;
4988c2ecf20Sopenharmony_ci
4998c2ecf20Sopenharmony_ci	return core->ops->isr(core);
5008c2ecf20Sopenharmony_ci}
5018c2ecf20Sopenharmony_ci
5028c2ecf20Sopenharmony_ciint hfi_create(struct venus_core *core, const struct hfi_core_ops *ops)
5038c2ecf20Sopenharmony_ci{
5048c2ecf20Sopenharmony_ci	int ret;
5058c2ecf20Sopenharmony_ci
5068c2ecf20Sopenharmony_ci	if (!ops)
5078c2ecf20Sopenharmony_ci		return -EINVAL;
5088c2ecf20Sopenharmony_ci
5098c2ecf20Sopenharmony_ci	atomic_set(&core->insts_count, 0);
5108c2ecf20Sopenharmony_ci	core->core_ops = ops;
5118c2ecf20Sopenharmony_ci	core->state = CORE_UNINIT;
5128c2ecf20Sopenharmony_ci	init_completion(&core->done);
5138c2ecf20Sopenharmony_ci	pkt_set_version(core->res->hfi_version);
5148c2ecf20Sopenharmony_ci	ret = venus_hfi_create(core);
5158c2ecf20Sopenharmony_ci
5168c2ecf20Sopenharmony_ci	return ret;
5178c2ecf20Sopenharmony_ci}
5188c2ecf20Sopenharmony_ci
5198c2ecf20Sopenharmony_civoid hfi_destroy(struct venus_core *core)
5208c2ecf20Sopenharmony_ci{
5218c2ecf20Sopenharmony_ci	venus_hfi_destroy(core);
5228c2ecf20Sopenharmony_ci}
5238c2ecf20Sopenharmony_ci
5248c2ecf20Sopenharmony_civoid hfi_reinit(struct venus_core *core)
5258c2ecf20Sopenharmony_ci{
5268c2ecf20Sopenharmony_ci	venus_hfi_queues_reinit(core);
5278c2ecf20Sopenharmony_ci}
528