xref: /aosp_15_r20/external/igt-gpu-tools/tests/i915/gem_create.c (revision d83cc019efdc2edc6c4b16e9034a3ceb8d35d77c)
1*d83cc019SAndroid Build Coastguard Worker /*
2*d83cc019SAndroid Build Coastguard Worker  * Copyright © 2015 Intel Corporation
3*d83cc019SAndroid Build Coastguard Worker  *
4*d83cc019SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
5*d83cc019SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the "Software"),
6*d83cc019SAndroid Build Coastguard Worker  * to deal in the Software without restriction, including without limitation
7*d83cc019SAndroid Build Coastguard Worker  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*d83cc019SAndroid Build Coastguard Worker  * and/or sell copies of the Software, and to permit persons to whom the
9*d83cc019SAndroid Build Coastguard Worker  * Software is furnished to do so, subject to the following conditions:
10*d83cc019SAndroid Build Coastguard Worker  *
11*d83cc019SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the next
12*d83cc019SAndroid Build Coastguard Worker  * paragraph) shall be included in all copies or substantial portions of the
13*d83cc019SAndroid Build Coastguard Worker  * Software.
14*d83cc019SAndroid Build Coastguard Worker  *
15*d83cc019SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*d83cc019SAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*d83cc019SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18*d83cc019SAndroid Build Coastguard Worker  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*d83cc019SAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*d83cc019SAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21*d83cc019SAndroid Build Coastguard Worker  * IN THE SOFTWARE.
22*d83cc019SAndroid Build Coastguard Worker  *
23*d83cc019SAndroid Build Coastguard Worker  * Authors:
24*d83cc019SAndroid Build Coastguard Worker  *    Ankitprasad Sharma <ankitprasad.r.sharma at intel.com>
25*d83cc019SAndroid Build Coastguard Worker  *
26*d83cc019SAndroid Build Coastguard Worker  */
27*d83cc019SAndroid Build Coastguard Worker 
28*d83cc019SAndroid Build Coastguard Worker /** @file gem_create.c
29*d83cc019SAndroid Build Coastguard Worker  *
30*d83cc019SAndroid Build Coastguard Worker  * This is a test for the extended and old gem_create ioctl, that
31*d83cc019SAndroid Build Coastguard Worker  * includes allocation of object from stolen memory and shmem.
32*d83cc019SAndroid Build Coastguard Worker  *
33*d83cc019SAndroid Build Coastguard Worker  * The goal is to simply ensure that basics work and invalid input
34*d83cc019SAndroid Build Coastguard Worker  * combinations are rejected.
35*d83cc019SAndroid Build Coastguard Worker  */
36*d83cc019SAndroid Build Coastguard Worker 
37*d83cc019SAndroid Build Coastguard Worker #include <stdlib.h>
38*d83cc019SAndroid Build Coastguard Worker #include <sys/ioctl.h>
39*d83cc019SAndroid Build Coastguard Worker #include <stdio.h>
40*d83cc019SAndroid Build Coastguard Worker #include <string.h>
41*d83cc019SAndroid Build Coastguard Worker #include <fcntl.h>
42*d83cc019SAndroid Build Coastguard Worker #include <inttypes.h>
43*d83cc019SAndroid Build Coastguard Worker #include <errno.h>
44*d83cc019SAndroid Build Coastguard Worker #include <sys/stat.h>
45*d83cc019SAndroid Build Coastguard Worker #include <sys/time.h>
46*d83cc019SAndroid Build Coastguard Worker #include <getopt.h>
47*d83cc019SAndroid Build Coastguard Worker #include <pthread.h>
48*d83cc019SAndroid Build Coastguard Worker #include <stdatomic.h>
49*d83cc019SAndroid Build Coastguard Worker 
50*d83cc019SAndroid Build Coastguard Worker #include <drm.h>
51*d83cc019SAndroid Build Coastguard Worker 
52*d83cc019SAndroid Build Coastguard Worker #include "ioctl_wrappers.h"
53*d83cc019SAndroid Build Coastguard Worker #include "intel_bufmgr.h"
54*d83cc019SAndroid Build Coastguard Worker #include "intel_batchbuffer.h"
55*d83cc019SAndroid Build Coastguard Worker #include "intel_io.h"
56*d83cc019SAndroid Build Coastguard Worker #include "intel_chipset.h"
57*d83cc019SAndroid Build Coastguard Worker #include "igt_aux.h"
58*d83cc019SAndroid Build Coastguard Worker #include "drmtest.h"
59*d83cc019SAndroid Build Coastguard Worker #include "drm.h"
60*d83cc019SAndroid Build Coastguard Worker #include "i915_drm.h"
61*d83cc019SAndroid Build Coastguard Worker 
62*d83cc019SAndroid Build Coastguard Worker IGT_TEST_DESCRIPTION("This is a test for the extended & old gem_create ioctl,"
63*d83cc019SAndroid Build Coastguard Worker 		     " that includes allocation of object from stolen memory"
64*d83cc019SAndroid Build Coastguard Worker 		     " and shmem.");
65*d83cc019SAndroid Build Coastguard Worker 
66*d83cc019SAndroid Build Coastguard Worker #define CLEAR(s) memset(&s, 0, sizeof(s))
67*d83cc019SAndroid Build Coastguard Worker #define PAGE_SIZE 4096
68*d83cc019SAndroid Build Coastguard Worker 
69*d83cc019SAndroid Build Coastguard Worker struct local_i915_gem_create_v2 {
70*d83cc019SAndroid Build Coastguard Worker 	uint64_t size;
71*d83cc019SAndroid Build Coastguard Worker 	uint32_t handle;
72*d83cc019SAndroid Build Coastguard Worker 	uint32_t pad;
73*d83cc019SAndroid Build Coastguard Worker #define I915_CREATE_PLACEMENT_STOLEN (1<<0)
74*d83cc019SAndroid Build Coastguard Worker 	uint32_t flags;
75*d83cc019SAndroid Build Coastguard Worker } create_v2;
76*d83cc019SAndroid Build Coastguard Worker 
77*d83cc019SAndroid Build Coastguard Worker #define LOCAL_IOCTL_I915_GEM_CREATE       DRM_IOWR(DRM_COMMAND_BASE + DRM_I915_GEM_CREATE, struct local_i915_gem_create_v2)
78*d83cc019SAndroid Build Coastguard Worker 
invalid_flag_test(int fd)79*d83cc019SAndroid Build Coastguard Worker static void invalid_flag_test(int fd)
80*d83cc019SAndroid Build Coastguard Worker {
81*d83cc019SAndroid Build Coastguard Worker 	int ret;
82*d83cc019SAndroid Build Coastguard Worker 
83*d83cc019SAndroid Build Coastguard Worker 	gem_require_stolen_support(fd);
84*d83cc019SAndroid Build Coastguard Worker 
85*d83cc019SAndroid Build Coastguard Worker 	create_v2.handle = 0;
86*d83cc019SAndroid Build Coastguard Worker 	create_v2.size = PAGE_SIZE;
87*d83cc019SAndroid Build Coastguard Worker 	create_v2.flags = ~I915_CREATE_PLACEMENT_STOLEN;
88*d83cc019SAndroid Build Coastguard Worker 	ret = drmIoctl(fd, LOCAL_IOCTL_I915_GEM_CREATE, &create_v2);
89*d83cc019SAndroid Build Coastguard Worker 
90*d83cc019SAndroid Build Coastguard Worker 	igt_assert(ret <= 0);
91*d83cc019SAndroid Build Coastguard Worker 
92*d83cc019SAndroid Build Coastguard Worker 	create_v2.flags = ~0;
93*d83cc019SAndroid Build Coastguard Worker 	ret = drmIoctl(fd, LOCAL_IOCTL_I915_GEM_CREATE, &create_v2);
94*d83cc019SAndroid Build Coastguard Worker 
95*d83cc019SAndroid Build Coastguard Worker 	igt_assert(ret <= 0);
96*d83cc019SAndroid Build Coastguard Worker }
97*d83cc019SAndroid Build Coastguard Worker 
create_ioctl(int fd,struct drm_i915_gem_create * create)98*d83cc019SAndroid Build Coastguard Worker static int create_ioctl(int fd, struct drm_i915_gem_create *create)
99*d83cc019SAndroid Build Coastguard Worker {
100*d83cc019SAndroid Build Coastguard Worker         int err = 0;
101*d83cc019SAndroid Build Coastguard Worker 
102*d83cc019SAndroid Build Coastguard Worker         if (igt_ioctl(fd, DRM_IOCTL_I915_GEM_CREATE, create)) {
103*d83cc019SAndroid Build Coastguard Worker                 err = -errno;
104*d83cc019SAndroid Build Coastguard Worker                 igt_assume(err != 0);
105*d83cc019SAndroid Build Coastguard Worker         }
106*d83cc019SAndroid Build Coastguard Worker 
107*d83cc019SAndroid Build Coastguard Worker         errno = 0;
108*d83cc019SAndroid Build Coastguard Worker         return err;
109*d83cc019SAndroid Build Coastguard Worker }
110*d83cc019SAndroid Build Coastguard Worker 
invalid_size_test(int fd)111*d83cc019SAndroid Build Coastguard Worker static void invalid_size_test(int fd)
112*d83cc019SAndroid Build Coastguard Worker {
113*d83cc019SAndroid Build Coastguard Worker 	struct drm_i915_gem_create create = {
114*d83cc019SAndroid Build Coastguard Worker 		.size = 0,
115*d83cc019SAndroid Build Coastguard Worker 	};
116*d83cc019SAndroid Build Coastguard Worker 
117*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(create_ioctl(fd, &create), -EINVAL);
118*d83cc019SAndroid Build Coastguard Worker }
119*d83cc019SAndroid Build Coastguard Worker 
120*d83cc019SAndroid Build Coastguard Worker /*
121*d83cc019SAndroid Build Coastguard Worker  * Creating an object with non-aligned size and trying to access it with an
122*d83cc019SAndroid Build Coastguard Worker  * offset, which is greater than the requested size but smaller than the
123*d83cc019SAndroid Build Coastguard Worker  * object's last page boundary. pwrite here must be successful.
124*d83cc019SAndroid Build Coastguard Worker  */
valid_nonaligned_size(int fd)125*d83cc019SAndroid Build Coastguard Worker static void valid_nonaligned_size(int fd)
126*d83cc019SAndroid Build Coastguard Worker {
127*d83cc019SAndroid Build Coastguard Worker 	struct drm_i915_gem_create create = {
128*d83cc019SAndroid Build Coastguard Worker 		.size = PAGE_SIZE / 2,
129*d83cc019SAndroid Build Coastguard Worker 	};
130*d83cc019SAndroid Build Coastguard Worker 	char buf[PAGE_SIZE];
131*d83cc019SAndroid Build Coastguard Worker 
132*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(create_ioctl(fd, &create), 0);
133*d83cc019SAndroid Build Coastguard Worker 
134*d83cc019SAndroid Build Coastguard Worker 	gem_write(fd, create.handle, PAGE_SIZE / 2, buf, PAGE_SIZE / 2);
135*d83cc019SAndroid Build Coastguard Worker 
136*d83cc019SAndroid Build Coastguard Worker 	gem_close(fd, create.handle);
137*d83cc019SAndroid Build Coastguard Worker }
138*d83cc019SAndroid Build Coastguard Worker 
139*d83cc019SAndroid Build Coastguard Worker /*
140*d83cc019SAndroid Build Coastguard Worker  * Creating an object with non-aligned size and trying to access it with an
141*d83cc019SAndroid Build Coastguard Worker  * offset, which is greater than the requested size and larger than the
142*d83cc019SAndroid Build Coastguard Worker  * object's last page boundary. pwrite here must fail.
143*d83cc019SAndroid Build Coastguard Worker  */
invalid_nonaligned_size(int fd)144*d83cc019SAndroid Build Coastguard Worker static void invalid_nonaligned_size(int fd)
145*d83cc019SAndroid Build Coastguard Worker {
146*d83cc019SAndroid Build Coastguard Worker 	struct drm_i915_gem_create create = {
147*d83cc019SAndroid Build Coastguard Worker 		.size = PAGE_SIZE / 2,
148*d83cc019SAndroid Build Coastguard Worker 	};
149*d83cc019SAndroid Build Coastguard Worker 	char buf[PAGE_SIZE];
150*d83cc019SAndroid Build Coastguard Worker 
151*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(create_ioctl(fd, &create), 0);
152*d83cc019SAndroid Build Coastguard Worker 
153*d83cc019SAndroid Build Coastguard Worker 	/* This should fail. Hence cannot use gem_write. */
154*d83cc019SAndroid Build Coastguard Worker 	igt_assert(__gem_write(fd, create.handle,
155*d83cc019SAndroid Build Coastguard Worker 			       PAGE_SIZE / 2, buf, PAGE_SIZE));
156*d83cc019SAndroid Build Coastguard Worker 
157*d83cc019SAndroid Build Coastguard Worker 	gem_close(fd, create.handle);
158*d83cc019SAndroid Build Coastguard Worker }
159*d83cc019SAndroid Build Coastguard Worker 
atomic_compare_swap_u64(_Atomic (uint64_t)* ptr,uint64_t oldval,uint64_t newval)160*d83cc019SAndroid Build Coastguard Worker static uint64_t atomic_compare_swap_u64(_Atomic(uint64_t) *ptr,
161*d83cc019SAndroid Build Coastguard Worker 					uint64_t oldval, uint64_t newval)
162*d83cc019SAndroid Build Coastguard Worker {
163*d83cc019SAndroid Build Coastguard Worker 	atomic_compare_exchange_strong(ptr, &oldval, newval);
164*d83cc019SAndroid Build Coastguard Worker 	return oldval;
165*d83cc019SAndroid Build Coastguard Worker }
166*d83cc019SAndroid Build Coastguard Worker 
get_npages(_Atomic (uint64_t)* global,uint64_t npages)167*d83cc019SAndroid Build Coastguard Worker static uint64_t get_npages(_Atomic(uint64_t) *global, uint64_t npages)
168*d83cc019SAndroid Build Coastguard Worker {
169*d83cc019SAndroid Build Coastguard Worker 	uint64_t try, old, max;
170*d83cc019SAndroid Build Coastguard Worker 
171*d83cc019SAndroid Build Coastguard Worker 	max = *global;
172*d83cc019SAndroid Build Coastguard Worker 	do {
173*d83cc019SAndroid Build Coastguard Worker 		old = max;
174*d83cc019SAndroid Build Coastguard Worker 		try = 1 + npages % (max / 2);
175*d83cc019SAndroid Build Coastguard Worker 		max -= try;
176*d83cc019SAndroid Build Coastguard Worker 	} while ((max = atomic_compare_swap_u64(global, old, max)) != old);
177*d83cc019SAndroid Build Coastguard Worker 
178*d83cc019SAndroid Build Coastguard Worker 	return try;
179*d83cc019SAndroid Build Coastguard Worker }
180*d83cc019SAndroid Build Coastguard Worker 
181*d83cc019SAndroid Build Coastguard Worker struct thread_clear {
182*d83cc019SAndroid Build Coastguard Worker 	_Atomic(uint64_t) max;
183*d83cc019SAndroid Build Coastguard Worker 	int timeout;
184*d83cc019SAndroid Build Coastguard Worker 	int i915;
185*d83cc019SAndroid Build Coastguard Worker };
186*d83cc019SAndroid Build Coastguard Worker 
thread_clear(void * data)187*d83cc019SAndroid Build Coastguard Worker static void *thread_clear(void *data)
188*d83cc019SAndroid Build Coastguard Worker {
189*d83cc019SAndroid Build Coastguard Worker 	struct thread_clear *arg = data;
190*d83cc019SAndroid Build Coastguard Worker 	unsigned long checked = 0;
191*d83cc019SAndroid Build Coastguard Worker 	int i915 = arg->i915;
192*d83cc019SAndroid Build Coastguard Worker 
193*d83cc019SAndroid Build Coastguard Worker 	igt_until_timeout(arg->timeout) {
194*d83cc019SAndroid Build Coastguard Worker 		struct drm_i915_gem_create create = {};
195*d83cc019SAndroid Build Coastguard Worker 		uint64_t npages;
196*d83cc019SAndroid Build Coastguard Worker 
197*d83cc019SAndroid Build Coastguard Worker 		npages = random();
198*d83cc019SAndroid Build Coastguard Worker 		npages <<= 32;
199*d83cc019SAndroid Build Coastguard Worker 		npages |= random();
200*d83cc019SAndroid Build Coastguard Worker 		npages = get_npages(&arg->max, npages);
201*d83cc019SAndroid Build Coastguard Worker 		create.size = npages << 12;
202*d83cc019SAndroid Build Coastguard Worker 
203*d83cc019SAndroid Build Coastguard Worker 		create_ioctl(i915, &create);
204*d83cc019SAndroid Build Coastguard Worker 		for (uint64_t page = 0; page < npages; page++) {
205*d83cc019SAndroid Build Coastguard Worker 			uint64_t x;
206*d83cc019SAndroid Build Coastguard Worker 
207*d83cc019SAndroid Build Coastguard Worker 			gem_read(i915, create.handle,
208*d83cc019SAndroid Build Coastguard Worker 				 page * 4096 + (page % (4096 - sizeof(x))),
209*d83cc019SAndroid Build Coastguard Worker 				 &x, sizeof(x));
210*d83cc019SAndroid Build Coastguard Worker 			igt_assert_eq_u64(x, 0);
211*d83cc019SAndroid Build Coastguard Worker 		}
212*d83cc019SAndroid Build Coastguard Worker 		gem_close(i915, create.handle);
213*d83cc019SAndroid Build Coastguard Worker 		checked += npages;
214*d83cc019SAndroid Build Coastguard Worker 
215*d83cc019SAndroid Build Coastguard Worker 		atomic_fetch_add(&arg->max, npages);
216*d83cc019SAndroid Build Coastguard Worker 	}
217*d83cc019SAndroid Build Coastguard Worker 
218*d83cc019SAndroid Build Coastguard Worker 	return (void *)(uintptr_t)checked;
219*d83cc019SAndroid Build Coastguard Worker }
220*d83cc019SAndroid Build Coastguard Worker 
always_clear(int i915,int timeout)221*d83cc019SAndroid Build Coastguard Worker static void always_clear(int i915, int timeout)
222*d83cc019SAndroid Build Coastguard Worker {
223*d83cc019SAndroid Build Coastguard Worker 	struct thread_clear arg = {
224*d83cc019SAndroid Build Coastguard Worker 		.i915 = i915,
225*d83cc019SAndroid Build Coastguard Worker 		.timeout = timeout,
226*d83cc019SAndroid Build Coastguard Worker 		.max = intel_get_avail_ram_mb() << (20 - 12), /* in pages */
227*d83cc019SAndroid Build Coastguard Worker 	};
228*d83cc019SAndroid Build Coastguard Worker 	const int ncpus = sysconf(_SC_NPROCESSORS_ONLN);
229*d83cc019SAndroid Build Coastguard Worker 	unsigned long checked;
230*d83cc019SAndroid Build Coastguard Worker 	pthread_t thread[ncpus];
231*d83cc019SAndroid Build Coastguard Worker 	void *result;
232*d83cc019SAndroid Build Coastguard Worker 
233*d83cc019SAndroid Build Coastguard Worker 	for (int i = 0; i < ncpus; i++)
234*d83cc019SAndroid Build Coastguard Worker 		pthread_create(&thread[i], NULL, thread_clear, &arg);
235*d83cc019SAndroid Build Coastguard Worker 
236*d83cc019SAndroid Build Coastguard Worker 	checked = 0;
237*d83cc019SAndroid Build Coastguard Worker 	for (int i = 0; i < ncpus; i++) {
238*d83cc019SAndroid Build Coastguard Worker 		pthread_join(thread[i], &result);
239*d83cc019SAndroid Build Coastguard Worker 		checked += (uintptr_t)result;
240*d83cc019SAndroid Build Coastguard Worker 	}
241*d83cc019SAndroid Build Coastguard Worker 	igt_info("Checked %'lu page allocations\n", checked);
242*d83cc019SAndroid Build Coastguard Worker }
243*d83cc019SAndroid Build Coastguard Worker 
size_update(int fd)244*d83cc019SAndroid Build Coastguard Worker static void size_update(int fd)
245*d83cc019SAndroid Build Coastguard Worker {
246*d83cc019SAndroid Build Coastguard Worker 	int size_initial_nonaligned = 15;
247*d83cc019SAndroid Build Coastguard Worker 
248*d83cc019SAndroid Build Coastguard Worker 	struct drm_i915_gem_create create = {
249*d83cc019SAndroid Build Coastguard Worker 		.size = size_initial_nonaligned,
250*d83cc019SAndroid Build Coastguard Worker 	};
251*d83cc019SAndroid Build Coastguard Worker 
252*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(create_ioctl(fd, &create), 0);
253*d83cc019SAndroid Build Coastguard Worker 	igt_assert_neq(create.size, size_initial_nonaligned);
254*d83cc019SAndroid Build Coastguard Worker }
255*d83cc019SAndroid Build Coastguard Worker 
256*d83cc019SAndroid Build Coastguard Worker igt_main
257*d83cc019SAndroid Build Coastguard Worker {
258*d83cc019SAndroid Build Coastguard Worker 	int fd = -1;
259*d83cc019SAndroid Build Coastguard Worker 
260*d83cc019SAndroid Build Coastguard Worker 	igt_skip_on_simulation();
261*d83cc019SAndroid Build Coastguard Worker 
262*d83cc019SAndroid Build Coastguard Worker 	igt_fixture {
263*d83cc019SAndroid Build Coastguard Worker 		fd = drm_open_driver(DRIVER_INTEL);
264*d83cc019SAndroid Build Coastguard Worker 	}
265*d83cc019SAndroid Build Coastguard Worker 
266*d83cc019SAndroid Build Coastguard Worker 	igt_subtest("stolen-invalid-flag")
267*d83cc019SAndroid Build Coastguard Worker 		invalid_flag_test(fd);
268*d83cc019SAndroid Build Coastguard Worker 
269*d83cc019SAndroid Build Coastguard Worker 	igt_subtest("create-invalid-size")
270*d83cc019SAndroid Build Coastguard Worker 		invalid_size_test(fd);
271*d83cc019SAndroid Build Coastguard Worker 
272*d83cc019SAndroid Build Coastguard Worker 	igt_subtest("create-valid-nonaligned")
273*d83cc019SAndroid Build Coastguard Worker 		valid_nonaligned_size(fd);
274*d83cc019SAndroid Build Coastguard Worker 
275*d83cc019SAndroid Build Coastguard Worker 	igt_subtest("create-invalid-nonaligned")
276*d83cc019SAndroid Build Coastguard Worker 		invalid_nonaligned_size(fd);
277*d83cc019SAndroid Build Coastguard Worker 
278*d83cc019SAndroid Build Coastguard Worker 	igt_subtest("create-size-update")
279*d83cc019SAndroid Build Coastguard Worker 		size_update(fd);
280*d83cc019SAndroid Build Coastguard Worker 
281*d83cc019SAndroid Build Coastguard Worker 	igt_subtest("create-clear")
282*d83cc019SAndroid Build Coastguard Worker 		always_clear(fd, 30);
283*d83cc019SAndroid Build Coastguard Worker }
284