Skip to content

Commit 5d01b56

Browse files
committed
drm/panthor: Fix firmware initialization on systems with a page size > 4k
The system and GPU MMU page size might differ, which becomes a problem for FW sections that need to be mapped at explicit addresses since our PAGE_SIZE alignment might cover a VA range that's expected to be used for another section. Make sure we never map more than we need. Changes in v3: - Add R-bs Changes in v2: - Plan for per-VM page sizes so the MCU VM and user VM can have different pages sizes Fixes: 2718d91 ("drm/panthor: Add the FW logical block") Signed-off-by: Boris Brezillon <[email protected]> Reviewed-by: Steven Price <[email protected]> Reviewed-by: Liviu Dudau <[email protected]> Link: https://patchwork.freedesktop.org/patch/msgid/[email protected]
1 parent 72f7e16 commit 5d01b56

File tree

4 files changed

+24
-8
lines changed

4 files changed

+24
-8
lines changed

drivers/gpu/drm/panthor/panthor_fw.c

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -487,6 +487,7 @@ static int panthor_fw_load_section_entry(struct panthor_device *ptdev,
487487
struct panthor_fw_binary_iter *iter,
488488
u32 ehdr)
489489
{
490+
ssize_t vm_pgsz = panthor_vm_page_size(ptdev->fw->vm);
490491
struct panthor_fw_binary_section_entry_hdr hdr;
491492
struct panthor_fw_section *section;
492493
u32 section_size;
@@ -515,8 +516,7 @@ static int panthor_fw_load_section_entry(struct panthor_device *ptdev,
515516
return -EINVAL;
516517
}
517518

518-
if ((hdr.va.start & ~PAGE_MASK) != 0 ||
519-
(hdr.va.end & ~PAGE_MASK) != 0) {
519+
if (!IS_ALIGNED(hdr.va.start, vm_pgsz) || !IS_ALIGNED(hdr.va.end, vm_pgsz)) {
520520
drm_err(&ptdev->base, "Firmware corrupted, virtual addresses not page aligned: 0x%x-0x%x\n",
521521
hdr.va.start, hdr.va.end);
522522
return -EINVAL;

drivers/gpu/drm/panthor/panthor_gem.c

Lines changed: 8 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -44,8 +44,7 @@ void panthor_kernel_bo_destroy(struct panthor_kernel_bo *bo)
4444
to_panthor_bo(bo->obj)->exclusive_vm_root_gem != panthor_vm_root_gem(vm)))
4545
goto out_free_bo;
4646

47-
ret = panthor_vm_unmap_range(vm, bo->va_node.start,
48-
panthor_kernel_bo_size(bo));
47+
ret = panthor_vm_unmap_range(vm, bo->va_node.start, bo->va_node.size);
4948
if (ret)
5049
goto out_free_bo;
5150

@@ -95,10 +94,16 @@ panthor_kernel_bo_create(struct panthor_device *ptdev, struct panthor_vm *vm,
9594
}
9695

9796
bo = to_panthor_bo(&obj->base);
98-
size = obj->base.size;
9997
kbo->obj = &obj->base;
10098
bo->flags = bo_flags;
10199

100+
/* The system and GPU MMU page size might differ, which becomes a
101+
* problem for FW sections that need to be mapped at explicit address
102+
* since our PAGE_SIZE alignment might cover a VA range that's
103+
* expected to be used for another section.
104+
* Make sure we never map more than we need.
105+
*/
106+
size = ALIGN(size, panthor_vm_page_size(vm));
102107
ret = panthor_vm_alloc_va(vm, gpu_va, size, &kbo->va_node);
103108
if (ret)
104109
goto err_put_obj;

drivers/gpu/drm/panthor/panthor_mmu.c

Lines changed: 13 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -826,6 +826,14 @@ void panthor_vm_idle(struct panthor_vm *vm)
826826
mutex_unlock(&ptdev->mmu->as.slots_lock);
827827
}
828828

829+
u32 panthor_vm_page_size(struct panthor_vm *vm)
830+
{
831+
const struct io_pgtable *pgt = io_pgtable_ops_to_pgtable(vm->pgtbl_ops);
832+
u32 pg_shift = ffs(pgt->cfg.pgsize_bitmap) - 1;
833+
834+
return 1u << pg_shift;
835+
}
836+
829837
static void panthor_vm_stop(struct panthor_vm *vm)
830838
{
831839
drm_sched_stop(&vm->sched, NULL);
@@ -1025,12 +1033,13 @@ int
10251033
panthor_vm_alloc_va(struct panthor_vm *vm, u64 va, u64 size,
10261034
struct drm_mm_node *va_node)
10271035
{
1036+
ssize_t vm_pgsz = panthor_vm_page_size(vm);
10281037
int ret;
10291038

1030-
if (!size || (size & ~PAGE_MASK))
1039+
if (!size || !IS_ALIGNED(size, vm_pgsz))
10311040
return -EINVAL;
10321041

1033-
if (va != PANTHOR_VM_KERNEL_AUTO_VA && (va & ~PAGE_MASK))
1042+
if (va != PANTHOR_VM_KERNEL_AUTO_VA && !IS_ALIGNED(va, vm_pgsz))
10341043
return -EINVAL;
10351044

10361045
mutex_lock(&vm->mm_lock);
@@ -2366,11 +2375,12 @@ panthor_vm_bind_prepare_op_ctx(struct drm_file *file,
23662375
const struct drm_panthor_vm_bind_op *op,
23672376
struct panthor_vm_op_ctx *op_ctx)
23682377
{
2378+
ssize_t vm_pgsz = panthor_vm_page_size(vm);
23692379
struct drm_gem_object *gem;
23702380
int ret;
23712381

23722382
/* Aligned on page size. */
2373-
if ((op->va | op->size) & ~PAGE_MASK)
2383+
if (!IS_ALIGNED(op->va | op->size, vm_pgsz))
23742384
return -EINVAL;
23752385

23762386
switch (op->flags & DRM_PANTHOR_VM_BIND_OP_TYPE_MASK) {

drivers/gpu/drm/panthor/panthor_mmu.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ panthor_vm_get_bo_for_va(struct panthor_vm *vm, u64 va, u64 *bo_offset);
3030

3131
int panthor_vm_active(struct panthor_vm *vm);
3232
void panthor_vm_idle(struct panthor_vm *vm);
33+
u32 panthor_vm_page_size(struct panthor_vm *vm);
3334
int panthor_vm_as(struct panthor_vm *vm);
3435
int panthor_vm_flush_all(struct panthor_vm *vm);
3536

0 commit comments

Comments
 (0)