Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
torvalds
GitHub Repository: torvalds/linux
Path: blob/master/tools/testing/selftests/dmabuf-heaps/dmabuf-heap.c
26285 views
1
// SPDX-License-Identifier: GPL-2.0
2
3
#include <dirent.h>
4
#include <errno.h>
5
#include <fcntl.h>
6
#include <stdio.h>
7
#include <stdlib.h>
8
#include <stdint.h>
9
#include <string.h>
10
#include <unistd.h>
11
#include <sys/ioctl.h>
12
#include <sys/mman.h>
13
#include <sys/types.h>
14
15
#include <linux/dma-buf.h>
16
#include <linux/dma-heap.h>
17
#include <drm/drm.h>
18
#include "../kselftest.h"
19
20
#define DEVPATH "/dev/dma_heap"
21
22
static int check_vgem(int fd)
23
{
24
drm_version_t version = { 0 };
25
char name[5];
26
int ret;
27
28
version.name_len = 4;
29
version.name = name;
30
31
ret = ioctl(fd, DRM_IOCTL_VERSION, &version);
32
if (ret || version.name_len != 4)
33
return 0;
34
35
name[4] = '\0';
36
37
return !strcmp(name, "vgem");
38
}
39
40
static int open_vgem(void)
41
{
42
int i, fd;
43
const char *drmstr = "/dev/dri/card";
44
45
fd = -1;
46
for (i = 0; i < 16; i++) {
47
char name[80];
48
49
snprintf(name, 80, "%s%u", drmstr, i);
50
51
fd = open(name, O_RDWR);
52
if (fd < 0)
53
continue;
54
55
if (!check_vgem(fd)) {
56
close(fd);
57
fd = -1;
58
continue;
59
} else {
60
break;
61
}
62
}
63
return fd;
64
}
65
66
static int import_vgem_fd(int vgem_fd, int dma_buf_fd, uint32_t *handle)
67
{
68
struct drm_prime_handle import_handle = {
69
.fd = dma_buf_fd,
70
.flags = 0,
71
.handle = 0,
72
};
73
int ret;
74
75
ret = ioctl(vgem_fd, DRM_IOCTL_PRIME_FD_TO_HANDLE, &import_handle);
76
if (ret == 0)
77
*handle = import_handle.handle;
78
return ret;
79
}
80
81
static void close_handle(int vgem_fd, uint32_t handle)
82
{
83
struct drm_gem_close close = {
84
.handle = handle,
85
};
86
87
ioctl(vgem_fd, DRM_IOCTL_GEM_CLOSE, &close);
88
}
89
90
static int dmabuf_heap_open(char *name)
91
{
92
int ret, fd;
93
char buf[256];
94
95
ret = snprintf(buf, 256, "%s/%s", DEVPATH, name);
96
if (ret < 0)
97
ksft_exit_fail_msg("snprintf failed! %d\n", ret);
98
99
fd = open(buf, O_RDWR);
100
if (fd < 0)
101
ksft_exit_fail_msg("open %s failed: %s\n", buf, strerror(errno));
102
103
return fd;
104
}
105
106
static int dmabuf_heap_alloc_fdflags(int fd, size_t len, unsigned int fd_flags,
107
unsigned int heap_flags, int *dmabuf_fd)
108
{
109
struct dma_heap_allocation_data data = {
110
.len = len,
111
.fd = 0,
112
.fd_flags = fd_flags,
113
.heap_flags = heap_flags,
114
};
115
int ret;
116
117
if (!dmabuf_fd)
118
return -EINVAL;
119
120
ret = ioctl(fd, DMA_HEAP_IOCTL_ALLOC, &data);
121
if (ret < 0)
122
return ret;
123
*dmabuf_fd = (int)data.fd;
124
return ret;
125
}
126
127
static int dmabuf_heap_alloc(int fd, size_t len, unsigned int flags,
128
int *dmabuf_fd)
129
{
130
return dmabuf_heap_alloc_fdflags(fd, len, O_RDWR | O_CLOEXEC, flags,
131
dmabuf_fd);
132
}
133
134
static int dmabuf_sync(int fd, int start_stop)
135
{
136
struct dma_buf_sync sync = {
137
.flags = start_stop | DMA_BUF_SYNC_RW,
138
};
139
140
return ioctl(fd, DMA_BUF_IOCTL_SYNC, &sync);
141
}
142
143
#define ONE_MEG (1024 * 1024)
144
145
static void test_alloc_and_import(char *heap_name)
146
{
147
int heap_fd = -1, dmabuf_fd = -1, importer_fd = -1;
148
uint32_t handle = 0;
149
void *p = NULL;
150
int ret;
151
152
heap_fd = dmabuf_heap_open(heap_name);
153
154
ksft_print_msg("Testing allocation and importing:\n");
155
ret = dmabuf_heap_alloc(heap_fd, ONE_MEG, 0, &dmabuf_fd);
156
if (ret) {
157
ksft_test_result_fail("FAIL (Allocation Failed!) %d\n", ret);
158
return;
159
}
160
161
/* mmap and write a simple pattern */
162
p = mmap(NULL, ONE_MEG, PROT_READ | PROT_WRITE, MAP_SHARED, dmabuf_fd, 0);
163
if (p == MAP_FAILED) {
164
ksft_test_result_fail("FAIL (mmap() failed): %s\n", strerror(errno));
165
goto close_and_return;
166
}
167
168
dmabuf_sync(dmabuf_fd, DMA_BUF_SYNC_START);
169
memset(p, 1, ONE_MEG / 2);
170
memset((char *)p + ONE_MEG / 2, 0, ONE_MEG / 2);
171
dmabuf_sync(dmabuf_fd, DMA_BUF_SYNC_END);
172
173
importer_fd = open_vgem();
174
if (importer_fd < 0) {
175
ksft_test_result_skip("Could not open vgem %d\n", importer_fd);
176
} else {
177
ret = import_vgem_fd(importer_fd, dmabuf_fd, &handle);
178
ksft_test_result(ret >= 0, "Import buffer %d\n", ret);
179
}
180
181
ret = dmabuf_sync(dmabuf_fd, DMA_BUF_SYNC_START);
182
if (ret < 0) {
183
ksft_print_msg("FAIL (DMA_BUF_SYNC_START failed!) %d\n", ret);
184
goto out;
185
}
186
187
memset(p, 0xff, ONE_MEG);
188
ret = dmabuf_sync(dmabuf_fd, DMA_BUF_SYNC_END);
189
if (ret < 0) {
190
ksft_print_msg("FAIL (DMA_BUF_SYNC_END failed!) %d\n", ret);
191
goto out;
192
}
193
194
close_handle(importer_fd, handle);
195
ksft_test_result_pass("%s dmabuf sync succeeded\n", __func__);
196
return;
197
198
out:
199
ksft_test_result_fail("%s dmabuf sync failed\n", __func__);
200
munmap(p, ONE_MEG);
201
close(importer_fd);
202
203
close_and_return:
204
close(dmabuf_fd);
205
close(heap_fd);
206
}
207
208
static void test_alloc_zeroed(char *heap_name, size_t size)
209
{
210
int heap_fd = -1, dmabuf_fd[32];
211
int i, j, k, ret;
212
void *p = NULL;
213
char *c;
214
215
ksft_print_msg("Testing alloced %ldk buffers are zeroed:\n", size / 1024);
216
heap_fd = dmabuf_heap_open(heap_name);
217
218
/* Allocate and fill a bunch of buffers */
219
for (i = 0; i < 32; i++) {
220
ret = dmabuf_heap_alloc(heap_fd, size, 0, &dmabuf_fd[i]);
221
if (ret) {
222
ksft_test_result_fail("FAIL (Allocation (%i) failed) %d\n", i, ret);
223
goto close_and_return;
224
}
225
226
/* mmap and fill with simple pattern */
227
p = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, dmabuf_fd[i], 0);
228
if (p == MAP_FAILED) {
229
ksft_test_result_fail("FAIL (mmap() failed!): %s\n", strerror(errno));
230
goto close_and_return;
231
}
232
233
dmabuf_sync(dmabuf_fd[i], DMA_BUF_SYNC_START);
234
memset(p, 0xff, size);
235
dmabuf_sync(dmabuf_fd[i], DMA_BUF_SYNC_END);
236
munmap(p, size);
237
}
238
/* close them all */
239
for (i = 0; i < 32; i++)
240
close(dmabuf_fd[i]);
241
ksft_test_result_pass("Allocate and fill a bunch of buffers\n");
242
243
/* Allocate and validate all buffers are zeroed */
244
for (i = 0; i < 32; i++) {
245
ret = dmabuf_heap_alloc(heap_fd, size, 0, &dmabuf_fd[i]);
246
if (ret < 0) {
247
ksft_test_result_fail("FAIL (Allocation (%i) failed) %d\n", i, ret);
248
goto close_and_return;
249
}
250
251
/* mmap and validate everything is zero */
252
p = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, dmabuf_fd[i], 0);
253
if (p == MAP_FAILED) {
254
ksft_test_result_fail("FAIL (mmap() failed!): %s\n", strerror(errno));
255
goto close_and_return;
256
}
257
258
dmabuf_sync(dmabuf_fd[i], DMA_BUF_SYNC_START);
259
c = (char *)p;
260
for (j = 0; j < size; j++) {
261
if (c[j] != 0) {
262
ksft_print_msg("FAIL (Allocated buffer not zeroed @ %i)\n", j);
263
dmabuf_sync(dmabuf_fd[i], DMA_BUF_SYNC_END);
264
munmap(p, size);
265
goto out;
266
}
267
}
268
dmabuf_sync(dmabuf_fd[i], DMA_BUF_SYNC_END);
269
munmap(p, size);
270
}
271
272
out:
273
ksft_test_result(i == 32, "Allocate and validate all buffers are zeroed\n");
274
275
close_and_return:
276
/* close them all */
277
for (k = 0; k < i; k++)
278
close(dmabuf_fd[k]);
279
280
close(heap_fd);
281
return;
282
}
283
284
/* Test the ioctl version compatibility w/ a smaller structure then expected */
285
static int dmabuf_heap_alloc_older(int fd, size_t len, unsigned int flags,
286
int *dmabuf_fd)
287
{
288
int ret;
289
unsigned int older_alloc_ioctl;
290
struct dma_heap_allocation_data_smaller {
291
__u64 len;
292
__u32 fd;
293
__u32 fd_flags;
294
} data = {
295
.len = len,
296
.fd = 0,
297
.fd_flags = O_RDWR | O_CLOEXEC,
298
};
299
300
older_alloc_ioctl = _IOWR(DMA_HEAP_IOC_MAGIC, 0x0,
301
struct dma_heap_allocation_data_smaller);
302
if (!dmabuf_fd)
303
return -EINVAL;
304
305
ret = ioctl(fd, older_alloc_ioctl, &data);
306
if (ret < 0)
307
return ret;
308
*dmabuf_fd = (int)data.fd;
309
return ret;
310
}
311
312
/* Test the ioctl version compatibility w/ a larger structure then expected */
313
static int dmabuf_heap_alloc_newer(int fd, size_t len, unsigned int flags,
314
int *dmabuf_fd)
315
{
316
int ret;
317
unsigned int newer_alloc_ioctl;
318
struct dma_heap_allocation_data_bigger {
319
__u64 len;
320
__u32 fd;
321
__u32 fd_flags;
322
__u64 heap_flags;
323
__u64 garbage1;
324
__u64 garbage2;
325
__u64 garbage3;
326
} data = {
327
.len = len,
328
.fd = 0,
329
.fd_flags = O_RDWR | O_CLOEXEC,
330
.heap_flags = flags,
331
.garbage1 = 0xffffffff,
332
.garbage2 = 0x88888888,
333
.garbage3 = 0x11111111,
334
};
335
336
newer_alloc_ioctl = _IOWR(DMA_HEAP_IOC_MAGIC, 0x0,
337
struct dma_heap_allocation_data_bigger);
338
if (!dmabuf_fd)
339
return -EINVAL;
340
341
ret = ioctl(fd, newer_alloc_ioctl, &data);
342
if (ret < 0)
343
return ret;
344
345
*dmabuf_fd = (int)data.fd;
346
return ret;
347
}
348
349
static void test_alloc_compat(char *heap_name)
350
{
351
int ret, heap_fd = -1, dmabuf_fd = -1;
352
353
heap_fd = dmabuf_heap_open(heap_name);
354
355
ksft_print_msg("Testing (theoretical) older alloc compat:\n");
356
ret = dmabuf_heap_alloc_older(heap_fd, ONE_MEG, 0, &dmabuf_fd);
357
if (dmabuf_fd >= 0)
358
close(dmabuf_fd);
359
ksft_test_result(!ret, "dmabuf_heap_alloc_older\n");
360
361
ksft_print_msg("Testing (theoretical) newer alloc compat:\n");
362
ret = dmabuf_heap_alloc_newer(heap_fd, ONE_MEG, 0, &dmabuf_fd);
363
if (dmabuf_fd >= 0)
364
close(dmabuf_fd);
365
ksft_test_result(!ret, "dmabuf_heap_alloc_newer\n");
366
367
close(heap_fd);
368
}
369
370
static void test_alloc_errors(char *heap_name)
371
{
372
int heap_fd = -1, dmabuf_fd = -1;
373
int ret;
374
375
heap_fd = dmabuf_heap_open(heap_name);
376
377
ksft_print_msg("Testing expected error cases:\n");
378
ret = dmabuf_heap_alloc(0, ONE_MEG, 0x111111, &dmabuf_fd);
379
ksft_test_result(ret, "Error expected on invalid fd %d\n", ret);
380
381
ret = dmabuf_heap_alloc(heap_fd, ONE_MEG, 0x111111, &dmabuf_fd);
382
ksft_test_result(ret, "Error expected on invalid heap flags %d\n", ret);
383
384
ret = dmabuf_heap_alloc_fdflags(heap_fd, ONE_MEG,
385
~(O_RDWR | O_CLOEXEC), 0, &dmabuf_fd);
386
ksft_test_result(ret, "Error expected on invalid heap flags %d\n", ret);
387
388
if (dmabuf_fd >= 0)
389
close(dmabuf_fd);
390
close(heap_fd);
391
}
392
393
static int numer_of_heaps(void)
394
{
395
DIR *d = opendir(DEVPATH);
396
struct dirent *dir;
397
int heaps = 0;
398
399
while ((dir = readdir(d))) {
400
if (!strncmp(dir->d_name, ".", 2))
401
continue;
402
if (!strncmp(dir->d_name, "..", 3))
403
continue;
404
heaps++;
405
}
406
407
return heaps;
408
}
409
410
int main(void)
411
{
412
struct dirent *dir;
413
DIR *d;
414
415
ksft_print_header();
416
417
d = opendir(DEVPATH);
418
if (!d) {
419
ksft_print_msg("No %s directory?\n", DEVPATH);
420
return KSFT_SKIP;
421
}
422
423
ksft_set_plan(11 * numer_of_heaps());
424
425
while ((dir = readdir(d))) {
426
if (!strncmp(dir->d_name, ".", 2))
427
continue;
428
if (!strncmp(dir->d_name, "..", 3))
429
continue;
430
431
ksft_print_msg("Testing heap: %s\n", dir->d_name);
432
ksft_print_msg("=======================================\n");
433
test_alloc_and_import(dir->d_name);
434
test_alloc_zeroed(dir->d_name, 4 * 1024);
435
test_alloc_zeroed(dir->d_name, ONE_MEG);
436
test_alloc_compat(dir->d_name);
437
test_alloc_errors(dir->d_name);
438
}
439
closedir(d);
440
441
ksft_finished();
442
}
443
444