| // SPDX-License-Identifier: GPL-2.0-only | 
 | /* | 
 |  * Copyright (c) 2015-2016 MediaTek Inc. | 
 |  * Author: Houlong Wei <houlong.wei@mediatek.com> | 
 |  *         Ming Hsiu Tsai <minghsiu.tsai@mediatek.com> | 
 |  */ | 
 |  | 
 | #include "mtk_mdp_core.h" | 
 | #include "mtk_mdp_vpu.h" | 
 | #include "mtk_vpu.h" | 
 |  | 
 |  | 
 | static inline struct mtk_mdp_ctx *vpu_to_ctx(struct mtk_mdp_vpu *vpu) | 
 | { | 
 | 	return container_of(vpu, struct mtk_mdp_ctx, vpu); | 
 | } | 
 |  | 
 | static void mtk_mdp_vpu_handle_init_ack(struct mdp_ipi_comm_ack *msg) | 
 | { | 
 | 	struct mtk_mdp_vpu *vpu = (struct mtk_mdp_vpu *) | 
 | 					(unsigned long)msg->ap_inst; | 
 |  | 
 | 	/* mapping VPU address to kernel virtual address */ | 
 | 	vpu->vsi = (struct mdp_process_vsi *) | 
 | 			vpu_mapping_dm_addr(vpu->pdev, msg->vpu_inst_addr); | 
 | 	vpu->inst_addr = msg->vpu_inst_addr; | 
 | } | 
 |  | 
 | static void mtk_mdp_vpu_ipi_handler(void *data, unsigned int len, void *priv) | 
 | { | 
 | 	unsigned int msg_id = *(unsigned int *)data; | 
 | 	struct mdp_ipi_comm_ack *msg = (struct mdp_ipi_comm_ack *)data; | 
 | 	struct mtk_mdp_vpu *vpu = (struct mtk_mdp_vpu *) | 
 | 					(unsigned long)msg->ap_inst; | 
 | 	struct mtk_mdp_ctx *ctx; | 
 |  | 
 | 	vpu->failure = msg->status; | 
 | 	if (!vpu->failure) { | 
 | 		switch (msg_id) { | 
 | 		case VPU_MDP_INIT_ACK: | 
 | 			mtk_mdp_vpu_handle_init_ack(data); | 
 | 			break; | 
 | 		case VPU_MDP_DEINIT_ACK: | 
 | 		case VPU_MDP_PROCESS_ACK: | 
 | 			break; | 
 | 		default: | 
 | 			ctx = vpu_to_ctx(vpu); | 
 | 			dev_err(&ctx->mdp_dev->pdev->dev, | 
 | 				"handle unknown ipi msg:0x%x\n", | 
 | 				msg_id); | 
 | 			break; | 
 | 		} | 
 | 	} else { | 
 | 		ctx = vpu_to_ctx(vpu); | 
 | 		mtk_mdp_dbg(0, "[%d]:msg 0x%x, failure:%d", ctx->id, | 
 | 			    msg_id, vpu->failure); | 
 | 	} | 
 | } | 
 |  | 
 | int mtk_mdp_vpu_register(struct platform_device *pdev) | 
 | { | 
 | 	struct mtk_mdp_dev *mdp = platform_get_drvdata(pdev); | 
 | 	int err; | 
 |  | 
 | 	err = vpu_ipi_register(mdp->vpu_dev, IPI_MDP, | 
 | 			       mtk_mdp_vpu_ipi_handler, "mdp_vpu", NULL); | 
 | 	if (err) | 
 | 		dev_err(&mdp->pdev->dev, | 
 | 			"vpu_ipi_registration fail status=%d\n", err); | 
 |  | 
 | 	return err; | 
 | } | 
 |  | 
 | static int mtk_mdp_vpu_send_msg(void *msg, int len, struct mtk_mdp_vpu *vpu, | 
 | 				int id) | 
 | { | 
 | 	struct mtk_mdp_ctx *ctx = vpu_to_ctx(vpu); | 
 | 	int err; | 
 |  | 
 | 	if (!vpu->pdev) { | 
 | 		mtk_mdp_dbg(1, "[%d]:vpu pdev is NULL", ctx->id); | 
 | 		return -EINVAL; | 
 | 	} | 
 |  | 
 | 	mutex_lock(&ctx->mdp_dev->vpulock); | 
 | 	err = vpu_ipi_send(vpu->pdev, (enum ipi_id)id, msg, len); | 
 | 	if (err) | 
 | 		dev_err(&ctx->mdp_dev->pdev->dev, | 
 | 			"vpu_ipi_send fail status %d\n", err); | 
 | 	mutex_unlock(&ctx->mdp_dev->vpulock); | 
 |  | 
 | 	return err; | 
 | } | 
 |  | 
 | static int mtk_mdp_vpu_send_ap_ipi(struct mtk_mdp_vpu *vpu, uint32_t msg_id) | 
 | { | 
 | 	int err; | 
 | 	struct mdp_ipi_comm msg; | 
 |  | 
 | 	msg.msg_id = msg_id; | 
 | 	msg.ipi_id = IPI_MDP; | 
 | 	msg.vpu_inst_addr = vpu->inst_addr; | 
 | 	msg.ap_inst = (unsigned long)vpu; | 
 | 	err = mtk_mdp_vpu_send_msg((void *)&msg, sizeof(msg), vpu, IPI_MDP); | 
 | 	if (!err && vpu->failure) | 
 | 		err = -EINVAL; | 
 |  | 
 | 	return err; | 
 | } | 
 |  | 
 | int mtk_mdp_vpu_init(struct mtk_mdp_vpu *vpu) | 
 | { | 
 | 	int err; | 
 | 	struct mdp_ipi_init msg; | 
 | 	struct mtk_mdp_ctx *ctx = vpu_to_ctx(vpu); | 
 |  | 
 | 	vpu->pdev = ctx->mdp_dev->vpu_dev; | 
 |  | 
 | 	msg.msg_id = AP_MDP_INIT; | 
 | 	msg.ipi_id = IPI_MDP; | 
 | 	msg.ap_inst = (unsigned long)vpu; | 
 | 	err = mtk_mdp_vpu_send_msg((void *)&msg, sizeof(msg), vpu, IPI_MDP); | 
 | 	if (!err && vpu->failure) | 
 | 		err = -EINVAL; | 
 |  | 
 | 	return err; | 
 | } | 
 |  | 
 | int mtk_mdp_vpu_deinit(struct mtk_mdp_vpu *vpu) | 
 | { | 
 | 	return mtk_mdp_vpu_send_ap_ipi(vpu, AP_MDP_DEINIT); | 
 | } | 
 |  | 
 | int mtk_mdp_vpu_process(struct mtk_mdp_vpu *vpu) | 
 | { | 
 | 	return mtk_mdp_vpu_send_ap_ipi(vpu, AP_MDP_PROCESS); | 
 | } |