#define _GNU_SOURCE
#include <stdio.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <fcntl.h>
#include <stdint.h>
#include <unistd.h>
#include <string.h>
#include <sys/mman.h>
#include "vm_util.h"
#include "kselftest.h"
void run_dio_using_hugetlb(unsigned int start_off, unsigned int end_off)
{
int fd;
char *buffer = NULL;
char *orig_buffer = NULL;
size_t h_pagesize = 0;
size_t writesize;
int free_hpage_b = 0;
int free_hpage_a = 0;
const int mmap_flags = MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB;
const int mmap_prot = PROT_READ | PROT_WRITE;
writesize = end_off - start_off;
h_pagesize = default_huge_page_size();
if (!h_pagesize)
ksft_exit_fail_msg("Unable to determine huge page size\n");
fd = open("/tmp", O_TMPFILE | O_RDWR | O_DIRECT, 0664);
if (fd < 0)
ksft_exit_fail_perror("Error opening file\n");
free_hpage_b = get_free_hugepages();
if (free_hpage_b == 0) {
close(fd);
ksft_exit_skip("No free hugepage, exiting!\n");
}
orig_buffer = mmap(NULL, h_pagesize, mmap_prot, mmap_flags, -1, 0);
if (orig_buffer == MAP_FAILED) {
close(fd);
ksft_exit_fail_perror("Error mapping memory\n");
}
buffer = orig_buffer;
buffer += start_off;
memset(buffer, 'A', writesize);
if (write(fd, buffer, writesize) != (writesize)) {
munmap(orig_buffer, h_pagesize);
close(fd);
ksft_exit_fail_perror("Error writing to file\n");
}
munmap(orig_buffer, h_pagesize);
close(fd);
free_hpage_a = get_free_hugepages();
ksft_print_msg("No. Free pages before allocation : %d\n", free_hpage_b);
ksft_print_msg("No. Free pages after munmap : %d\n", free_hpage_a);
ksft_test_result(free_hpage_a == free_hpage_b,
"free huge pages from %u-%u\n", start_off, end_off);
}
int main(void)
{
size_t pagesize = 0;
int fd;
ksft_print_header();
fd = open("/tmp", O_TMPFILE | O_RDWR | O_DIRECT, 0664);
if (fd < 0)
ksft_exit_skip("Unable to allocate file: %s\n", strerror(errno));
close(fd);
if (!get_free_hugepages())
ksft_exit_skip("No free hugepage, exiting\n");
ksft_set_plan(4);
pagesize = psize();
run_dio_using_hugetlb(0, (pagesize * 3));
run_dio_using_hugetlb(0, (pagesize * 3) - (pagesize / 2));
run_dio_using_hugetlb(pagesize / 2, (pagesize * 3));
run_dio_using_hugetlb(pagesize / 2, (pagesize * 3) + (pagesize / 2));
ksft_finished();
}