mirror of
https://github.com/hardkernel/linux.git
synced 2026-06-05 10:31:46 +09:00
drm/v3d: Address race-condition in MMU flush
[ Upstream commit cf1becb7f996a0a23ea2c270cf6bb0911ec3ca1a ]
We must first flush the MMU cache and then, flush the TLB, not the other
way around. Currently, we can see a race condition between the MMU cache
and the TLB when running multiple rendering processes at the same time.
This is evidenced by MMU errors triggered by the IRQ.
Fix the MMU flush order by flushing the MMU cache and then the TLB.
Also, in order to address the race condition, wait for the MMU cache flush
to finish before starting the TLB flush.
Fixes: 57692c94dc ("drm/v3d: Introduce a new DRM driver for Broadcom V3D V3.x+")
Signed-off-by: Maíra Canal <mcanal@igalia.com>
Reviewed-by: Iago Toral Quiroga <itoral@igalia.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20240923141348.2422499-2-mcanal@igalia.com
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
372e43bdc7
commit
1ec51f8404
@@ -34,32 +34,23 @@ static int v3d_mmu_flush_all(struct v3d_dev *v3d)
|
||||
{
|
||||
int ret;
|
||||
|
||||
/* Make sure that another flush isn't already running when we
|
||||
* start this one.
|
||||
*/
|
||||
ret = wait_for(!(V3D_READ(V3D_MMU_CTL) &
|
||||
V3D_MMU_CTL_TLB_CLEARING), 100);
|
||||
if (ret)
|
||||
dev_err(v3d->drm.dev, "TLB clear wait idle pre-wait failed\n");
|
||||
V3D_WRITE(V3D_MMUC_CONTROL, V3D_MMUC_CONTROL_FLUSH |
|
||||
V3D_MMUC_CONTROL_ENABLE);
|
||||
|
||||
ret = wait_for(!(V3D_READ(V3D_MMUC_CONTROL) &
|
||||
V3D_MMUC_CONTROL_FLUSHING), 100);
|
||||
if (ret) {
|
||||
dev_err(v3d->drm.dev, "MMUC flush wait idle failed\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
V3D_WRITE(V3D_MMU_CTL, V3D_READ(V3D_MMU_CTL) |
|
||||
V3D_MMU_CTL_TLB_CLEAR);
|
||||
|
||||
V3D_WRITE(V3D_MMUC_CONTROL,
|
||||
V3D_MMUC_CONTROL_FLUSH |
|
||||
V3D_MMUC_CONTROL_ENABLE);
|
||||
|
||||
ret = wait_for(!(V3D_READ(V3D_MMU_CTL) &
|
||||
V3D_MMU_CTL_TLB_CLEARING), 100);
|
||||
if (ret) {
|
||||
dev_err(v3d->drm.dev, "TLB clear wait idle failed\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = wait_for(!(V3D_READ(V3D_MMUC_CONTROL) &
|
||||
V3D_MMUC_CONTROL_FLUSHING), 100);
|
||||
if (ret)
|
||||
dev_err(v3d->drm.dev, "MMUC flush wait idle failed\n");
|
||||
dev_err(v3d->drm.dev, "MMU TLB clear wait idle failed\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user