| /* |
| * Copyright © 2008 Intel Corporation |
| * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. |
| * |
| * 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. |
| * |
| * Authors: |
| * Eric Anholt <eric@anholt.net> |
| * |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include "config.h" |
| #endif |
| |
| #include <unistd.h> |
| #include <stdlib.h> |
| #include <stdio.h> |
| #include <string.h> |
| #include <errno.h> |
| #include <err.h> |
| #include <assert.h> |
| #include <sys/ioctl.h> |
| #include <fcntl.h> |
| #include <sys/stat.h> |
| #include <sys/mman.h> |
| #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM |
| #include <sys/sysinfo.h> |
| #elif defined(HAVE_SWAPCTL) /* Solaris */ |
| #include <sys/swap.h> |
| #endif |
| #include <sys/resource.h> |
| |
| #include "intel_io.h" |
| #include "drmtest.h" |
| #include "igt_aux.h" |
| #include "igt_debugfs.h" |
| |
| /** |
| * intel_get_total_ram_mb: |
| * |
| * Returns: |
| * The total amount of system RAM available in MB. |
| */ |
| uint64_t |
| intel_get_total_ram_mb(void) |
| { |
| uint64_t retval; |
| |
| #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM /* Linux */ |
| struct sysinfo sysinf; |
| |
| igt_assert(sysinfo(&sysinf) == 0); |
| retval = sysinf.totalram; |
| retval *= sysinf.mem_unit; |
| #elif defined(_SC_PAGESIZE) && defined(_SC_PHYS_PAGES) /* Solaris */ |
| long pagesize, npages; |
| |
| pagesize = sysconf(_SC_PAGESIZE); |
| npages = sysconf(_SC_PHYS_PAGES); |
| |
| retval = (uint64_t) pagesize * npages; |
| #else |
| #error "Unknown how to get RAM size for this OS" |
| #endif |
| |
| return retval / (1024*1024); |
| } |
| |
| /** |
| * intel_get_avail_ram_mb: |
| * |
| * Returns: |
| * The amount of unused system RAM available in MB. |
| */ |
| uint64_t |
| intel_get_avail_ram_mb(void) |
| { |
| uint64_t retval; |
| |
| #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM /* Linux */ |
| struct sysinfo sysinf; |
| int fd; |
| |
| fd = drm_open_driver(DRIVER_INTEL); |
| intel_purge_vm_caches(fd); |
| close(fd); |
| |
| igt_assert(sysinfo(&sysinf) == 0); |
| retval = sysinf.freeram; |
| retval *= sysinf.mem_unit; |
| #elif defined(_SC_PAGESIZE) && defined(_SC_AVPHYS_PAGES) /* Solaris */ |
| long pagesize, npages; |
| |
| pagesize = sysconf(_SC_PAGESIZE); |
| npages = sysconf(_SC_AVPHYS_PAGES); |
| |
| retval = (uint64_t) pagesize * npages; |
| #else |
| #error "Unknown how to get available RAM for this OS" |
| #endif |
| |
| return retval / (1024*1024); |
| } |
| |
| /** |
| * intel_get_total_swap_mb: |
| * |
| * Returns: |
| * The total amount of swap space available in MB. |
| */ |
| uint64_t |
| intel_get_total_swap_mb(void) |
| { |
| uint64_t retval; |
| |
| #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM /* Linux */ |
| struct sysinfo sysinf; |
| |
| igt_assert(sysinfo(&sysinf) == 0); |
| retval = sysinf.freeswap; |
| retval *= sysinf.mem_unit; |
| #elif defined(HAVE_SWAPCTL) /* Solaris */ |
| long pagesize = sysconf(_SC_PAGESIZE); |
| uint64_t totalpages = 0; |
| swaptbl_t *swt; |
| char *buf; |
| int n, i; |
| |
| if ((n = swapctl(SC_GETNSWP, NULL)) == -1) { |
| igt_warn("swapctl: GETNSWP"); |
| return 0; |
| } |
| if (n == 0) { |
| /* no error, but no swap devices either */ |
| return 0; |
| } |
| |
| swt = malloc(sizeof(struct swaptable) + (n * sizeof(swapent_t))); |
| buf = malloc(n * MAXPATHLEN); |
| if (!swt || !buf) { |
| igt_warn("malloc"); |
| } else { |
| swt->swt_n = n; |
| for (i = 0 ; i < n; i++) { |
| swt->swt_ent[i].ste_path = buf + (i * MAXPATHLEN); |
| } |
| |
| if ((n = swapctl(SC_LIST, swt)) == -1) { |
| igt_warn("swapctl: LIST"); |
| } else { |
| for (i = 0; i < swt->swt_n; i++) { |
| totalpages += swt->swt_ent[i].ste_pages; |
| } |
| } |
| } |
| free(swt); |
| free(buf); |
| |
| retval = (uint64_t) pagesize * totalpages; |
| #else |
| #warning "Unknown how to get swap size for this OS" |
| return 0; |
| #endif |
| |
| return retval / (1024*1024); |
| } |
| |
| static uint64_t vfs_file_max(void) |
| { |
| static long long unsigned max; |
| if (max == 0) { |
| FILE *file = fopen("/proc/sys/fs/file-max", "r"); |
| max = 80000; |
| if (file) { |
| igt_assert(fscanf(file, "%llu", &max) == 1); |
| fclose(file); |
| } |
| } |
| return max; |
| } |
| |
| static unsigned max_open_files(void) |
| { |
| struct rlimit rlim; |
| |
| if (getrlimit(RLIMIT_NOFILE, &rlim)) |
| rlim.rlim_cur = 64 << 10; |
| |
| return rlim.rlim_cur; |
| } |
| |
| /** |
| * intel_require_files: |
| * @count: number of files that will be created |
| * |
| * Does the system support enough file descriptors for the test? |
| */ |
| void intel_require_files(uint64_t count) |
| { |
| igt_require_f(count < max_open_files(), |
| "Estimated that we need %'llu files, but the process maximum is only %'llu\n", |
| (long long)count, (long long)max_open_files()); |
| } |
| |
| int __intel_check_memory(uint64_t count, uint64_t size, unsigned mode, |
| uint64_t *out_required, uint64_t *out_total) |
| { |
| /* rough estimate of how many bytes the kernel requires to track each object */ |
| #define KERNEL_BO_OVERHEAD 512 |
| uint64_t required, total; |
| |
| required = count; |
| required *= size + KERNEL_BO_OVERHEAD; |
| required = ALIGN(required, 4096); |
| |
| igt_debug("Checking %'llu surfaces of size %'llu bytes (total %'llu) against %s%s\n", |
| (long long)count, (long long)size, (long long)required, |
| mode & (CHECK_RAM | CHECK_SWAP) ? "RAM" : "", |
| mode & CHECK_SWAP ? " + swap": ""); |
| |
| total = 0; |
| if (mode & (CHECK_RAM | CHECK_SWAP)) |
| total += intel_get_avail_ram_mb(); |
| if (mode & CHECK_SWAP) |
| total += intel_get_total_swap_mb(); |
| total *= 1024 * 1024; |
| |
| if (out_required) |
| *out_required = required; |
| |
| if (out_total) |
| *out_total = total; |
| |
| if (count > vfs_file_max()) |
| return false; |
| |
| return required < total; |
| } |
| |
| /** |
| * intel_require_memory: |
| * @count: number of surfaces that will be created |
| * @size: the size in bytes of each surface |
| * @mode: a bit field declaring whether the test will be run in RAM or in SWAP |
| * |
| * Computes the total amount of memory required to allocate @count surfaces, |
| * each of @size bytes, and includes an estimate for kernel overhead. It then |
| * queries the kernel for the available amount of memory on the system (either |
| * RAM and/or SWAP depending upon @mode) and determines whether there is |
| * sufficient to run the test. |
| * |
| * Most tests should check that there is enough RAM to hold their working set. |
| * The rare swap thrashing tests should check that there is enough RAM + SWAP |
| * for their tests. oom-killer tests should only run if this reports that |
| * there is not enough RAM + SWAP! |
| * |
| * If there is not enough RAM this function calls igt_skip with an appropriate |
| * message. It only ever returns if the requirement is fulfilled. This function |
| * also causes the test to be skipped automatically on simulation under the |
| * assumption that any test that needs to check for memory requirements is a |
| * thrashing test unsuitable for slow simulated systems. |
| */ |
| void intel_require_memory(uint64_t count, uint64_t size, unsigned mode) |
| { |
| uint64_t required, total; |
| |
| igt_require_f(__intel_check_memory(count, size, mode, |
| &required, &total), |
| "Estimated that we need %'llu objects and %'llu MiB for the test, but only have %'llu MiB available (%s%s) and a maximum of %'llu objects\n", |
| (long long)count, |
| (long long)((required + ((1<<20) - 1)) >> 20), |
| (long long)(total >> 20), |
| mode & (CHECK_RAM | CHECK_SWAP) ? "RAM" : "", |
| mode & CHECK_SWAP ? " + swap": "", |
| (long long)vfs_file_max()); |
| |
| igt_skip_on_simulation(); |
| } |
| |
| void intel_purge_vm_caches(int drm_fd) |
| { |
| int fd; |
| |
| igt_drop_caches_set(drm_fd, DROP_SHRINK_ALL); |
| |
| fd = open("/proc/sys/vm/drop_caches", O_WRONLY); |
| if (fd >= 0) { |
| /* BIT(2): Be quiet. Cannot be combined with other operations, |
| * the sysctl has a max value of 4. |
| */ |
| igt_ignore_warn(write(fd, "4\n", 2)); |
| close(fd); |
| } |
| |
| /* Reset write position back to start. Do as a seperate write to keep |
| * the stages segregated and avoid failure from the squelching stopping |
| * the purge. |
| */ |
| fd = open("/proc/sys/vm/drop_caches", O_WRONLY); |
| if (fd < 0) |
| return; |
| |
| /* BIT(0): Drop page cache |
| * BIT(1): Drop slab cache |
| */ |
| igt_ignore_warn(write(fd, "3\n", 2)); |
| close(fd); |
| } |
| |
| |
| /* |
| * When testing a port to a new platform, create a standalone test binary |
| * by running: |
| * cc -o porttest intel_drm.c -I.. -DSTANDALONE_TEST `pkg-config --cflags libdrm` |
| * and then running the resulting porttest program. |
| */ |
| #ifdef STANDALONE_TEST |
| void *mmio; |
| |
| int main(int argc, char **argv) |
| { |
| igt_info("Total RAM: %"PRIu64" Mb\n", intel_get_total_ram_mb()); |
| igt_info("Total Swap: %"PRIu64" Mb\n", intel_get_total_swap_mb()); |
| |
| return 0; |
| } |
| #endif /* STANDALONE_TEST */ |