xref: /aosp_15_r20/external/igt-gpu-tools/lib/intel_os.c (revision d83cc019efdc2edc6c4b16e9034a3ceb8d35d77c)
1*d83cc019SAndroid Build Coastguard Worker /*
2*d83cc019SAndroid Build Coastguard Worker  * Copyright © 2008 Intel Corporation
3*d83cc019SAndroid Build Coastguard Worker  * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved.
4*d83cc019SAndroid Build Coastguard Worker  *
5*d83cc019SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
6*d83cc019SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the "Software"),
7*d83cc019SAndroid Build Coastguard Worker  * to deal in the Software without restriction, including without limitation
8*d83cc019SAndroid Build Coastguard Worker  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9*d83cc019SAndroid Build Coastguard Worker  * and/or sell copies of the Software, and to permit persons to whom the
10*d83cc019SAndroid Build Coastguard Worker  * Software is furnished to do so, subject to the following conditions:
11*d83cc019SAndroid Build Coastguard Worker  *
12*d83cc019SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the next
13*d83cc019SAndroid Build Coastguard Worker  * paragraph) shall be included in all copies or substantial portions of the
14*d83cc019SAndroid Build Coastguard Worker  * Software.
15*d83cc019SAndroid Build Coastguard Worker  *
16*d83cc019SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*d83cc019SAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*d83cc019SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
19*d83cc019SAndroid Build Coastguard Worker  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*d83cc019SAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*d83cc019SAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22*d83cc019SAndroid Build Coastguard Worker  * DEALINGS IN THE SOFTWARE.
23*d83cc019SAndroid Build Coastguard Worker  *
24*d83cc019SAndroid Build Coastguard Worker  * Authors:
25*d83cc019SAndroid Build Coastguard Worker  *    Eric Anholt <[email protected]>
26*d83cc019SAndroid Build Coastguard Worker  *
27*d83cc019SAndroid Build Coastguard Worker  */
28*d83cc019SAndroid Build Coastguard Worker 
29*d83cc019SAndroid Build Coastguard Worker #include "config.h"
30*d83cc019SAndroid Build Coastguard Worker 
31*d83cc019SAndroid Build Coastguard Worker #include <inttypes.h>
32*d83cc019SAndroid Build Coastguard Worker #include <unistd.h>
33*d83cc019SAndroid Build Coastguard Worker #include <stdlib.h>
34*d83cc019SAndroid Build Coastguard Worker #include <stdio.h>
35*d83cc019SAndroid Build Coastguard Worker #include <string.h>
36*d83cc019SAndroid Build Coastguard Worker #include <errno.h>
37*d83cc019SAndroid Build Coastguard Worker #include <err.h>
38*d83cc019SAndroid Build Coastguard Worker #include <assert.h>
39*d83cc019SAndroid Build Coastguard Worker #include <sys/ioctl.h>
40*d83cc019SAndroid Build Coastguard Worker #include <fcntl.h>
41*d83cc019SAndroid Build Coastguard Worker #include <sys/stat.h>
42*d83cc019SAndroid Build Coastguard Worker #include <sys/mman.h>
43*d83cc019SAndroid Build Coastguard Worker #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM
44*d83cc019SAndroid Build Coastguard Worker #include <sys/sysinfo.h>
45*d83cc019SAndroid Build Coastguard Worker #elif defined(HAVE_SWAPCTL) /* Solaris */
46*d83cc019SAndroid Build Coastguard Worker #include <sys/swap.h>
47*d83cc019SAndroid Build Coastguard Worker #endif
48*d83cc019SAndroid Build Coastguard Worker #include <sys/resource.h>
49*d83cc019SAndroid Build Coastguard Worker 
50*d83cc019SAndroid Build Coastguard Worker #include "intel_io.h"
51*d83cc019SAndroid Build Coastguard Worker #include "drmtest.h"
52*d83cc019SAndroid Build Coastguard Worker #include "igt_aux.h"
53*d83cc019SAndroid Build Coastguard Worker #include "igt_debugfs.h"
54*d83cc019SAndroid Build Coastguard Worker #include "igt_sysfs.h"
55*d83cc019SAndroid Build Coastguard Worker 
56*d83cc019SAndroid Build Coastguard Worker /**
57*d83cc019SAndroid Build Coastguard Worker  * intel_get_total_ram_mb:
58*d83cc019SAndroid Build Coastguard Worker  *
59*d83cc019SAndroid Build Coastguard Worker  * Returns:
60*d83cc019SAndroid Build Coastguard Worker  * The total amount of system RAM available in MB.
61*d83cc019SAndroid Build Coastguard Worker  */
62*d83cc019SAndroid Build Coastguard Worker uint64_t
intel_get_total_ram_mb(void)63*d83cc019SAndroid Build Coastguard Worker intel_get_total_ram_mb(void)
64*d83cc019SAndroid Build Coastguard Worker {
65*d83cc019SAndroid Build Coastguard Worker 	uint64_t retval;
66*d83cc019SAndroid Build Coastguard Worker 
67*d83cc019SAndroid Build Coastguard Worker #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM /* Linux */
68*d83cc019SAndroid Build Coastguard Worker 	struct sysinfo sysinf;
69*d83cc019SAndroid Build Coastguard Worker 
70*d83cc019SAndroid Build Coastguard Worker 	igt_assert(sysinfo(&sysinf) == 0);
71*d83cc019SAndroid Build Coastguard Worker 	retval = sysinf.totalram;
72*d83cc019SAndroid Build Coastguard Worker 	retval *= sysinf.mem_unit;
73*d83cc019SAndroid Build Coastguard Worker #elif defined(_SC_PAGESIZE) && defined(_SC_PHYS_PAGES) /* Solaris */
74*d83cc019SAndroid Build Coastguard Worker 	long pagesize, npages;
75*d83cc019SAndroid Build Coastguard Worker 
76*d83cc019SAndroid Build Coastguard Worker 	pagesize = sysconf(_SC_PAGESIZE);
77*d83cc019SAndroid Build Coastguard Worker         npages = sysconf(_SC_PHYS_PAGES);
78*d83cc019SAndroid Build Coastguard Worker 
79*d83cc019SAndroid Build Coastguard Worker 	retval = (uint64_t) pagesize * npages;
80*d83cc019SAndroid Build Coastguard Worker #else
81*d83cc019SAndroid Build Coastguard Worker #error "Unknown how to get RAM size for this OS"
82*d83cc019SAndroid Build Coastguard Worker #endif
83*d83cc019SAndroid Build Coastguard Worker 
84*d83cc019SAndroid Build Coastguard Worker 	return retval / (1024*1024);
85*d83cc019SAndroid Build Coastguard Worker }
86*d83cc019SAndroid Build Coastguard Worker 
get_meminfo(const char * info,const char * tag)87*d83cc019SAndroid Build Coastguard Worker static uint64_t get_meminfo(const char *info, const char *tag)
88*d83cc019SAndroid Build Coastguard Worker {
89*d83cc019SAndroid Build Coastguard Worker 	const char *str;
90*d83cc019SAndroid Build Coastguard Worker 	unsigned long val;
91*d83cc019SAndroid Build Coastguard Worker 
92*d83cc019SAndroid Build Coastguard Worker 	str = strstr(info, tag);
93*d83cc019SAndroid Build Coastguard Worker 	if (str && sscanf(str + strlen(tag), " %lu", &val) == 1)
94*d83cc019SAndroid Build Coastguard Worker 		return (uint64_t)val << 10;
95*d83cc019SAndroid Build Coastguard Worker 
96*d83cc019SAndroid Build Coastguard Worker 	igt_warn("Unrecognised /proc/meminfo field: '%s'\n", tag);
97*d83cc019SAndroid Build Coastguard Worker 	return 0;
98*d83cc019SAndroid Build Coastguard Worker }
99*d83cc019SAndroid Build Coastguard Worker 
100*d83cc019SAndroid Build Coastguard Worker /**
101*d83cc019SAndroid Build Coastguard Worker  * intel_get_avail_ram_mb:
102*d83cc019SAndroid Build Coastguard Worker  *
103*d83cc019SAndroid Build Coastguard Worker  * Returns:
104*d83cc019SAndroid Build Coastguard Worker  * The amount of unused system RAM available in MB.
105*d83cc019SAndroid Build Coastguard Worker  */
106*d83cc019SAndroid Build Coastguard Worker uint64_t
intel_get_avail_ram_mb(void)107*d83cc019SAndroid Build Coastguard Worker intel_get_avail_ram_mb(void)
108*d83cc019SAndroid Build Coastguard Worker {
109*d83cc019SAndroid Build Coastguard Worker 	uint64_t retval;
110*d83cc019SAndroid Build Coastguard Worker 
111*d83cc019SAndroid Build Coastguard Worker #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM /* Linux */
112*d83cc019SAndroid Build Coastguard Worker 	char *info;
113*d83cc019SAndroid Build Coastguard Worker 	int fd;
114*d83cc019SAndroid Build Coastguard Worker 
115*d83cc019SAndroid Build Coastguard Worker 	fd = drm_open_driver(DRIVER_INTEL);
116*d83cc019SAndroid Build Coastguard Worker 	intel_purge_vm_caches(fd);
117*d83cc019SAndroid Build Coastguard Worker 	close(fd);
118*d83cc019SAndroid Build Coastguard Worker 
119*d83cc019SAndroid Build Coastguard Worker 	fd = open("/proc", O_RDONLY);
120*d83cc019SAndroid Build Coastguard Worker 	info = igt_sysfs_get(fd, "meminfo");
121*d83cc019SAndroid Build Coastguard Worker 	close(fd);
122*d83cc019SAndroid Build Coastguard Worker 
123*d83cc019SAndroid Build Coastguard Worker 	if (info) {
124*d83cc019SAndroid Build Coastguard Worker 		retval  = get_meminfo(info, "MemAvailable:");
125*d83cc019SAndroid Build Coastguard Worker 		retval += get_meminfo(info, "Buffers:");
126*d83cc019SAndroid Build Coastguard Worker 		/*
127*d83cc019SAndroid Build Coastguard Worker 		 * Include the file+swap cache as "available" for the test.
128*d83cc019SAndroid Build Coastguard Worker 		 * We believe that we can revoke these pages back to their
129*d83cc019SAndroid Build Coastguard Worker 		 * on disk counterpart, with no loss of functionality while
130*d83cc019SAndroid Build Coastguard Worker 		 * the test runs using those pages for ourselves without the
131*d83cc019SAndroid Build Coastguard Worker 		 * test itself being swapped to disk.
132*d83cc019SAndroid Build Coastguard Worker 		 */
133*d83cc019SAndroid Build Coastguard Worker 		retval += get_meminfo(info, "Cached:");
134*d83cc019SAndroid Build Coastguard Worker 		retval += get_meminfo(info, "SwapCached:");
135*d83cc019SAndroid Build Coastguard Worker 		free(info);
136*d83cc019SAndroid Build Coastguard Worker 	} else {
137*d83cc019SAndroid Build Coastguard Worker 		struct sysinfo sysinf;
138*d83cc019SAndroid Build Coastguard Worker 
139*d83cc019SAndroid Build Coastguard Worker 		igt_assert(sysinfo(&sysinf) == 0);
140*d83cc019SAndroid Build Coastguard Worker 		retval = sysinf.freeram;
141*d83cc019SAndroid Build Coastguard Worker 		retval += min(sysinf.freeswap, sysinf.bufferram);
142*d83cc019SAndroid Build Coastguard Worker 		retval *= sysinf.mem_unit;
143*d83cc019SAndroid Build Coastguard Worker 	}
144*d83cc019SAndroid Build Coastguard Worker #elif defined(_SC_PAGESIZE) && defined(_SC_AVPHYS_PAGES) /* Solaris */
145*d83cc019SAndroid Build Coastguard Worker 	long pagesize, npages;
146*d83cc019SAndroid Build Coastguard Worker 
147*d83cc019SAndroid Build Coastguard Worker 	pagesize = sysconf(_SC_PAGESIZE);
148*d83cc019SAndroid Build Coastguard Worker         npages = sysconf(_SC_AVPHYS_PAGES);
149*d83cc019SAndroid Build Coastguard Worker 
150*d83cc019SAndroid Build Coastguard Worker 	retval = (uint64_t) pagesize * npages;
151*d83cc019SAndroid Build Coastguard Worker #else
152*d83cc019SAndroid Build Coastguard Worker #error "Unknown how to get available RAM for this OS"
153*d83cc019SAndroid Build Coastguard Worker #endif
154*d83cc019SAndroid Build Coastguard Worker 
155*d83cc019SAndroid Build Coastguard Worker 	return retval / (1024*1024);
156*d83cc019SAndroid Build Coastguard Worker }
157*d83cc019SAndroid Build Coastguard Worker 
158*d83cc019SAndroid Build Coastguard Worker /**
159*d83cc019SAndroid Build Coastguard Worker  * intel_get_total_swap_mb:
160*d83cc019SAndroid Build Coastguard Worker  *
161*d83cc019SAndroid Build Coastguard Worker  * Returns:
162*d83cc019SAndroid Build Coastguard Worker  * The total amount of swap space available in MB.
163*d83cc019SAndroid Build Coastguard Worker  */
164*d83cc019SAndroid Build Coastguard Worker uint64_t
intel_get_total_swap_mb(void)165*d83cc019SAndroid Build Coastguard Worker intel_get_total_swap_mb(void)
166*d83cc019SAndroid Build Coastguard Worker {
167*d83cc019SAndroid Build Coastguard Worker 	uint64_t retval;
168*d83cc019SAndroid Build Coastguard Worker 
169*d83cc019SAndroid Build Coastguard Worker #ifdef HAVE_STRUCT_SYSINFO_TOTALRAM /* Linux */
170*d83cc019SAndroid Build Coastguard Worker 	struct sysinfo sysinf;
171*d83cc019SAndroid Build Coastguard Worker 
172*d83cc019SAndroid Build Coastguard Worker 	igt_assert(sysinfo(&sysinf) == 0);
173*d83cc019SAndroid Build Coastguard Worker 	retval = sysinf.freeswap;
174*d83cc019SAndroid Build Coastguard Worker 	retval *= sysinf.mem_unit;
175*d83cc019SAndroid Build Coastguard Worker #elif defined(HAVE_SWAPCTL) /* Solaris */
176*d83cc019SAndroid Build Coastguard Worker 	long pagesize = sysconf(_SC_PAGESIZE);
177*d83cc019SAndroid Build Coastguard Worker 	uint64_t totalpages = 0;
178*d83cc019SAndroid Build Coastguard Worker 	swaptbl_t *swt;
179*d83cc019SAndroid Build Coastguard Worker 	char *buf;
180*d83cc019SAndroid Build Coastguard Worker 	int n, i;
181*d83cc019SAndroid Build Coastguard Worker 
182*d83cc019SAndroid Build Coastguard Worker 	if ((n = swapctl(SC_GETNSWP, NULL)) == -1) {
183*d83cc019SAndroid Build Coastguard Worker 	    igt_warn("swapctl: GETNSWP");
184*d83cc019SAndroid Build Coastguard Worker 	    return 0;
185*d83cc019SAndroid Build Coastguard Worker 	}
186*d83cc019SAndroid Build Coastguard Worker 	if (n == 0) {
187*d83cc019SAndroid Build Coastguard Worker 	    /* no error, but no swap devices either */
188*d83cc019SAndroid Build Coastguard Worker 	    return 0;
189*d83cc019SAndroid Build Coastguard Worker 	}
190*d83cc019SAndroid Build Coastguard Worker 
191*d83cc019SAndroid Build Coastguard Worker 	swt = malloc(sizeof(struct swaptable) + (n * sizeof(swapent_t)));
192*d83cc019SAndroid Build Coastguard Worker 	buf = malloc(n * MAXPATHLEN);
193*d83cc019SAndroid Build Coastguard Worker 	if (!swt || !buf) {
194*d83cc019SAndroid Build Coastguard Worker 	    igt_warn("malloc");
195*d83cc019SAndroid Build Coastguard Worker 	} else {
196*d83cc019SAndroid Build Coastguard Worker 	    swt->swt_n = n;
197*d83cc019SAndroid Build Coastguard Worker 	    for (i = 0 ; i < n; i++) {
198*d83cc019SAndroid Build Coastguard Worker 		swt->swt_ent[i].ste_path = buf + (i * MAXPATHLEN);
199*d83cc019SAndroid Build Coastguard Worker 	    }
200*d83cc019SAndroid Build Coastguard Worker 
201*d83cc019SAndroid Build Coastguard Worker 	    if ((n = swapctl(SC_LIST, swt)) == -1) {
202*d83cc019SAndroid Build Coastguard Worker 		igt_warn("swapctl: LIST");
203*d83cc019SAndroid Build Coastguard Worker 	    } else {
204*d83cc019SAndroid Build Coastguard Worker 		for (i = 0; i < swt->swt_n; i++) {
205*d83cc019SAndroid Build Coastguard Worker 		    totalpages += swt->swt_ent[i].ste_pages;
206*d83cc019SAndroid Build Coastguard Worker 		}
207*d83cc019SAndroid Build Coastguard Worker 	    }
208*d83cc019SAndroid Build Coastguard Worker 	}
209*d83cc019SAndroid Build Coastguard Worker 	free(swt);
210*d83cc019SAndroid Build Coastguard Worker 	free(buf);
211*d83cc019SAndroid Build Coastguard Worker 
212*d83cc019SAndroid Build Coastguard Worker 	retval = (uint64_t) pagesize * totalpages;
213*d83cc019SAndroid Build Coastguard Worker #else
214*d83cc019SAndroid Build Coastguard Worker #warning "Unknown how to get swap size for this OS"
215*d83cc019SAndroid Build Coastguard Worker 	return 0;
216*d83cc019SAndroid Build Coastguard Worker #endif
217*d83cc019SAndroid Build Coastguard Worker 
218*d83cc019SAndroid Build Coastguard Worker 	return retval / (1024*1024);
219*d83cc019SAndroid Build Coastguard Worker }
220*d83cc019SAndroid Build Coastguard Worker 
221*d83cc019SAndroid Build Coastguard Worker /**
222*d83cc019SAndroid Build Coastguard Worker  * intel_get_total_pinnable_mem:
223*d83cc019SAndroid Build Coastguard Worker  *
224*d83cc019SAndroid Build Coastguard Worker  * Compute the amount of memory that we're able to safely lock.
225*d83cc019SAndroid Build Coastguard Worker  * Note that in order to achieve this, we're attempting to repeatedly lock more
226*d83cc019SAndroid Build Coastguard Worker  * and more memory, which is a time consuming process.
227*d83cc019SAndroid Build Coastguard Worker  *
228*d83cc019SAndroid Build Coastguard Worker  * Returns: Amount of memory that can be safely pinned, in bytes.
229*d83cc019SAndroid Build Coastguard Worker  */
intel_get_total_pinnable_mem(size_t * total)230*d83cc019SAndroid Build Coastguard Worker void *intel_get_total_pinnable_mem(size_t *total)
231*d83cc019SAndroid Build Coastguard Worker {
232*d83cc019SAndroid Build Coastguard Worker 	uint64_t *can_mlock, pin, avail;
233*d83cc019SAndroid Build Coastguard Worker 
234*d83cc019SAndroid Build Coastguard Worker 	pin = (intel_get_total_ram_mb() + 1) << 20;
235*d83cc019SAndroid Build Coastguard Worker 	avail = (intel_get_avail_ram_mb() + 1) << 20;
236*d83cc019SAndroid Build Coastguard Worker 
237*d83cc019SAndroid Build Coastguard Worker 	can_mlock = mmap(NULL, pin, PROT_WRITE, MAP_SHARED | MAP_ANON, -1, 0);
238*d83cc019SAndroid Build Coastguard Worker 	igt_require(can_mlock != MAP_FAILED);
239*d83cc019SAndroid Build Coastguard Worker 
240*d83cc019SAndroid Build Coastguard Worker 	/*
241*d83cc019SAndroid Build Coastguard Worker 	 * We can reasonably assume that we should be able to lock at
242*d83cc019SAndroid Build Coastguard Worker 	 * least 3/4 of available RAM
243*d83cc019SAndroid Build Coastguard Worker 	 */
244*d83cc019SAndroid Build Coastguard Worker 	*can_mlock = (avail >> 1) + (avail >> 2);
245*d83cc019SAndroid Build Coastguard Worker 	if (mlock(can_mlock, *can_mlock)) {
246*d83cc019SAndroid Build Coastguard Worker 		munmap(can_mlock, pin);
247*d83cc019SAndroid Build Coastguard Worker 		return MAP_FAILED;
248*d83cc019SAndroid Build Coastguard Worker 	}
249*d83cc019SAndroid Build Coastguard Worker 
250*d83cc019SAndroid Build Coastguard Worker 	for (uint64_t inc = 1024 << 20; inc >= 4 << 10; inc >>= 2) {
251*d83cc019SAndroid Build Coastguard Worker 		uint64_t locked = *can_mlock;
252*d83cc019SAndroid Build Coastguard Worker 
253*d83cc019SAndroid Build Coastguard Worker 		igt_debug("Testing mlock %'"PRIu64"B (%'"PRIu64"MiB) + %'"PRIu64"B\n",
254*d83cc019SAndroid Build Coastguard Worker 			  locked, locked >> 20, inc);
255*d83cc019SAndroid Build Coastguard Worker 
256*d83cc019SAndroid Build Coastguard Worker 		igt_fork(child, 1) {
257*d83cc019SAndroid Build Coastguard Worker 			uint64_t bytes = *can_mlock;
258*d83cc019SAndroid Build Coastguard Worker 
259*d83cc019SAndroid Build Coastguard Worker 			while (bytes <= pin) {
260*d83cc019SAndroid Build Coastguard Worker 				if (mlock((void *)can_mlock + bytes, inc))
261*d83cc019SAndroid Build Coastguard Worker 					break;
262*d83cc019SAndroid Build Coastguard Worker 
263*d83cc019SAndroid Build Coastguard Worker 				*can_mlock = bytes += inc;
264*d83cc019SAndroid Build Coastguard Worker 				__sync_synchronize();
265*d83cc019SAndroid Build Coastguard Worker 			}
266*d83cc019SAndroid Build Coastguard Worker 		}
267*d83cc019SAndroid Build Coastguard Worker 		__igt_waitchildren();
268*d83cc019SAndroid Build Coastguard Worker 
269*d83cc019SAndroid Build Coastguard Worker 		if (*can_mlock > locked + inc) { /* Weird bit of mm/ lore */
270*d83cc019SAndroid Build Coastguard Worker 			*can_mlock -= inc;
271*d83cc019SAndroid Build Coastguard Worker 			igt_debug("Claiming mlock %'"PRIu64"B (%'"PRIu64"MiB)\n",
272*d83cc019SAndroid Build Coastguard Worker 				  *can_mlock, *can_mlock >> 20);
273*d83cc019SAndroid Build Coastguard Worker 			igt_assert(!mlock((void *)can_mlock + locked,
274*d83cc019SAndroid Build Coastguard Worker 					  *can_mlock - locked));
275*d83cc019SAndroid Build Coastguard Worker 		}
276*d83cc019SAndroid Build Coastguard Worker 	}
277*d83cc019SAndroid Build Coastguard Worker 
278*d83cc019SAndroid Build Coastguard Worker 	*total = pin;
279*d83cc019SAndroid Build Coastguard Worker 	return can_mlock;
280*d83cc019SAndroid Build Coastguard Worker }
281*d83cc019SAndroid Build Coastguard Worker 
max_open_files(void)282*d83cc019SAndroid Build Coastguard Worker static unsigned max_open_files(void)
283*d83cc019SAndroid Build Coastguard Worker {
284*d83cc019SAndroid Build Coastguard Worker 	struct rlimit rlim;
285*d83cc019SAndroid Build Coastguard Worker 
286*d83cc019SAndroid Build Coastguard Worker 	if (getrlimit(RLIMIT_NOFILE, &rlim))
287*d83cc019SAndroid Build Coastguard Worker 		rlim.rlim_cur = 64 << 10;
288*d83cc019SAndroid Build Coastguard Worker 
289*d83cc019SAndroid Build Coastguard Worker 	return rlim.rlim_cur;
290*d83cc019SAndroid Build Coastguard Worker }
291*d83cc019SAndroid Build Coastguard Worker 
292*d83cc019SAndroid Build Coastguard Worker /**
293*d83cc019SAndroid Build Coastguard Worker  * intel_require_files:
294*d83cc019SAndroid Build Coastguard Worker  * @count: number of files that will be created
295*d83cc019SAndroid Build Coastguard Worker  *
296*d83cc019SAndroid Build Coastguard Worker  * Does the system support enough file descriptors for the test?
297*d83cc019SAndroid Build Coastguard Worker  */
intel_require_files(uint64_t count)298*d83cc019SAndroid Build Coastguard Worker void intel_require_files(uint64_t count)
299*d83cc019SAndroid Build Coastguard Worker {
300*d83cc019SAndroid Build Coastguard Worker 	igt_require_f(count < max_open_files(),
301*d83cc019SAndroid Build Coastguard Worker 		      "Estimated that we need %'llu files, but the process maximum is only %'llu\n",
302*d83cc019SAndroid Build Coastguard Worker 		      (long long)count, (long long)max_open_files());
303*d83cc019SAndroid Build Coastguard Worker }
304*d83cc019SAndroid Build Coastguard Worker 
__intel_check_memory(uint64_t count,uint64_t size,unsigned mode,uint64_t * out_required,uint64_t * out_total)305*d83cc019SAndroid Build Coastguard Worker int __intel_check_memory(uint64_t count, uint64_t size, unsigned mode,
306*d83cc019SAndroid Build Coastguard Worker 			 uint64_t *out_required, uint64_t *out_total)
307*d83cc019SAndroid Build Coastguard Worker {
308*d83cc019SAndroid Build Coastguard Worker /* rough estimate of how many bytes the kernel requires to track each object */
309*d83cc019SAndroid Build Coastguard Worker #define KERNEL_BO_OVERHEAD 512
310*d83cc019SAndroid Build Coastguard Worker 	uint64_t required, total;
311*d83cc019SAndroid Build Coastguard Worker 
312*d83cc019SAndroid Build Coastguard Worker 	required = count;
313*d83cc019SAndroid Build Coastguard Worker 	required *= size + KERNEL_BO_OVERHEAD;
314*d83cc019SAndroid Build Coastguard Worker 	required = ALIGN(required, 4096);
315*d83cc019SAndroid Build Coastguard Worker 
316*d83cc019SAndroid Build Coastguard Worker 	igt_debug("Checking %'llu surfaces of size %'llu bytes (total %'llu) against %s%s\n",
317*d83cc019SAndroid Build Coastguard Worker 		  (long long)count, (long long)size, (long long)required,
318*d83cc019SAndroid Build Coastguard Worker 		  mode & (CHECK_RAM | CHECK_SWAP) ? "RAM" : "",
319*d83cc019SAndroid Build Coastguard Worker 		  mode & CHECK_SWAP ? " + swap": "");
320*d83cc019SAndroid Build Coastguard Worker 
321*d83cc019SAndroid Build Coastguard Worker 	total = 0;
322*d83cc019SAndroid Build Coastguard Worker 	if (mode & (CHECK_RAM | CHECK_SWAP))
323*d83cc019SAndroid Build Coastguard Worker 		total += intel_get_avail_ram_mb();
324*d83cc019SAndroid Build Coastguard Worker 	if (mode & CHECK_SWAP)
325*d83cc019SAndroid Build Coastguard Worker 		total += intel_get_total_swap_mb();
326*d83cc019SAndroid Build Coastguard Worker 	total *= 1024 * 1024;
327*d83cc019SAndroid Build Coastguard Worker 
328*d83cc019SAndroid Build Coastguard Worker 	if (out_required)
329*d83cc019SAndroid Build Coastguard Worker 		*out_required = required;
330*d83cc019SAndroid Build Coastguard Worker 
331*d83cc019SAndroid Build Coastguard Worker 	if (out_total)
332*d83cc019SAndroid Build Coastguard Worker 		*out_total = total;
333*d83cc019SAndroid Build Coastguard Worker 
334*d83cc019SAndroid Build Coastguard Worker 	if (count > vfs_file_max())
335*d83cc019SAndroid Build Coastguard Worker 		return false;
336*d83cc019SAndroid Build Coastguard Worker 
337*d83cc019SAndroid Build Coastguard Worker 	return required < total;
338*d83cc019SAndroid Build Coastguard Worker }
339*d83cc019SAndroid Build Coastguard Worker 
340*d83cc019SAndroid Build Coastguard Worker /**
341*d83cc019SAndroid Build Coastguard Worker  * intel_require_memory:
342*d83cc019SAndroid Build Coastguard Worker  * @count: number of surfaces that will be created
343*d83cc019SAndroid Build Coastguard Worker  * @size: the size in bytes of each surface
344*d83cc019SAndroid Build Coastguard Worker  * @mode: a bit field declaring whether the test will be run in RAM or in SWAP
345*d83cc019SAndroid Build Coastguard Worker  *
346*d83cc019SAndroid Build Coastguard Worker  * Computes the total amount of memory required to allocate @count surfaces,
347*d83cc019SAndroid Build Coastguard Worker  * each of @size bytes, and includes an estimate for kernel overhead. It then
348*d83cc019SAndroid Build Coastguard Worker  * queries the kernel for the available amount of memory on the system (either
349*d83cc019SAndroid Build Coastguard Worker  * RAM and/or SWAP depending upon @mode) and determines whether there is
350*d83cc019SAndroid Build Coastguard Worker  * sufficient to run the test.
351*d83cc019SAndroid Build Coastguard Worker  *
352*d83cc019SAndroid Build Coastguard Worker  * Most tests should check that there is enough RAM to hold their working set.
353*d83cc019SAndroid Build Coastguard Worker  * The rare swap thrashing tests should check that there is enough RAM + SWAP
354*d83cc019SAndroid Build Coastguard Worker  * for their tests. oom-killer tests should only run if this reports that
355*d83cc019SAndroid Build Coastguard Worker  * there is not enough RAM + SWAP!
356*d83cc019SAndroid Build Coastguard Worker  *
357*d83cc019SAndroid Build Coastguard Worker  * If there is not enough RAM this function calls igt_skip with an appropriate
358*d83cc019SAndroid Build Coastguard Worker  * message. It only ever returns if the requirement is fulfilled. This function
359*d83cc019SAndroid Build Coastguard Worker  * also causes the test to be skipped automatically on simulation under the
360*d83cc019SAndroid Build Coastguard Worker  * assumption that any test that needs to check for memory requirements is a
361*d83cc019SAndroid Build Coastguard Worker  * thrashing test unsuitable for slow simulated systems.
362*d83cc019SAndroid Build Coastguard Worker  */
intel_require_memory(uint64_t count,uint64_t size,unsigned mode)363*d83cc019SAndroid Build Coastguard Worker void intel_require_memory(uint64_t count, uint64_t size, unsigned mode)
364*d83cc019SAndroid Build Coastguard Worker {
365*d83cc019SAndroid Build Coastguard Worker 	uint64_t required, total;
366*d83cc019SAndroid Build Coastguard Worker 	bool sufficient_memory;
367*d83cc019SAndroid Build Coastguard Worker 
368*d83cc019SAndroid Build Coastguard Worker 	igt_skip_on_simulation();
369*d83cc019SAndroid Build Coastguard Worker 
370*d83cc019SAndroid Build Coastguard Worker 	sufficient_memory = __intel_check_memory(count, size, mode,
371*d83cc019SAndroid Build Coastguard Worker 						 &required, &total);
372*d83cc019SAndroid Build Coastguard Worker 	if (!sufficient_memory) {
373*d83cc019SAndroid Build Coastguard Worker 		int dir = open("/proc", O_RDONLY);
374*d83cc019SAndroid Build Coastguard Worker 		char *info;
375*d83cc019SAndroid Build Coastguard Worker 
376*d83cc019SAndroid Build Coastguard Worker 		info = igt_sysfs_get(dir, "meminfo");
377*d83cc019SAndroid Build Coastguard Worker 		if (info) {
378*d83cc019SAndroid Build Coastguard Worker 			igt_warn("Insufficient free memory; /proc/meminfo:\n%s",
379*d83cc019SAndroid Build Coastguard Worker 				 info);
380*d83cc019SAndroid Build Coastguard Worker 			free(info);
381*d83cc019SAndroid Build Coastguard Worker 		}
382*d83cc019SAndroid Build Coastguard Worker 
383*d83cc019SAndroid Build Coastguard Worker 		info = igt_sysfs_get(dir, "slabinfo");
384*d83cc019SAndroid Build Coastguard Worker 		if (info) {
385*d83cc019SAndroid Build Coastguard Worker 			igt_warn("Insufficient free memory; /proc/slabinfo:\n%s",
386*d83cc019SAndroid Build Coastguard Worker 				 info);
387*d83cc019SAndroid Build Coastguard Worker 			free(info);
388*d83cc019SAndroid Build Coastguard Worker 		}
389*d83cc019SAndroid Build Coastguard Worker 
390*d83cc019SAndroid Build Coastguard Worker 		close(dir);
391*d83cc019SAndroid Build Coastguard Worker 	}
392*d83cc019SAndroid Build Coastguard Worker 
393*d83cc019SAndroid Build Coastguard Worker 	igt_require_f(sufficient_memory,
394*d83cc019SAndroid Build Coastguard Worker 		      "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",
395*d83cc019SAndroid Build Coastguard Worker 		      (long long)count,
396*d83cc019SAndroid Build Coastguard Worker 		      (long long)((required + ((1<<20) - 1)) >> 20),
397*d83cc019SAndroid Build Coastguard Worker 		      (long long)(total >> 20),
398*d83cc019SAndroid Build Coastguard Worker 		      mode & (CHECK_RAM | CHECK_SWAP) ? "RAM" : "",
399*d83cc019SAndroid Build Coastguard Worker 		      mode & CHECK_SWAP ? " + swap": "",
400*d83cc019SAndroid Build Coastguard Worker 		      (long long)vfs_file_max());
401*d83cc019SAndroid Build Coastguard Worker }
402*d83cc019SAndroid Build Coastguard Worker 
intel_purge_vm_caches(int drm_fd)403*d83cc019SAndroid Build Coastguard Worker void intel_purge_vm_caches(int drm_fd)
404*d83cc019SAndroid Build Coastguard Worker {
405*d83cc019SAndroid Build Coastguard Worker 	int fd;
406*d83cc019SAndroid Build Coastguard Worker 
407*d83cc019SAndroid Build Coastguard Worker 	fd = open("/proc/sys/vm/drop_caches", O_WRONLY);
408*d83cc019SAndroid Build Coastguard Worker 	if (fd >= 0) {
409*d83cc019SAndroid Build Coastguard Worker 		/*
410*d83cc019SAndroid Build Coastguard Worker 		 * BIT(2): Be quiet. Cannot be combined with other operations,
411*d83cc019SAndroid Build Coastguard Worker 		 * the sysctl has a max value of 4.
412*d83cc019SAndroid Build Coastguard Worker 		 */
413*d83cc019SAndroid Build Coastguard Worker 		igt_ignore_warn(write(fd, "4\n", 2));
414*d83cc019SAndroid Build Coastguard Worker 		close(fd);
415*d83cc019SAndroid Build Coastguard Worker 	}
416*d83cc019SAndroid Build Coastguard Worker 
417*d83cc019SAndroid Build Coastguard Worker 	for (int loop = 0; loop < 2; loop++) {
418*d83cc019SAndroid Build Coastguard Worker 		igt_drop_caches_set(drm_fd,
419*d83cc019SAndroid Build Coastguard Worker 				    DROP_SHRINK_ALL | DROP_IDLE | DROP_FREED);
420*d83cc019SAndroid Build Coastguard Worker 
421*d83cc019SAndroid Build Coastguard Worker 		fd = open("/proc/sys/vm/drop_caches", O_WRONLY);
422*d83cc019SAndroid Build Coastguard Worker 		if (fd < 0)
423*d83cc019SAndroid Build Coastguard Worker 			continue;
424*d83cc019SAndroid Build Coastguard Worker 
425*d83cc019SAndroid Build Coastguard Worker 		/*
426*d83cc019SAndroid Build Coastguard Worker 		 * BIT(0): Drop page cache
427*d83cc019SAndroid Build Coastguard Worker 		 * BIT(1): Drop slab cache
428*d83cc019SAndroid Build Coastguard Worker 		 */
429*d83cc019SAndroid Build Coastguard Worker 		igt_ignore_warn(write(fd, "3\n", 2));
430*d83cc019SAndroid Build Coastguard Worker 		close(fd);
431*d83cc019SAndroid Build Coastguard Worker 	}
432*d83cc019SAndroid Build Coastguard Worker 
433*d83cc019SAndroid Build Coastguard Worker 	errno = 0;
434*d83cc019SAndroid Build Coastguard Worker }
435*d83cc019SAndroid Build Coastguard Worker 
436*d83cc019SAndroid Build Coastguard Worker /*
437*d83cc019SAndroid Build Coastguard Worker  * When testing a port to a new platform, create a standalone test binary
438*d83cc019SAndroid Build Coastguard Worker  * by running:
439*d83cc019SAndroid Build Coastguard Worker  * cc -o porttest intel_drm.c -I.. -DSTANDALONE_TEST `pkg-config --cflags libdrm`
440*d83cc019SAndroid Build Coastguard Worker  * and then running the resulting porttest program.
441*d83cc019SAndroid Build Coastguard Worker  */
442*d83cc019SAndroid Build Coastguard Worker #ifdef STANDALONE_TEST
443*d83cc019SAndroid Build Coastguard Worker void *mmio;
444*d83cc019SAndroid Build Coastguard Worker 
main(int argc,char ** argv)445*d83cc019SAndroid Build Coastguard Worker int main(int argc, char **argv)
446*d83cc019SAndroid Build Coastguard Worker {
447*d83cc019SAndroid Build Coastguard Worker     igt_info("Total RAM:  %"PRIu64" Mb\n", intel_get_total_ram_mb());
448*d83cc019SAndroid Build Coastguard Worker     igt_info("Total Swap: %"PRIu64" Mb\n", intel_get_total_swap_mb());
449*d83cc019SAndroid Build Coastguard Worker 
450*d83cc019SAndroid Build Coastguard Worker     return 0;
451*d83cc019SAndroid Build Coastguard Worker }
452*d83cc019SAndroid Build Coastguard Worker #endif /* STANDALONE_TEST */
453