File: panthor_group.c

package info (click to toggle)
intel-gpu-tools 2.3-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 64,504 kB
  • sloc: xml: 781,458; ansic: 378,272; python: 8,407; yacc: 2,781; perl: 1,196; sh: 1,177; lex: 487; asm: 227; lisp: 35; makefile: 30
file content (123 lines) | stat: -rw-r--r-- 3,691 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
// SPDX-License-Identifier: MIT
// SPDX-FileCopyrightText: Copyright (C) 2025 Collabora Ltd.

#include <stdint.h>
#include <sys/mman.h>
#include <unistd.h>

#include "igt.h"
#include "igt_panthor.h"
#include "igt_syncobj.h"
#include "panthor_drm.h"

static size_t
issue_store_multiple(uint8_t *cs, uint64_t kernel_va, uint32_t constant)
{
	const uint8_t kernel_va_reg = 68;
	const uint8_t constant_reg = 70;
	uint64_t instrs[] = {
		/* MOV48: Load the source register ([r68; r69]) with the kernel address */
		cs_mov48(kernel_va_reg, kernel_va),
		/* MOV32: Load a known constant into r70 */
		cs_mov32(constant_reg, constant),
		/* STORE_MULTIPLE: Store the first register to the address pointed
		 * to by [r68; r69]
		 */
		cs_stm32(kernel_va_reg, constant_reg, 0),
		/* FLUSH all Wait for all cores */
		cs_wait(0xff, false),
		/* MOV32: Clear r70 to 0 */
		cs_mov32(constant_reg, 0),
		/* FLUSH_CACHE: Clean and invalidate all caches */
		cs_flush(CS_FLUSH_MODE_CLEAN_AND_INVALIDATE,
			 CS_FLUSH_MODE_CLEAN_AND_INVALIDATE,
			 CS_FLUSH_MODE_INVALIDATE,
			 0, constant_reg, 1),
		cs_wait(0xff, false),
	};

	memcpy(cs, instrs, sizeof(instrs));
	return sizeof(instrs);
}

int igt_main() {
	int fd;

	igt_fixture() {
		fd = drm_open_driver(DRIVER_PANTHOR);
	}

	igt_describe("Create and destroy a CSF group.");
	igt_subtest("group_create") {
		struct drm_panthor_vm_create vm_create = {};
		struct drm_panthor_vm_destroy vm_destroy = {};
		uint32_t group_handle;

		vm_create.flags = 0;
		do_ioctl(fd, DRM_IOCTL_PANTHOR_VM_CREATE, &vm_create);
		igt_assert_neq(vm_create.id, 0);

		group_handle = igt_panthor_group_create_simple(fd, vm_create.id, 0);
		igt_assert_neq(group_handle, 0);

		igt_panthor_group_destroy(fd, group_handle, 0);

		vm_destroy = (struct drm_panthor_vm_destroy) { .id = vm_create.id };
		do_ioctl(fd, DRM_IOCTL_PANTHOR_VM_DESTROY, &vm_destroy);
	}

	igt_describe("Submit a job to a group and wait for completion. "
		     "The job writes a known value to a buffer object that is then "
		     "mmaped and checked.");
	igt_subtest("group_submit") {
		uint32_t vm_id;
		uint32_t group_handle;
		struct panthor_bo cmd_buf_bo = {};
		struct panthor_bo result_bo = {};
		uint64_t command_stream_gpu_addr;
		uint32_t command_stream_size;
		uint64_t result_gpu_addr;
		uint32_t syncobj_handle;
		const int INITIAL_VA = 0x1000000;

		igt_panthor_vm_create(fd, &vm_id, 0);

		igt_panthor_bo_create_mapped(fd, &cmd_buf_bo, 4096, 0, 0);
		igt_panthor_vm_bind(fd, vm_id, cmd_buf_bo.handle, INITIAL_VA,
				    cmd_buf_bo.size, DRM_PANTHOR_VM_BIND_OP_TYPE_MAP, 0);
		command_stream_gpu_addr = INITIAL_VA;

		/* Create the BO to receive the result of the store. */
		igt_panthor_bo_create_mapped(fd, &result_bo, 4096, 0, 0);
		/* Also bind the result BO. */
		igt_panthor_vm_bind(fd, vm_id, result_bo.handle, INITIAL_VA + 4096,
				    result_bo.size, DRM_PANTHOR_VM_BIND_OP_TYPE_MAP, 0);
		result_gpu_addr = INITIAL_VA + 4096;

		command_stream_size = issue_store_multiple(cmd_buf_bo.map, result_gpu_addr, 0xdeadbeef);

		group_handle = igt_panthor_group_create_simple(fd, vm_id, 0);
		igt_assert_neq(group_handle, 0);

		syncobj_handle = syncobj_create(fd, 0);

		igt_panthor_group_submit_simple(fd, group_handle, 0, command_stream_gpu_addr, command_stream_size, syncobj_handle, 0);

		igt_assert(syncobj_wait(fd, &syncobj_handle, 1, INT64_MAX, 0, NULL));

		igt_assert_eq(*(uint32_t *)result_bo.map, 0xdeadbeef);

		syncobj_destroy(fd, syncobj_handle);

		igt_panthor_group_destroy(fd, group_handle, 0);

		igt_panthor_vm_destroy(fd, vm_id, 0);

		igt_panthor_free_bo(fd, &cmd_buf_bo);
		igt_panthor_free_bo(fd, &result_bo);
	}

	igt_fixture() {
		drm_close_driver(fd);
	}
}