Commit e6599adf authored by Hsin-Yi Wang's avatar Hsin-Yi Wang Committed by Mauro Carvalho Chehab

media: mtk-vpu: avoid unaligned access to DTCM buffer.

Previously, vpu->recv_buf and send_buf are forced cast from
void __iomem *tcm. vpu->recv_buf->share_buf is passed to
vpu_ipi_desc.handler(). It's not able to do unaligned access. Otherwise
kernel would crash due to unable to handle kernel paging request.

struct vpu_run {
	u32 signaled;
	char fw_ver[VPU_FW_VER_LEN];
	unsigned int	dec_capability;
	unsigned int	enc_capability;
	wait_queue_head_t wq;
};

fw_ver starts at 4 byte boundary. If system enables
CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS, strscpy() will do
read_word_at_a_time(), which tries to read 8-byte: *(unsigned long *)addr

vpu_init_ipi_handler() calls strscpy(), which would lead to crash.

vpu_init_ipi_handler() and several other handlers (eg.
vpu_dec_ipi_handler) only do read access to this data, so they can be
const, and we can use memcpy_fromio() to copy the buf to another non iomem
buffer then pass to handler.

Fixes: 85709cbf ("media: replace strncpy() by strscpy()")
Signed-off-by: default avatarHsin-Yi Wang <hsinyi@chromium.org>
Signed-off-by: default avatarHans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+huawei@kernel.org>
parent 6f704b2f
...@@ -15,7 +15,7 @@ static inline struct mtk_mdp_ctx *vpu_to_ctx(struct mtk_mdp_vpu *vpu) ...@@ -15,7 +15,7 @@ static inline struct mtk_mdp_ctx *vpu_to_ctx(struct mtk_mdp_vpu *vpu)
return container_of(vpu, struct mtk_mdp_ctx, vpu); return container_of(vpu, struct mtk_mdp_ctx, vpu);
} }
static void mtk_mdp_vpu_handle_init_ack(struct mdp_ipi_comm_ack *msg) static void mtk_mdp_vpu_handle_init_ack(const struct mdp_ipi_comm_ack *msg)
{ {
struct mtk_mdp_vpu *vpu = (struct mtk_mdp_vpu *) struct mtk_mdp_vpu *vpu = (struct mtk_mdp_vpu *)
(unsigned long)msg->ap_inst; (unsigned long)msg->ap_inst;
...@@ -26,10 +26,11 @@ static void mtk_mdp_vpu_handle_init_ack(struct mdp_ipi_comm_ack *msg) ...@@ -26,10 +26,11 @@ static void mtk_mdp_vpu_handle_init_ack(struct mdp_ipi_comm_ack *msg)
vpu->inst_addr = 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) static void mtk_mdp_vpu_ipi_handler(const void *data, unsigned int len,
void *priv)
{ {
unsigned int msg_id = *(unsigned int *)data; const struct mdp_ipi_comm_ack *msg = data;
struct mdp_ipi_comm_ack *msg = (struct mdp_ipi_comm_ack *)data; unsigned int msg_id = msg->msg_id;
struct mtk_mdp_vpu *vpu = (struct mtk_mdp_vpu *) struct mtk_mdp_vpu *vpu = (struct mtk_mdp_vpu *)
(unsigned long)msg->ap_inst; (unsigned long)msg->ap_inst;
struct mtk_mdp_ctx *ctx; struct mtk_mdp_ctx *ctx;
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
#include "vdec_ipi_msg.h" #include "vdec_ipi_msg.h"
#include "vdec_vpu_if.h" #include "vdec_vpu_if.h"
static void handle_init_ack_msg(struct vdec_vpu_ipi_init_ack *msg) static void handle_init_ack_msg(const struct vdec_vpu_ipi_init_ack *msg)
{ {
struct vdec_vpu_inst *vpu = (struct vdec_vpu_inst *) struct vdec_vpu_inst *vpu = (struct vdec_vpu_inst *)
(unsigned long)msg->ap_inst_addr; (unsigned long)msg->ap_inst_addr;
...@@ -34,9 +34,9 @@ static void handle_init_ack_msg(struct vdec_vpu_ipi_init_ack *msg) ...@@ -34,9 +34,9 @@ static void handle_init_ack_msg(struct vdec_vpu_ipi_init_ack *msg)
* This function runs in interrupt context and it means there's an IPI MSG * This function runs in interrupt context and it means there's an IPI MSG
* from VPU. * from VPU.
*/ */
static void vpu_dec_ipi_handler(void *data, unsigned int len, void *priv) static void vpu_dec_ipi_handler(const void *data, unsigned int len, void *priv)
{ {
struct vdec_vpu_ipi_ack *msg = data; const struct vdec_vpu_ipi_ack *msg = data;
struct vdec_vpu_inst *vpu = (struct vdec_vpu_inst *) struct vdec_vpu_inst *vpu = (struct vdec_vpu_inst *)
(unsigned long)msg->ap_inst_addr; (unsigned long)msg->ap_inst_addr;
......
...@@ -8,26 +8,26 @@ ...@@ -8,26 +8,26 @@
#include "venc_ipi_msg.h" #include "venc_ipi_msg.h"
#include "venc_vpu_if.h" #include "venc_vpu_if.h"
static void handle_enc_init_msg(struct venc_vpu_inst *vpu, void *data) static void handle_enc_init_msg(struct venc_vpu_inst *vpu, const void *data)
{ {
struct venc_vpu_ipi_msg_init *msg = data; const struct venc_vpu_ipi_msg_init *msg = data;
vpu->inst_addr = msg->vpu_inst_addr; vpu->inst_addr = msg->vpu_inst_addr;
vpu->vsi = vpu_mapping_dm_addr(vpu->dev, msg->vpu_inst_addr); vpu->vsi = vpu_mapping_dm_addr(vpu->dev, msg->vpu_inst_addr);
} }
static void handle_enc_encode_msg(struct venc_vpu_inst *vpu, void *data) static void handle_enc_encode_msg(struct venc_vpu_inst *vpu, const void *data)
{ {
struct venc_vpu_ipi_msg_enc *msg = data; const struct venc_vpu_ipi_msg_enc *msg = data;
vpu->state = msg->state; vpu->state = msg->state;
vpu->bs_size = msg->bs_size; vpu->bs_size = msg->bs_size;
vpu->is_key_frm = msg->is_key_frm; vpu->is_key_frm = msg->is_key_frm;
} }
static void vpu_enc_ipi_handler(void *data, unsigned int len, void *priv) static void vpu_enc_ipi_handler(const void *data, unsigned int len, void *priv)
{ {
struct venc_vpu_ipi_msg_common *msg = data; const struct venc_vpu_ipi_msg_common *msg = data;
struct venc_vpu_inst *vpu = struct venc_vpu_inst *vpu =
(struct venc_vpu_inst *)(unsigned long)msg->venc_inst; (struct venc_vpu_inst *)(unsigned long)msg->venc_inst;
......
...@@ -203,8 +203,8 @@ struct mtk_vpu { ...@@ -203,8 +203,8 @@ struct mtk_vpu {
struct vpu_run run; struct vpu_run run;
struct vpu_wdt wdt; struct vpu_wdt wdt;
struct vpu_ipi_desc ipi_desc[IPI_MAX]; struct vpu_ipi_desc ipi_desc[IPI_MAX];
struct share_obj *recv_buf; struct share_obj __iomem *recv_buf;
struct share_obj *send_buf; struct share_obj __iomem *send_buf;
struct device *dev; struct device *dev;
struct clk *clk; struct clk *clk;
bool fw_loaded; bool fw_loaded;
...@@ -292,7 +292,7 @@ int vpu_ipi_send(struct platform_device *pdev, ...@@ -292,7 +292,7 @@ int vpu_ipi_send(struct platform_device *pdev,
unsigned int len) unsigned int len)
{ {
struct mtk_vpu *vpu = platform_get_drvdata(pdev); struct mtk_vpu *vpu = platform_get_drvdata(pdev);
struct share_obj *send_obj = vpu->send_buf; struct share_obj __iomem *send_obj = vpu->send_buf;
unsigned long timeout; unsigned long timeout;
int ret = 0; int ret = 0;
...@@ -325,9 +325,9 @@ int vpu_ipi_send(struct platform_device *pdev, ...@@ -325,9 +325,9 @@ int vpu_ipi_send(struct platform_device *pdev,
} }
} while (vpu_cfg_readl(vpu, HOST_TO_VPU)); } while (vpu_cfg_readl(vpu, HOST_TO_VPU));
memcpy((void *)send_obj->share_buf, buf, len); memcpy_toio(send_obj->share_buf, buf, len);
send_obj->len = len; writel(len, &send_obj->len);
send_obj->id = id; writel(id, &send_obj->id);
vpu->ipi_id_ack[id] = false; vpu->ipi_id_ack[id] = false;
/* send the command to VPU */ /* send the command to VPU */
...@@ -600,10 +600,10 @@ int vpu_load_firmware(struct platform_device *pdev) ...@@ -600,10 +600,10 @@ int vpu_load_firmware(struct platform_device *pdev)
} }
EXPORT_SYMBOL_GPL(vpu_load_firmware); EXPORT_SYMBOL_GPL(vpu_load_firmware);
static void vpu_init_ipi_handler(void *data, unsigned int len, void *priv) static void vpu_init_ipi_handler(const void *data, unsigned int len, void *priv)
{ {
struct mtk_vpu *vpu = (struct mtk_vpu *)priv; struct mtk_vpu *vpu = priv;
struct vpu_run *run = (struct vpu_run *)data; const struct vpu_run *run = data;
vpu->run.signaled = run->signaled; vpu->run.signaled = run->signaled;
strscpy(vpu->run.fw_ver, run->fw_ver, sizeof(vpu->run.fw_ver)); strscpy(vpu->run.fw_ver, run->fw_ver, sizeof(vpu->run.fw_ver));
...@@ -700,19 +700,21 @@ static int vpu_alloc_ext_mem(struct mtk_vpu *vpu, u32 fw_type) ...@@ -700,19 +700,21 @@ static int vpu_alloc_ext_mem(struct mtk_vpu *vpu, u32 fw_type)
static void vpu_ipi_handler(struct mtk_vpu *vpu) static void vpu_ipi_handler(struct mtk_vpu *vpu)
{ {
struct share_obj *rcv_obj = vpu->recv_buf; struct share_obj __iomem *rcv_obj = vpu->recv_buf;
struct vpu_ipi_desc *ipi_desc = vpu->ipi_desc; struct vpu_ipi_desc *ipi_desc = vpu->ipi_desc;
unsigned char data[SHARE_BUF_SIZE];
if (rcv_obj->id < IPI_MAX && ipi_desc[rcv_obj->id].handler) { s32 id = readl(&rcv_obj->id);
ipi_desc[rcv_obj->id].handler(rcv_obj->share_buf,
rcv_obj->len, memcpy_fromio(data, rcv_obj->share_buf, sizeof(data));
ipi_desc[rcv_obj->id].priv); if (id < IPI_MAX && ipi_desc[id].handler) {
if (rcv_obj->id > IPI_VPU_INIT) { ipi_desc[id].handler(data, readl(&rcv_obj->len),
vpu->ipi_id_ack[rcv_obj->id] = true; ipi_desc[id].priv);
if (id > IPI_VPU_INIT) {
vpu->ipi_id_ack[id] = true;
wake_up(&vpu->ack_wq); wake_up(&vpu->ack_wq);
} }
} else { } else {
dev_err(vpu->dev, "No such ipi id = %d\n", rcv_obj->id); dev_err(vpu->dev, "No such ipi id = %d\n", id);
} }
} }
...@@ -722,11 +724,10 @@ static int vpu_ipi_init(struct mtk_vpu *vpu) ...@@ -722,11 +724,10 @@ static int vpu_ipi_init(struct mtk_vpu *vpu)
vpu_cfg_writel(vpu, 0x0, VPU_TO_HOST); vpu_cfg_writel(vpu, 0x0, VPU_TO_HOST);
/* shared buffer initialization */ /* shared buffer initialization */
vpu->recv_buf = (__force struct share_obj *)(vpu->reg.tcm + vpu->recv_buf = vpu->reg.tcm + VPU_DTCM_OFFSET;
VPU_DTCM_OFFSET);
vpu->send_buf = vpu->recv_buf + 1; vpu->send_buf = vpu->recv_buf + 1;
memset(vpu->recv_buf, 0, sizeof(struct share_obj)); memset_io(vpu->recv_buf, 0, sizeof(struct share_obj));
memset(vpu->send_buf, 0, sizeof(struct share_obj)); memset_io(vpu->send_buf, 0, sizeof(struct share_obj));
return 0; return 0;
} }
......
...@@ -15,7 +15,7 @@ ...@@ -15,7 +15,7 @@
* VPU interfaces with other blocks by share memory and interrupt. * VPU interfaces with other blocks by share memory and interrupt.
**/ **/
typedef void (*ipi_handler_t) (void *data, typedef void (*ipi_handler_t) (const void *data,
unsigned int len, unsigned int len,
void *priv); void *priv);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment