| /* |
| * Copyright © 2015 Canonical Ltd. (Maarten Lankhorst) |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining a |
| * copy of this software and associated documentation files (the "Software"), |
| * to deal in the Software without restriction, including without limitation |
| * the rights to use, copy, modify, merge, publish, distribute, sublicense, |
| * and/or sell copies of the Software, and to permit persons to whom the |
| * Software is furnished to do so, subject to the following conditions: |
| * |
| * The above copyright notice and this permission notice shall be included in |
| * all copies or substantial portions of the Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR |
| * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, |
| * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR |
| * OTHER DEALINGS IN THE SOFTWARE. |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| # include "config.h" |
| #endif |
| |
| #include <sys/ioctl.h> |
| #include <dlfcn.h> |
| #include <fcntl.h> |
| #include <stdio.h> |
| #include <unistd.h> |
| #include <errno.h> |
| #include <pthread.h> |
| |
| #include "xf86drm.h" |
| #include "nouveau.h" |
| |
| static typeof(ioctl) *old_ioctl; |
| static int failed; |
| |
| static int import_fd; |
| |
| int ioctl(int fd, unsigned long request, ...) |
| { |
| va_list va; |
| int ret; |
| void *arg; |
| |
| va_start(va, request); |
| arg = va_arg(va, void *); |
| ret = old_ioctl(fd, request, arg); |
| va_end(va); |
| |
| if (ret < 0 && request == DRM_IOCTL_GEM_CLOSE && errno == EINVAL) |
| failed = 1; |
| |
| return ret; |
| } |
| |
| static void * |
| openclose(void *dev) |
| { |
| struct nouveau_device *nvdev = dev; |
| struct nouveau_bo *bo = NULL; |
| int i; |
| |
| for (i = 0; i < 100000; ++i) { |
| if (!nouveau_bo_prime_handle_ref(nvdev, import_fd, &bo)) |
| nouveau_bo_ref(NULL, &bo); |
| } |
| return NULL; |
| } |
| |
| int main(int argc, char *argv[]) |
| { |
| drmVersionPtr version; |
| const char *device = NULL; |
| int err, fd, fd2; |
| struct nouveau_device *nvdev, *nvdev2; |
| struct nouveau_bo *bo; |
| pthread_t t1, t2; |
| |
| old_ioctl = dlsym(RTLD_NEXT, "ioctl"); |
| |
| if (argc < 2) { |
| fd = drmOpenWithType("nouveau", NULL, DRM_NODE_RENDER); |
| if (fd >= 0) |
| fd2 = drmOpenWithType("nouveau", NULL, DRM_NODE_RENDER); |
| } else { |
| device = argv[1]; |
| |
| fd = open(device, O_RDWR); |
| if (fd >= 0) |
| fd2 = open(device, O_RDWR); |
| else |
| fd2 = fd = -errno; |
| } |
| |
| if (fd < 0) { |
| fprintf(stderr, "Opening nouveau render node failed with %i\n", fd); |
| return device ? -fd : 77; |
| } |
| |
| if (fd2 < 0) { |
| fprintf(stderr, "Opening second nouveau render node failed with %i\n", -errno); |
| return errno; |
| } |
| |
| version = drmGetVersion(fd); |
| if (version) { |
| printf("Version: %d.%d.%d\n", version->version_major, |
| version->version_minor, version->version_patchlevel); |
| printf(" Name: %s\n", version->name); |
| printf(" Date: %s\n", version->date); |
| printf(" Description: %s\n", version->desc); |
| |
| drmFreeVersion(version); |
| } |
| |
| err = nouveau_device_wrap(fd, 0, &nvdev); |
| if (!err) |
| err = nouveau_device_wrap(fd2, 0, &nvdev2); |
| if (err < 0) |
| return 1; |
| |
| err = nouveau_bo_new(nvdev2, NOUVEAU_BO_GART, 0, 4096, NULL, &bo); |
| if (!err) |
| err = nouveau_bo_set_prime(bo, &import_fd); |
| |
| if (!err) { |
| pthread_create(&t1, NULL, openclose, nvdev); |
| pthread_create(&t2, NULL, openclose, nvdev); |
| } |
| |
| pthread_join(t1, NULL); |
| pthread_join(t2, NULL); |
| |
| close(import_fd); |
| nouveau_bo_ref(NULL, &bo); |
| |
| nouveau_device_del(&nvdev2); |
| nouveau_device_del(&nvdev); |
| if (device) { |
| close(fd2); |
| close(fd); |
| } else { |
| drmClose(fd2); |
| drmClose(fd); |
| } |
| |
| if (failed) |
| fprintf(stderr, "DRM_IOCTL_GEM_CLOSE failed with EINVAL,\n" |
| "race in opening/closing bo is likely.\n"); |
| |
| return failed; |
| } |