// SPDX-License-Identifier: GPL-2.0 /* * Example of using hugepage memory in a user application using the mmap * system call with MAP_HUGETLB flag. Before running this program make * sure the administrator has allocated enough default sized huge pages * to cover the 256 MB allocation. * * For ia64 architecture, Linux kernel reserves Region number 4 for hugepages. * That means the addresses starting with 0x800000... will need to be * specified. Specifying a fixed address is not required on ppc64, i386 * or x86_64. */ #include #include #include #include #include #include "vm_util.h" #include "../kselftest.h" #define LENGTH (256UL*1024*1024) #define PROTECTION (PROT_READ | PROT_WRITE) /* Only ia64 requires this */ #ifdef __ia64__ #define ADDR (void *)(0x8000000000000000UL) #define FLAGS (MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB | MAP_FIXED) #else #define ADDR (void *)(0x0UL) #define FLAGS (MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB) #endif static void check_bytes(char *addr) { ksft_print_msg("First hex is %x\n", *((unsigned int *)addr)); } static void write_bytes(char *addr, size_t length) { unsigned long i; for (i = 0; i < length; i++) *(addr + i) = (char)i; } static void read_bytes(char *addr, size_t length) { unsigned long i; check_bytes(addr); for (i = 0; i < length; i++) if (*(addr + i) != (char)i) ksft_exit_fail_msg("Mismatch at %lu\n", i); ksft_test_result_pass("Read correct data\n"); } int main(int argc, char **argv) { void *addr; size_t hugepage_size; size_t length = LENGTH; int flags = FLAGS; int shift = 0; hugepage_size = default_huge_page_size(); /* munmap with fail if the length is not page aligned */ if (hugepage_size > length) length = hugepage_size; ksft_print_header(); ksft_set_plan(1); if (argc > 1) length = atol(argv[1]) << 20; if (argc > 2) { shift = atoi(argv[2]); if (shift) flags |= (shift & MAP_HUGE_MASK) << MAP_HUGE_SHIFT; } if (shift) ksft_print_msg("%u kB hugepages\n", 1 << (shift - 10)); else ksft_print_msg("Default size hugepages\n"); ksft_print_msg("Mapping %lu Mbytes\n", (unsigned long)length >> 20); addr = mmap(ADDR, length, PROTECTION, flags, -1, 0); if (addr == MAP_FAILED) ksft_exit_fail_msg("mmap: %s\n", strerror(errno)); ksft_print_msg("Returned address is %p\n", addr); check_bytes(addr); write_bytes(addr, length); read_bytes(addr, length); /* munmap() length of MAP_HUGETLB memory must be hugepage aligned */ if (munmap(addr, length)) ksft_exit_fail_msg("munmap: %s\n", strerror(errno)); ksft_finished(); }