1*ba91e7e5SJiaqi Yan // SPDX-License-Identifier: GPL-2.0
2*ba91e7e5SJiaqi Yan
3*ba91e7e5SJiaqi Yan #define _GNU_SOURCE
4*ba91e7e5SJiaqi Yan #include <stdlib.h>
5*ba91e7e5SJiaqi Yan #include <stdio.h>
6*ba91e7e5SJiaqi Yan #include <string.h>
7*ba91e7e5SJiaqi Yan
8*ba91e7e5SJiaqi Yan #include <linux/magic.h>
9*ba91e7e5SJiaqi Yan #include <sys/mman.h>
10*ba91e7e5SJiaqi Yan #include <sys/statfs.h>
11*ba91e7e5SJiaqi Yan #include <errno.h>
12*ba91e7e5SJiaqi Yan #include <stdbool.h>
13*ba91e7e5SJiaqi Yan
14*ba91e7e5SJiaqi Yan #include "../kselftest.h"
15*ba91e7e5SJiaqi Yan
16*ba91e7e5SJiaqi Yan #define PREFIX " ... "
17*ba91e7e5SJiaqi Yan #define ERROR_PREFIX " !!! "
18*ba91e7e5SJiaqi Yan
19*ba91e7e5SJiaqi Yan #define MAX_WRITE_READ_CHUNK_SIZE (getpagesize() * 16)
20*ba91e7e5SJiaqi Yan #define MAX(a, b) (((a) > (b)) ? (a) : (b))
21*ba91e7e5SJiaqi Yan
22*ba91e7e5SJiaqi Yan enum test_status {
23*ba91e7e5SJiaqi Yan TEST_PASSED = 0,
24*ba91e7e5SJiaqi Yan TEST_FAILED = 1,
25*ba91e7e5SJiaqi Yan TEST_SKIPPED = 2,
26*ba91e7e5SJiaqi Yan };
27*ba91e7e5SJiaqi Yan
status_to_str(enum test_status status)28*ba91e7e5SJiaqi Yan static char *status_to_str(enum test_status status)
29*ba91e7e5SJiaqi Yan {
30*ba91e7e5SJiaqi Yan switch (status) {
31*ba91e7e5SJiaqi Yan case TEST_PASSED:
32*ba91e7e5SJiaqi Yan return "TEST_PASSED";
33*ba91e7e5SJiaqi Yan case TEST_FAILED:
34*ba91e7e5SJiaqi Yan return "TEST_FAILED";
35*ba91e7e5SJiaqi Yan case TEST_SKIPPED:
36*ba91e7e5SJiaqi Yan return "TEST_SKIPPED";
37*ba91e7e5SJiaqi Yan default:
38*ba91e7e5SJiaqi Yan return "TEST_???";
39*ba91e7e5SJiaqi Yan }
40*ba91e7e5SJiaqi Yan }
41*ba91e7e5SJiaqi Yan
setup_filemap(char * filemap,size_t len,size_t wr_chunk_size)42*ba91e7e5SJiaqi Yan static int setup_filemap(char *filemap, size_t len, size_t wr_chunk_size)
43*ba91e7e5SJiaqi Yan {
44*ba91e7e5SJiaqi Yan char iter = 0;
45*ba91e7e5SJiaqi Yan
46*ba91e7e5SJiaqi Yan for (size_t offset = 0; offset < len;
47*ba91e7e5SJiaqi Yan offset += wr_chunk_size) {
48*ba91e7e5SJiaqi Yan iter++;
49*ba91e7e5SJiaqi Yan memset(filemap + offset, iter, wr_chunk_size);
50*ba91e7e5SJiaqi Yan }
51*ba91e7e5SJiaqi Yan
52*ba91e7e5SJiaqi Yan return 0;
53*ba91e7e5SJiaqi Yan }
54*ba91e7e5SJiaqi Yan
verify_chunk(char * buf,size_t len,char val)55*ba91e7e5SJiaqi Yan static bool verify_chunk(char *buf, size_t len, char val)
56*ba91e7e5SJiaqi Yan {
57*ba91e7e5SJiaqi Yan size_t i;
58*ba91e7e5SJiaqi Yan
59*ba91e7e5SJiaqi Yan for (i = 0; i < len; ++i) {
60*ba91e7e5SJiaqi Yan if (buf[i] != val) {
61*ba91e7e5SJiaqi Yan printf(PREFIX ERROR_PREFIX "check fail: buf[%lu] = %u != %u\n",
62*ba91e7e5SJiaqi Yan i, buf[i], val);
63*ba91e7e5SJiaqi Yan return false;
64*ba91e7e5SJiaqi Yan }
65*ba91e7e5SJiaqi Yan }
66*ba91e7e5SJiaqi Yan
67*ba91e7e5SJiaqi Yan return true;
68*ba91e7e5SJiaqi Yan }
69*ba91e7e5SJiaqi Yan
seek_read_hugepage_filemap(int fd,size_t len,size_t wr_chunk_size,off_t offset,size_t expected)70*ba91e7e5SJiaqi Yan static bool seek_read_hugepage_filemap(int fd, size_t len, size_t wr_chunk_size,
71*ba91e7e5SJiaqi Yan off_t offset, size_t expected)
72*ba91e7e5SJiaqi Yan {
73*ba91e7e5SJiaqi Yan char buf[MAX_WRITE_READ_CHUNK_SIZE];
74*ba91e7e5SJiaqi Yan ssize_t ret_count = 0;
75*ba91e7e5SJiaqi Yan ssize_t total_ret_count = 0;
76*ba91e7e5SJiaqi Yan char val = offset / wr_chunk_size + offset % wr_chunk_size;
77*ba91e7e5SJiaqi Yan
78*ba91e7e5SJiaqi Yan printf(PREFIX PREFIX "init val=%u with offset=0x%lx\n", val, offset);
79*ba91e7e5SJiaqi Yan printf(PREFIX PREFIX "expect to read 0x%lx bytes of data in total\n",
80*ba91e7e5SJiaqi Yan expected);
81*ba91e7e5SJiaqi Yan if (lseek(fd, offset, SEEK_SET) < 0) {
82*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "seek failed");
83*ba91e7e5SJiaqi Yan return false;
84*ba91e7e5SJiaqi Yan }
85*ba91e7e5SJiaqi Yan
86*ba91e7e5SJiaqi Yan while (offset + total_ret_count < len) {
87*ba91e7e5SJiaqi Yan ret_count = read(fd, buf, wr_chunk_size);
88*ba91e7e5SJiaqi Yan if (ret_count == 0) {
89*ba91e7e5SJiaqi Yan printf(PREFIX PREFIX "read reach end of the file\n");
90*ba91e7e5SJiaqi Yan break;
91*ba91e7e5SJiaqi Yan } else if (ret_count < 0) {
92*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "read failed");
93*ba91e7e5SJiaqi Yan break;
94*ba91e7e5SJiaqi Yan }
95*ba91e7e5SJiaqi Yan ++val;
96*ba91e7e5SJiaqi Yan if (!verify_chunk(buf, ret_count, val))
97*ba91e7e5SJiaqi Yan return false;
98*ba91e7e5SJiaqi Yan
99*ba91e7e5SJiaqi Yan total_ret_count += ret_count;
100*ba91e7e5SJiaqi Yan }
101*ba91e7e5SJiaqi Yan printf(PREFIX PREFIX "actually read 0x%lx bytes of data in total\n",
102*ba91e7e5SJiaqi Yan total_ret_count);
103*ba91e7e5SJiaqi Yan
104*ba91e7e5SJiaqi Yan return total_ret_count == expected;
105*ba91e7e5SJiaqi Yan }
106*ba91e7e5SJiaqi Yan
read_hugepage_filemap(int fd,size_t len,size_t wr_chunk_size,size_t expected)107*ba91e7e5SJiaqi Yan static bool read_hugepage_filemap(int fd, size_t len,
108*ba91e7e5SJiaqi Yan size_t wr_chunk_size, size_t expected)
109*ba91e7e5SJiaqi Yan {
110*ba91e7e5SJiaqi Yan char buf[MAX_WRITE_READ_CHUNK_SIZE];
111*ba91e7e5SJiaqi Yan ssize_t ret_count = 0;
112*ba91e7e5SJiaqi Yan ssize_t total_ret_count = 0;
113*ba91e7e5SJiaqi Yan char val = 0;
114*ba91e7e5SJiaqi Yan
115*ba91e7e5SJiaqi Yan printf(PREFIX PREFIX "expect to read 0x%lx bytes of data in total\n",
116*ba91e7e5SJiaqi Yan expected);
117*ba91e7e5SJiaqi Yan while (total_ret_count < len) {
118*ba91e7e5SJiaqi Yan ret_count = read(fd, buf, wr_chunk_size);
119*ba91e7e5SJiaqi Yan if (ret_count == 0) {
120*ba91e7e5SJiaqi Yan printf(PREFIX PREFIX "read reach end of the file\n");
121*ba91e7e5SJiaqi Yan break;
122*ba91e7e5SJiaqi Yan } else if (ret_count < 0) {
123*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "read failed");
124*ba91e7e5SJiaqi Yan break;
125*ba91e7e5SJiaqi Yan }
126*ba91e7e5SJiaqi Yan ++val;
127*ba91e7e5SJiaqi Yan if (!verify_chunk(buf, ret_count, val))
128*ba91e7e5SJiaqi Yan return false;
129*ba91e7e5SJiaqi Yan
130*ba91e7e5SJiaqi Yan total_ret_count += ret_count;
131*ba91e7e5SJiaqi Yan }
132*ba91e7e5SJiaqi Yan printf(PREFIX PREFIX "actually read 0x%lx bytes of data in total\n",
133*ba91e7e5SJiaqi Yan total_ret_count);
134*ba91e7e5SJiaqi Yan
135*ba91e7e5SJiaqi Yan return total_ret_count == expected;
136*ba91e7e5SJiaqi Yan }
137*ba91e7e5SJiaqi Yan
138*ba91e7e5SJiaqi Yan static enum test_status
test_hugetlb_read(int fd,size_t len,size_t wr_chunk_size)139*ba91e7e5SJiaqi Yan test_hugetlb_read(int fd, size_t len, size_t wr_chunk_size)
140*ba91e7e5SJiaqi Yan {
141*ba91e7e5SJiaqi Yan enum test_status status = TEST_SKIPPED;
142*ba91e7e5SJiaqi Yan char *filemap = NULL;
143*ba91e7e5SJiaqi Yan
144*ba91e7e5SJiaqi Yan if (ftruncate(fd, len) < 0) {
145*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "ftruncate failed");
146*ba91e7e5SJiaqi Yan return status;
147*ba91e7e5SJiaqi Yan }
148*ba91e7e5SJiaqi Yan
149*ba91e7e5SJiaqi Yan filemap = mmap(NULL, len, PROT_READ | PROT_WRITE,
150*ba91e7e5SJiaqi Yan MAP_SHARED | MAP_POPULATE, fd, 0);
151*ba91e7e5SJiaqi Yan if (filemap == MAP_FAILED) {
152*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "mmap for primary mapping failed");
153*ba91e7e5SJiaqi Yan goto done;
154*ba91e7e5SJiaqi Yan }
155*ba91e7e5SJiaqi Yan
156*ba91e7e5SJiaqi Yan setup_filemap(filemap, len, wr_chunk_size);
157*ba91e7e5SJiaqi Yan status = TEST_FAILED;
158*ba91e7e5SJiaqi Yan
159*ba91e7e5SJiaqi Yan if (read_hugepage_filemap(fd, len, wr_chunk_size, len))
160*ba91e7e5SJiaqi Yan status = TEST_PASSED;
161*ba91e7e5SJiaqi Yan
162*ba91e7e5SJiaqi Yan munmap(filemap, len);
163*ba91e7e5SJiaqi Yan done:
164*ba91e7e5SJiaqi Yan if (ftruncate(fd, 0) < 0) {
165*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "ftruncate back to 0 failed");
166*ba91e7e5SJiaqi Yan status = TEST_FAILED;
167*ba91e7e5SJiaqi Yan }
168*ba91e7e5SJiaqi Yan
169*ba91e7e5SJiaqi Yan return status;
170*ba91e7e5SJiaqi Yan }
171*ba91e7e5SJiaqi Yan
172*ba91e7e5SJiaqi Yan static enum test_status
test_hugetlb_read_hwpoison(int fd,size_t len,size_t wr_chunk_size,bool skip_hwpoison_page)173*ba91e7e5SJiaqi Yan test_hugetlb_read_hwpoison(int fd, size_t len, size_t wr_chunk_size,
174*ba91e7e5SJiaqi Yan bool skip_hwpoison_page)
175*ba91e7e5SJiaqi Yan {
176*ba91e7e5SJiaqi Yan enum test_status status = TEST_SKIPPED;
177*ba91e7e5SJiaqi Yan char *filemap = NULL;
178*ba91e7e5SJiaqi Yan char *hwp_addr = NULL;
179*ba91e7e5SJiaqi Yan const unsigned long pagesize = getpagesize();
180*ba91e7e5SJiaqi Yan
181*ba91e7e5SJiaqi Yan if (ftruncate(fd, len) < 0) {
182*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "ftruncate failed");
183*ba91e7e5SJiaqi Yan return status;
184*ba91e7e5SJiaqi Yan }
185*ba91e7e5SJiaqi Yan
186*ba91e7e5SJiaqi Yan filemap = mmap(NULL, len, PROT_READ | PROT_WRITE,
187*ba91e7e5SJiaqi Yan MAP_SHARED | MAP_POPULATE, fd, 0);
188*ba91e7e5SJiaqi Yan if (filemap == MAP_FAILED) {
189*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "mmap for primary mapping failed");
190*ba91e7e5SJiaqi Yan goto done;
191*ba91e7e5SJiaqi Yan }
192*ba91e7e5SJiaqi Yan
193*ba91e7e5SJiaqi Yan setup_filemap(filemap, len, wr_chunk_size);
194*ba91e7e5SJiaqi Yan status = TEST_FAILED;
195*ba91e7e5SJiaqi Yan
196*ba91e7e5SJiaqi Yan /*
197*ba91e7e5SJiaqi Yan * Poisoned hugetlb page layout (assume hugepagesize=2MB):
198*ba91e7e5SJiaqi Yan * |<---------------------- 1MB ---------------------->|
199*ba91e7e5SJiaqi Yan * |<---- healthy page ---->|<---- HWPOISON page ----->|
200*ba91e7e5SJiaqi Yan * |<------------------- (1MB - 8KB) ----------------->|
201*ba91e7e5SJiaqi Yan */
202*ba91e7e5SJiaqi Yan hwp_addr = filemap + len / 2 + pagesize;
203*ba91e7e5SJiaqi Yan if (madvise(hwp_addr, pagesize, MADV_HWPOISON) < 0) {
204*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "MADV_HWPOISON failed");
205*ba91e7e5SJiaqi Yan goto unmap;
206*ba91e7e5SJiaqi Yan }
207*ba91e7e5SJiaqi Yan
208*ba91e7e5SJiaqi Yan if (!skip_hwpoison_page) {
209*ba91e7e5SJiaqi Yan /*
210*ba91e7e5SJiaqi Yan * Userspace should be able to read (1MB + 1 page) from
211*ba91e7e5SJiaqi Yan * the beginning of the HWPOISONed hugepage.
212*ba91e7e5SJiaqi Yan */
213*ba91e7e5SJiaqi Yan if (read_hugepage_filemap(fd, len, wr_chunk_size,
214*ba91e7e5SJiaqi Yan len / 2 + pagesize))
215*ba91e7e5SJiaqi Yan status = TEST_PASSED;
216*ba91e7e5SJiaqi Yan } else {
217*ba91e7e5SJiaqi Yan /*
218*ba91e7e5SJiaqi Yan * Userspace should be able to read (1MB - 2 pages) from
219*ba91e7e5SJiaqi Yan * HWPOISONed hugepage.
220*ba91e7e5SJiaqi Yan */
221*ba91e7e5SJiaqi Yan if (seek_read_hugepage_filemap(fd, len, wr_chunk_size,
222*ba91e7e5SJiaqi Yan len / 2 + MAX(2 * pagesize, wr_chunk_size),
223*ba91e7e5SJiaqi Yan len / 2 - MAX(2 * pagesize, wr_chunk_size)))
224*ba91e7e5SJiaqi Yan status = TEST_PASSED;
225*ba91e7e5SJiaqi Yan }
226*ba91e7e5SJiaqi Yan
227*ba91e7e5SJiaqi Yan unmap:
228*ba91e7e5SJiaqi Yan munmap(filemap, len);
229*ba91e7e5SJiaqi Yan done:
230*ba91e7e5SJiaqi Yan if (ftruncate(fd, 0) < 0) {
231*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "ftruncate back to 0 failed");
232*ba91e7e5SJiaqi Yan status = TEST_FAILED;
233*ba91e7e5SJiaqi Yan }
234*ba91e7e5SJiaqi Yan
235*ba91e7e5SJiaqi Yan return status;
236*ba91e7e5SJiaqi Yan }
237*ba91e7e5SJiaqi Yan
create_hugetlbfs_file(struct statfs * file_stat)238*ba91e7e5SJiaqi Yan static int create_hugetlbfs_file(struct statfs *file_stat)
239*ba91e7e5SJiaqi Yan {
240*ba91e7e5SJiaqi Yan int fd;
241*ba91e7e5SJiaqi Yan
242*ba91e7e5SJiaqi Yan fd = memfd_create("hugetlb_tmp", MFD_HUGETLB);
243*ba91e7e5SJiaqi Yan if (fd < 0) {
244*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "could not open hugetlbfs file");
245*ba91e7e5SJiaqi Yan return -1;
246*ba91e7e5SJiaqi Yan }
247*ba91e7e5SJiaqi Yan
248*ba91e7e5SJiaqi Yan memset(file_stat, 0, sizeof(*file_stat));
249*ba91e7e5SJiaqi Yan if (fstatfs(fd, file_stat)) {
250*ba91e7e5SJiaqi Yan perror(PREFIX ERROR_PREFIX "fstatfs failed");
251*ba91e7e5SJiaqi Yan goto close;
252*ba91e7e5SJiaqi Yan }
253*ba91e7e5SJiaqi Yan if (file_stat->f_type != HUGETLBFS_MAGIC) {
254*ba91e7e5SJiaqi Yan printf(PREFIX ERROR_PREFIX "not hugetlbfs file\n");
255*ba91e7e5SJiaqi Yan goto close;
256*ba91e7e5SJiaqi Yan }
257*ba91e7e5SJiaqi Yan
258*ba91e7e5SJiaqi Yan return fd;
259*ba91e7e5SJiaqi Yan close:
260*ba91e7e5SJiaqi Yan close(fd);
261*ba91e7e5SJiaqi Yan return -1;
262*ba91e7e5SJiaqi Yan }
263*ba91e7e5SJiaqi Yan
main(void)264*ba91e7e5SJiaqi Yan int main(void)
265*ba91e7e5SJiaqi Yan {
266*ba91e7e5SJiaqi Yan int fd;
267*ba91e7e5SJiaqi Yan struct statfs file_stat;
268*ba91e7e5SJiaqi Yan enum test_status status;
269*ba91e7e5SJiaqi Yan /* Test read() in different granularity. */
270*ba91e7e5SJiaqi Yan size_t wr_chunk_sizes[] = {
271*ba91e7e5SJiaqi Yan getpagesize() / 2, getpagesize(),
272*ba91e7e5SJiaqi Yan getpagesize() * 2, getpagesize() * 4
273*ba91e7e5SJiaqi Yan };
274*ba91e7e5SJiaqi Yan size_t i;
275*ba91e7e5SJiaqi Yan
276*ba91e7e5SJiaqi Yan for (i = 0; i < ARRAY_SIZE(wr_chunk_sizes); ++i) {
277*ba91e7e5SJiaqi Yan printf("Write/read chunk size=0x%lx\n",
278*ba91e7e5SJiaqi Yan wr_chunk_sizes[i]);
279*ba91e7e5SJiaqi Yan
280*ba91e7e5SJiaqi Yan fd = create_hugetlbfs_file(&file_stat);
281*ba91e7e5SJiaqi Yan if (fd < 0)
282*ba91e7e5SJiaqi Yan goto create_failure;
283*ba91e7e5SJiaqi Yan printf(PREFIX "HugeTLB read regression test...\n");
284*ba91e7e5SJiaqi Yan status = test_hugetlb_read(fd, file_stat.f_bsize,
285*ba91e7e5SJiaqi Yan wr_chunk_sizes[i]);
286*ba91e7e5SJiaqi Yan printf(PREFIX "HugeTLB read regression test...%s\n",
287*ba91e7e5SJiaqi Yan status_to_str(status));
288*ba91e7e5SJiaqi Yan close(fd);
289*ba91e7e5SJiaqi Yan if (status == TEST_FAILED)
290*ba91e7e5SJiaqi Yan return -1;
291*ba91e7e5SJiaqi Yan
292*ba91e7e5SJiaqi Yan fd = create_hugetlbfs_file(&file_stat);
293*ba91e7e5SJiaqi Yan if (fd < 0)
294*ba91e7e5SJiaqi Yan goto create_failure;
295*ba91e7e5SJiaqi Yan printf(PREFIX "HugeTLB read HWPOISON test...\n");
296*ba91e7e5SJiaqi Yan status = test_hugetlb_read_hwpoison(fd, file_stat.f_bsize,
297*ba91e7e5SJiaqi Yan wr_chunk_sizes[i], false);
298*ba91e7e5SJiaqi Yan printf(PREFIX "HugeTLB read HWPOISON test...%s\n",
299*ba91e7e5SJiaqi Yan status_to_str(status));
300*ba91e7e5SJiaqi Yan close(fd);
301*ba91e7e5SJiaqi Yan if (status == TEST_FAILED)
302*ba91e7e5SJiaqi Yan return -1;
303*ba91e7e5SJiaqi Yan
304*ba91e7e5SJiaqi Yan fd = create_hugetlbfs_file(&file_stat);
305*ba91e7e5SJiaqi Yan if (fd < 0)
306*ba91e7e5SJiaqi Yan goto create_failure;
307*ba91e7e5SJiaqi Yan printf(PREFIX "HugeTLB seek then read HWPOISON test...\n");
308*ba91e7e5SJiaqi Yan status = test_hugetlb_read_hwpoison(fd, file_stat.f_bsize,
309*ba91e7e5SJiaqi Yan wr_chunk_sizes[i], true);
310*ba91e7e5SJiaqi Yan printf(PREFIX "HugeTLB seek then read HWPOISON test...%s\n",
311*ba91e7e5SJiaqi Yan status_to_str(status));
312*ba91e7e5SJiaqi Yan close(fd);
313*ba91e7e5SJiaqi Yan if (status == TEST_FAILED)
314*ba91e7e5SJiaqi Yan return -1;
315*ba91e7e5SJiaqi Yan }
316*ba91e7e5SJiaqi Yan
317*ba91e7e5SJiaqi Yan return 0;
318*ba91e7e5SJiaqi Yan
319*ba91e7e5SJiaqi Yan create_failure:
320*ba91e7e5SJiaqi Yan printf(ERROR_PREFIX "Abort test: failed to create hugetlbfs file\n");
321*ba91e7e5SJiaqi Yan return -1;
322*ba91e7e5SJiaqi Yan }
323