-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathmincore.c
98 lines (88 loc) · 2.63 KB
/
mincore.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
/*
* Copyright (C) 2013-2018 Canonical, Ltd.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* This code is a complete clean re-write of the stress tool by
* Colin Ian King <[email protected]> and attempts to be
* backwardly compatible with the stress tool by Amos Waterland
* <[email protected]> but has more stress tests and more
* functionality.
*
*/
#include "stress-ng.h"
/*
* mincore_touch_pages_slow()
* touch pages, even when they are resident
*/
static void mincore_touch_pages_slow(
void *buf,
const size_t n_pages,
const size_t page_size)
{
size_t i;
volatile char *buffer;
for (buffer = buf, i = 0; i < n_pages; i++, buffer += page_size)
(*buffer)++;
for (buffer = buf, i = 0; i < n_pages; i++, buffer += page_size)
(*buffer)--;
}
/*
* mincore_touch_pages()
* touch a range of pages, ensure they are all in memory
*/
int mincore_touch_pages(void *buf, const size_t buf_len)
{
const size_t page_size = stress_get_pagesize();
const size_t n_pages = buf_len / page_size;
#if !defined(HAVE_MINCORE)
/* systems that don't have mincore */
mincore_touch_pages_slow(buf, n_pages, page_size);
return 0;
#else
/* systems that support mincore */
unsigned char *vec;
volatile char *buffer;
size_t i;
uintptr_t uintptr = (uintptr_t)buf & (page_size - 1);
if (!(g_opt_flags & OPT_FLAGS_MMAP_MINCORE))
return 0;
if (n_pages < 1)
return -1;
vec = calloc(n_pages, 1);
if (!vec) {
mincore_touch_pages_slow(buf, n_pages, page_size);
return 0;
}
/*
* Find range of pages that are not in memory
*/
if (shim_mincore((void *)uintptr, buf_len, vec) < 0) {
free(vec);
mincore_touch_pages_slow(buf, n_pages, page_size);
return 0;
}
/* If page is not resident in memory, touch it */
for (buffer = buf, i = 0; i < n_pages; i++, buffer += page_size)
if (!(vec[i] & 1))
(*buffer)++;
/* And restore contents */
for (buffer = buf, i = 0; i < n_pages; i++, buffer += page_size)
if (!(vec[i] & 1))
(*buffer)--;
free(vec);
return 0;
#endif
}