|  | #include <fcntl.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <unistd.h> | 
|  |  | 
|  | #include <sys/ioctl.h> | 
|  | #include <sys/mman.h> | 
|  | #include <sys/prctl.h> | 
|  | #include <sys/stat.h> | 
|  | #include <sys/types.h> | 
|  |  | 
|  | #include <linux/types.h> | 
|  |  | 
|  | #define MB (1UL << 20) | 
|  | #define PAGE_SIZE sysconf(_SC_PAGESIZE) | 
|  |  | 
|  | #define GUP_FAST_BENCHMARK	_IOWR('g', 1, struct gup_benchmark) | 
|  |  | 
|  | struct gup_benchmark { | 
|  | __u64 delta_usec; | 
|  | __u64 addr; | 
|  | __u64 size; | 
|  | __u32 nr_pages_per_call; | 
|  | __u32 flags; | 
|  | }; | 
|  |  | 
|  | int main(int argc, char **argv) | 
|  | { | 
|  | struct gup_benchmark gup; | 
|  | unsigned long size = 128 * MB; | 
|  | int i, fd, opt, nr_pages = 1, thp = -1, repeats = 1, write = 0; | 
|  | char *p; | 
|  |  | 
|  | while ((opt = getopt(argc, argv, "m:r:n:tT")) != -1) { | 
|  | switch (opt) { | 
|  | case 'm': | 
|  | size = atoi(optarg) * MB; | 
|  | break; | 
|  | case 'r': | 
|  | repeats = atoi(optarg); | 
|  | break; | 
|  | case 'n': | 
|  | nr_pages = atoi(optarg); | 
|  | break; | 
|  | case 't': | 
|  | thp = 1; | 
|  | break; | 
|  | case 'T': | 
|  | thp = 0; | 
|  | break; | 
|  | case 'w': | 
|  | write = 1; | 
|  | default: | 
|  | return -1; | 
|  | } | 
|  | } | 
|  |  | 
|  | gup.nr_pages_per_call = nr_pages; | 
|  | gup.flags = write; | 
|  |  | 
|  | fd = open("/sys/kernel/debug/gup_benchmark", O_RDWR); | 
|  | if (fd == -1) | 
|  | perror("open"), exit(1); | 
|  |  | 
|  | p = mmap(NULL, size, PROT_READ | PROT_WRITE, | 
|  | MAP_ANONYMOUS | MAP_PRIVATE, -1, 0); | 
|  | if (p == MAP_FAILED) | 
|  | perror("mmap"), exit(1); | 
|  | gup.addr = (unsigned long)p; | 
|  |  | 
|  | if (thp == 1) | 
|  | madvise(p, size, MADV_HUGEPAGE); | 
|  | else if (thp == 0) | 
|  | madvise(p, size, MADV_NOHUGEPAGE); | 
|  |  | 
|  | for (; (unsigned long)p < gup.addr + size; p += PAGE_SIZE) | 
|  | p[0] = 0; | 
|  |  | 
|  | for (i = 0; i < repeats; i++) { | 
|  | gup.size = size; | 
|  | if (ioctl(fd, GUP_FAST_BENCHMARK, &gup)) | 
|  | perror("ioctl"), exit(1); | 
|  |  | 
|  | printf("Time: %lld us", gup.delta_usec); | 
|  | if (gup.size != size) | 
|  | printf(", truncated (size: %lld)", gup.size); | 
|  | printf("\n"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } |