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