/* * Copyright © 2015 Intel Corporation * * 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 (including the next * paragraph) 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 AUTHORS OR COPYRIGHT HOLDERS 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. */ #define _GNU_SOURCE /* for RTLD_NEXT */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "intel_aub.h" #include "intel_chipset.h" static int (*libc_close)(int fd); static int (*libc_ioctl)(int fd, unsigned long request, void *argp); static int drm_fd = -1; static FILE *file; #define DRM_MAJOR 226 enum { ADD_BO = 0, DEL_BO, EXEC, }; struct trace_add_bo { uint8_t cmd; uint32_t handle; uint64_t size; } __attribute__((packed)); struct trace_del_bo { uint8_t cmd; uint32_t handle; }__attribute__((packed)); struct trace_exec { uint8_t cmd; uint32_t object_count; uint64_t flags; }__attribute__((packed)); struct trace_exec_object { uint32_t handle; uint32_t relocation_count; uint64_t alignment; uint64_t flags; uint64_t rsvd1; uint64_t rsvd2; }__attribute__((packed)); struct trace_exec_relocation { uint32_t target_handle; uint32_t delta; uint64_t offset; uint32_t read_domains; uint32_t write_domain; }__attribute__((packed)); static void __attribute__ ((format(__printf__, 2, 3))) fail_if(int cond, const char *format, ...) { va_list args; if (!cond) return; va_start(args, format); vfprintf(stderr, format, args); va_end(args); exit(1); } static void trace_exec(int fd, const struct drm_i915_gem_execbuffer2 *execbuffer2) { const struct drm_i915_gem_exec_object2 *exec_objects = (struct drm_i915_gem_exec_object2 *)(uintptr_t)execbuffer2->buffers_ptr; { struct trace_exec t = { EXEC, execbuffer2->buffer_count, execbuffer2->flags }; fwrite(&t, sizeof(t), 1, file); } for (uint32_t i = 0; i < execbuffer2->buffer_count; i++) { const struct drm_i915_gem_exec_object2 *obj = &exec_objects[i]; const struct drm_i915_gem_relocation_entry *relocs = (struct drm_i915_gem_relocation_entry *)(uintptr_t)obj->relocs_ptr; { struct trace_exec_object t = { obj->handle, obj->relocation_count, obj->alignment, obj->flags, obj->rsvd1, obj->rsvd2 }; fwrite(&t, sizeof(t), 1, file); } for (uint32_t j = 0; j < obj->relocation_count; j++) { struct trace_exec_relocation t = { relocs[j].target_handle, relocs[j].delta, relocs[j].offset, relocs[j].read_domains, relocs[j].write_domain, }; fwrite(&t, sizeof(t), 1, file); } } fflush(file); } static void trace_add(uint32_t handle, uint64_t size) { struct trace_add_bo t = { ADD_BO, handle, size }; fwrite(&t, sizeof(t), 1, file); } static void trace_del(uint32_t handle) { struct trace_del_bo t = { DEL_BO, handle }; fwrite(&t, sizeof(t), 1, file); } int close(int fd) { if (fd == drm_fd) drm_fd = -1; return libc_close(fd); } static unsigned long size_for_fb(const struct drm_mode_fb_cmd *cmd) { unsigned long size; #ifndef ALIGN #define ALIGN(x, y) (((x) + (y) - 1) & -(y)) #endif size = ALIGN(cmd->width * cmd->bpp, 64); size *= cmd->height; return ALIGN(size, 4096); } static int is_i915(int fd) { drm_version_t version; char name[5] = ""; memset(&version, 0, sizeof(version)); version.name_len = 4; version.name = name; if (libc_ioctl(fd, DRM_IOCTL_VERSION, &version)) return 0; return strcmp(name, "i915") == 0; } int ioctl(int fd, unsigned long request, ...) { va_list args; void *argp; int ret; va_start(args, request); argp = va_arg(args, void *); va_end(args); ret = libc_ioctl(fd, request, argp); if (ret) return ret; if (_IOC_TYPE(request) != DRM_IOCTL_BASE) return 0; if (drm_fd != fd) { char filename[80]; if (!is_i915(fd)) return 0; if (file) fclose(file); sprintf(filename, "/tmp/trace.%d", fd); file = fopen(filename, "w+"); drm_fd = fd; } switch (request) { case DRM_IOCTL_I915_GEM_EXECBUFFER2: trace_exec(fd, argp); break; case DRM_IOCTL_I915_GEM_CREATE: { struct drm_i915_gem_create *create = argp; trace_add(create->handle, create->size); break; } case DRM_IOCTL_I915_GEM_USERPTR: { struct drm_i915_gem_userptr *userptr = argp; trace_add(userptr->handle, userptr->user_size); break; } case DRM_IOCTL_GEM_CLOSE: { struct drm_gem_close *close = argp; trace_del(close->handle); break; } case DRM_IOCTL_GEM_OPEN: { struct drm_gem_open *open = argp; trace_add(open->handle, open->size); break; } case DRM_IOCTL_PRIME_FD_TO_HANDLE: { struct drm_prime_handle *prime = argp; off_t size = lseek(prime->fd, 0, SEEK_END); fail_if(size == -1, "failed to get prime bo size\n"); trace_add(prime->handle, size); break; } case DRM_IOCTL_MODE_GETFB: { struct drm_mode_fb_cmd *cmd = argp; trace_add(cmd->handle, size_for_fb(cmd)); break; } } return 0; } static void __attribute__ ((constructor)) init(void) { libc_close = dlsym(RTLD_NEXT, "close"); libc_ioctl = dlsym(RTLD_NEXT, "ioctl"); fail_if(libc_close == NULL || libc_ioctl == NULL, "failed to get libc ioctl or close\n"); }