hc
2023-02-18 84d1d1bed0120e0921c876885ca9006fb9ddf42e
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
#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;
   __u64 expansion[10];    /* For future use */
};
 
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;
           break;
       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;
}