blob: e5dea6e5a61978629b434ec22d4f4285f1277235 [file] [log] [blame]
/*
* 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 */