diff options
Diffstat (limited to 'tools/testing/selftests/mm/vm_util.c')
| -rw-r--r-- | tools/testing/selftests/mm/vm_util.c | 180 | 
1 files changed, 166 insertions, 14 deletions
diff --git a/tools/testing/selftests/mm/vm_util.c b/tools/testing/selftests/mm/vm_util.c index 40e795624ff3..9b06a5034808 100644 --- a/tools/testing/selftests/mm/vm_util.c +++ b/tools/testing/selftests/mm/vm_util.c @@ -1,6 +1,10 @@  // SPDX-License-Identifier: GPL-2.0  #include <string.h>  #include <fcntl.h> +#include <sys/ioctl.h> +#include <linux/userfaultfd.h> +#include <sys/syscall.h> +#include <unistd.h>  #include "../kselftest.h"  #include "vm_util.h" @@ -8,6 +12,9 @@  #define SMAP_FILE_PATH "/proc/self/smaps"  #define MAX_LINE_LENGTH 500 +unsigned int __page_size; +unsigned int __page_shift; +  uint64_t pagemap_get_entry(int fd, char *start)  {  	const unsigned long pfn = (unsigned long)start / getpagesize(); @@ -22,25 +29,17 @@ uint64_t pagemap_get_entry(int fd, char *start)  bool pagemap_is_softdirty(int fd, char *start)  { -	uint64_t entry = pagemap_get_entry(fd, start); - -	// Check if dirty bit (55th bit) is set -	return entry & 0x0080000000000000ull; +	return pagemap_get_entry(fd, start) & PM_SOFT_DIRTY;  }  bool pagemap_is_swapped(int fd, char *start)  { -	uint64_t entry = pagemap_get_entry(fd, start); - -	return entry & 0x4000000000000000ull; +	return pagemap_get_entry(fd, start) & PM_SWAP;  }  bool pagemap_is_populated(int fd, char *start)  { -	uint64_t entry = pagemap_get_entry(fd, start); - -	/* Present or swapped. */ -	return entry & 0xc000000000000000ull; +	return pagemap_get_entry(fd, start) & (PM_PRESENT | PM_SWAP);  }  unsigned long pagemap_get_pfn(int fd, char *start) @@ -48,7 +47,7 @@ unsigned long pagemap_get_pfn(int fd, char *start)  	uint64_t entry = pagemap_get_entry(fd, start);  	/* If present (63th bit), PFN is at bit 0 -- 54. */ -	if (entry & 0x8000000000000000ull) +	if (entry & PM_PRESENT)  		return entry & 0x007fffffffffffffull;  	return -1ul;  } @@ -84,12 +83,12 @@ uint64_t read_pmd_pagesize(void)  	fd = open(PMD_SIZE_FILE_PATH, O_RDONLY);  	if (fd == -1) -		ksft_exit_fail_msg("Open hpage_pmd_size failed\n"); +		return 0;  	num_read = read(fd, buf, 19);  	if (num_read < 1) {  		close(fd); -		ksft_exit_fail_msg("Read hpage_pmd_size failed\n"); +		return 0;  	}  	buf[num_read] = '\0';  	close(fd); @@ -149,3 +148,156 @@ bool check_huge_shmem(void *addr, int nr_hpages, uint64_t hpage_size)  {  	return __check_huge(addr, "ShmemPmdMapped:", nr_hpages, hpage_size);  } + +int64_t allocate_transhuge(void *ptr, int pagemap_fd) +{ +	uint64_t ent[2]; + +	/* drop pmd */ +	if (mmap(ptr, HPAGE_SIZE, PROT_READ | PROT_WRITE, +		 MAP_FIXED | MAP_ANONYMOUS | +		 MAP_NORESERVE | MAP_PRIVATE, -1, 0) != ptr) +		errx(2, "mmap transhuge"); + +	if (madvise(ptr, HPAGE_SIZE, MADV_HUGEPAGE)) +		err(2, "MADV_HUGEPAGE"); + +	/* allocate transparent huge page */ +	*(volatile void **)ptr = ptr; + +	if (pread(pagemap_fd, ent, sizeof(ent), +		  (uintptr_t)ptr >> (pshift() - 3)) != sizeof(ent)) +		err(2, "read pagemap"); + +	if (PAGEMAP_PRESENT(ent[0]) && PAGEMAP_PRESENT(ent[1]) && +	    PAGEMAP_PFN(ent[0]) + 1 == PAGEMAP_PFN(ent[1]) && +	    !(PAGEMAP_PFN(ent[0]) & ((1 << (HPAGE_SHIFT - pshift())) - 1))) +		return PAGEMAP_PFN(ent[0]); + +	return -1; +} + +unsigned long default_huge_page_size(void) +{ +	unsigned long hps = 0; +	char *line = NULL; +	size_t linelen = 0; +	FILE *f = fopen("/proc/meminfo", "r"); + +	if (!f) +		return 0; +	while (getline(&line, &linelen, f) > 0) { +		if (sscanf(line, "Hugepagesize:       %lu kB", &hps) == 1) { +			hps <<= 10; +			break; +		} +	} + +	free(line); +	fclose(f); +	return hps; +} + +/* If `ioctls' non-NULL, the allowed ioctls will be returned into the var */ +int uffd_register_with_ioctls(int uffd, void *addr, uint64_t len, +			      bool miss, bool wp, bool minor, uint64_t *ioctls) +{ +	struct uffdio_register uffdio_register = { 0 }; +	uint64_t mode = 0; +	int ret = 0; + +	if (miss) +		mode |= UFFDIO_REGISTER_MODE_MISSING; +	if (wp) +		mode |= UFFDIO_REGISTER_MODE_WP; +	if (minor) +		mode |= UFFDIO_REGISTER_MODE_MINOR; + +	uffdio_register.range.start = (unsigned long)addr; +	uffdio_register.range.len = len; +	uffdio_register.mode = mode; + +	if (ioctl(uffd, UFFDIO_REGISTER, &uffdio_register) == -1) +		ret = -errno; +	else if (ioctls) +		*ioctls = uffdio_register.ioctls; + +	return ret; +} + +int uffd_register(int uffd, void *addr, uint64_t len, +		  bool miss, bool wp, bool minor) +{ +	return uffd_register_with_ioctls(uffd, addr, len, +					 miss, wp, minor, NULL); +} + +int uffd_unregister(int uffd, void *addr, uint64_t len) +{ +	struct uffdio_range range = { .start = (uintptr_t)addr, .len = len }; +	int ret = 0; + +	if (ioctl(uffd, UFFDIO_UNREGISTER, &range) == -1) +		ret = -errno; + +	return ret; +} + +int uffd_open_dev(unsigned int flags) +{ +	int fd, uffd; + +	fd = open("/dev/userfaultfd", O_RDWR | O_CLOEXEC); +	if (fd < 0) +		return fd; +	uffd = ioctl(fd, USERFAULTFD_IOC_NEW, flags); +	close(fd); + +	return uffd; +} + +int uffd_open_sys(unsigned int flags) +{ +#ifdef __NR_userfaultfd +	return syscall(__NR_userfaultfd, flags); +#else +	return -1; +#endif +} + +int uffd_open(unsigned int flags) +{ +	int uffd = uffd_open_sys(flags); + +	if (uffd < 0) +		uffd = uffd_open_dev(flags); + +	return uffd; +} + +int uffd_get_features(uint64_t *features) +{ +	struct uffdio_api uffdio_api = { .api = UFFD_API, .features = 0 }; +	/* +	 * This should by default work in most kernels; the feature list +	 * will be the same no matter what we pass in here. +	 */ +	int fd = uffd_open(UFFD_USER_MODE_ONLY); + +	if (fd < 0) +		/* Maybe the kernel is older than user-only mode? */ +		fd = uffd_open(0); + +	if (fd < 0) +		return fd; + +	if (ioctl(fd, UFFDIO_API, &uffdio_api)) { +		close(fd); +		return -errno; +	} + +	*features = uffdio_api.features; +	close(fd); + +	return 0; +}  |