162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci
362306a36Sopenharmony_ci/*
462306a36Sopenharmony_ci * Tests for mremap w/ MREMAP_DONTUNMAP.
562306a36Sopenharmony_ci *
662306a36Sopenharmony_ci * Copyright 2020, Brian Geffon <bgeffon@google.com>
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci#define _GNU_SOURCE
962306a36Sopenharmony_ci#include <sys/mman.h>
1062306a36Sopenharmony_ci#include <linux/mman.h>
1162306a36Sopenharmony_ci#include <errno.h>
1262306a36Sopenharmony_ci#include <stdio.h>
1362306a36Sopenharmony_ci#include <stdlib.h>
1462306a36Sopenharmony_ci#include <string.h>
1562306a36Sopenharmony_ci#include <unistd.h>
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci#include "../kselftest.h"
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ciunsigned long page_size;
2062306a36Sopenharmony_cichar *page_buffer;
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistatic void dump_maps(void)
2362306a36Sopenharmony_ci{
2462306a36Sopenharmony_ci	char cmd[32];
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_ci	snprintf(cmd, sizeof(cmd), "cat /proc/%d/maps", getpid());
2762306a36Sopenharmony_ci	system(cmd);
2862306a36Sopenharmony_ci}
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci#define BUG_ON(condition, description)					      \
3162306a36Sopenharmony_ci	do {								      \
3262306a36Sopenharmony_ci		if (condition) {					      \
3362306a36Sopenharmony_ci			fprintf(stderr, "[FAIL]\t%s():%d\t%s:%s\n", __func__, \
3462306a36Sopenharmony_ci				__LINE__, (description), strerror(errno));    \
3562306a36Sopenharmony_ci			dump_maps();					  \
3662306a36Sopenharmony_ci			exit(1);					      \
3762306a36Sopenharmony_ci		} 							      \
3862306a36Sopenharmony_ci	} while (0)
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci// Try a simple operation for to "test" for kernel support this prevents
4162306a36Sopenharmony_ci// reporting tests as failed when it's run on an older kernel.
4262306a36Sopenharmony_cistatic int kernel_support_for_mremap_dontunmap()
4362306a36Sopenharmony_ci{
4462306a36Sopenharmony_ci	int ret = 0;
4562306a36Sopenharmony_ci	unsigned long num_pages = 1;
4662306a36Sopenharmony_ci	void *source_mapping = mmap(NULL, num_pages * page_size, PROT_NONE,
4762306a36Sopenharmony_ci				    MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
4862306a36Sopenharmony_ci	BUG_ON(source_mapping == MAP_FAILED, "mmap");
4962306a36Sopenharmony_ci
5062306a36Sopenharmony_ci	// This simple remap should only fail if MREMAP_DONTUNMAP isn't
5162306a36Sopenharmony_ci	// supported.
5262306a36Sopenharmony_ci	void *dest_mapping =
5362306a36Sopenharmony_ci	    mremap(source_mapping, num_pages * page_size, num_pages * page_size,
5462306a36Sopenharmony_ci		   MREMAP_DONTUNMAP | MREMAP_MAYMOVE, 0);
5562306a36Sopenharmony_ci	if (dest_mapping == MAP_FAILED) {
5662306a36Sopenharmony_ci		ret = errno;
5762306a36Sopenharmony_ci	} else {
5862306a36Sopenharmony_ci		BUG_ON(munmap(dest_mapping, num_pages * page_size) == -1,
5962306a36Sopenharmony_ci		       "unable to unmap destination mapping");
6062306a36Sopenharmony_ci	}
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci	BUG_ON(munmap(source_mapping, num_pages * page_size) == -1,
6362306a36Sopenharmony_ci	       "unable to unmap source mapping");
6462306a36Sopenharmony_ci	return ret;
6562306a36Sopenharmony_ci}
6662306a36Sopenharmony_ci
6762306a36Sopenharmony_ci// This helper will just validate that an entire mapping contains the expected
6862306a36Sopenharmony_ci// byte.
6962306a36Sopenharmony_cistatic int check_region_contains_byte(void *addr, unsigned long size, char byte)
7062306a36Sopenharmony_ci{
7162306a36Sopenharmony_ci	BUG_ON(size & (page_size - 1),
7262306a36Sopenharmony_ci	       "check_region_contains_byte expects page multiples");
7362306a36Sopenharmony_ci	BUG_ON((unsigned long)addr & (page_size - 1),
7462306a36Sopenharmony_ci	       "check_region_contains_byte expects page alignment");
7562306a36Sopenharmony_ci
7662306a36Sopenharmony_ci	memset(page_buffer, byte, page_size);
7762306a36Sopenharmony_ci
7862306a36Sopenharmony_ci	unsigned long num_pages = size / page_size;
7962306a36Sopenharmony_ci	unsigned long i;
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci	// Compare each page checking that it contains our expected byte.
8262306a36Sopenharmony_ci	for (i = 0; i < num_pages; ++i) {
8362306a36Sopenharmony_ci		int ret =
8462306a36Sopenharmony_ci		    memcmp(addr + (i * page_size), page_buffer, page_size);
8562306a36Sopenharmony_ci		if (ret) {
8662306a36Sopenharmony_ci			return ret;
8762306a36Sopenharmony_ci		}
8862306a36Sopenharmony_ci	}
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	return 0;
9162306a36Sopenharmony_ci}
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci// this test validates that MREMAP_DONTUNMAP moves the pagetables while leaving
9462306a36Sopenharmony_ci// the source mapping mapped.
9562306a36Sopenharmony_cistatic void mremap_dontunmap_simple()
9662306a36Sopenharmony_ci{
9762306a36Sopenharmony_ci	unsigned long num_pages = 5;
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	void *source_mapping =
10062306a36Sopenharmony_ci	    mmap(NULL, num_pages * page_size, PROT_READ | PROT_WRITE,
10162306a36Sopenharmony_ci		 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
10262306a36Sopenharmony_ci	BUG_ON(source_mapping == MAP_FAILED, "mmap");
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_ci	memset(source_mapping, 'a', num_pages * page_size);
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	// Try to just move the whole mapping anywhere (not fixed).
10762306a36Sopenharmony_ci	void *dest_mapping =
10862306a36Sopenharmony_ci	    mremap(source_mapping, num_pages * page_size, num_pages * page_size,
10962306a36Sopenharmony_ci		   MREMAP_DONTUNMAP | MREMAP_MAYMOVE, NULL);
11062306a36Sopenharmony_ci	BUG_ON(dest_mapping == MAP_FAILED, "mremap");
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ci	// Validate that the pages have been moved, we know they were moved if
11362306a36Sopenharmony_ci	// the dest_mapping contains a's.
11462306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte
11562306a36Sopenharmony_ci	       (dest_mapping, num_pages * page_size, 'a') != 0,
11662306a36Sopenharmony_ci	       "pages did not migrate");
11762306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte
11862306a36Sopenharmony_ci	       (source_mapping, num_pages * page_size, 0) != 0,
11962306a36Sopenharmony_ci	       "source should have no ptes");
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	BUG_ON(munmap(dest_mapping, num_pages * page_size) == -1,
12262306a36Sopenharmony_ci	       "unable to unmap destination mapping");
12362306a36Sopenharmony_ci	BUG_ON(munmap(source_mapping, num_pages * page_size) == -1,
12462306a36Sopenharmony_ci	       "unable to unmap source mapping");
12562306a36Sopenharmony_ci}
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_ci// This test validates that MREMAP_DONTUNMAP on a shared mapping works as expected.
12862306a36Sopenharmony_cistatic void mremap_dontunmap_simple_shmem()
12962306a36Sopenharmony_ci{
13062306a36Sopenharmony_ci	unsigned long num_pages = 5;
13162306a36Sopenharmony_ci
13262306a36Sopenharmony_ci	int mem_fd = memfd_create("memfd", MFD_CLOEXEC);
13362306a36Sopenharmony_ci	BUG_ON(mem_fd < 0, "memfd_create");
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci	BUG_ON(ftruncate(mem_fd, num_pages * page_size) < 0,
13662306a36Sopenharmony_ci			"ftruncate");
13762306a36Sopenharmony_ci
13862306a36Sopenharmony_ci	void *source_mapping =
13962306a36Sopenharmony_ci	    mmap(NULL, num_pages * page_size, PROT_READ | PROT_WRITE,
14062306a36Sopenharmony_ci		 MAP_FILE | MAP_SHARED, mem_fd, 0);
14162306a36Sopenharmony_ci	BUG_ON(source_mapping == MAP_FAILED, "mmap");
14262306a36Sopenharmony_ci
14362306a36Sopenharmony_ci	BUG_ON(close(mem_fd) < 0, "close");
14462306a36Sopenharmony_ci
14562306a36Sopenharmony_ci	memset(source_mapping, 'a', num_pages * page_size);
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_ci	// Try to just move the whole mapping anywhere (not fixed).
14862306a36Sopenharmony_ci	void *dest_mapping =
14962306a36Sopenharmony_ci	    mremap(source_mapping, num_pages * page_size, num_pages * page_size,
15062306a36Sopenharmony_ci		   MREMAP_DONTUNMAP | MREMAP_MAYMOVE, NULL);
15162306a36Sopenharmony_ci	if (dest_mapping == MAP_FAILED && errno == EINVAL) {
15262306a36Sopenharmony_ci		// Old kernel which doesn't support MREMAP_DONTUNMAP on shmem.
15362306a36Sopenharmony_ci		BUG_ON(munmap(source_mapping, num_pages * page_size) == -1,
15462306a36Sopenharmony_ci			"unable to unmap source mapping");
15562306a36Sopenharmony_ci		return;
15662306a36Sopenharmony_ci	}
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_ci	BUG_ON(dest_mapping == MAP_FAILED, "mremap");
15962306a36Sopenharmony_ci
16062306a36Sopenharmony_ci	// Validate that the pages have been moved, we know they were moved if
16162306a36Sopenharmony_ci	// the dest_mapping contains a's.
16262306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte
16362306a36Sopenharmony_ci	       (dest_mapping, num_pages * page_size, 'a') != 0,
16462306a36Sopenharmony_ci	       "pages did not migrate");
16562306a36Sopenharmony_ci
16662306a36Sopenharmony_ci	// Because the region is backed by shmem, we will actually see the same
16762306a36Sopenharmony_ci	// memory at the source location still.
16862306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte
16962306a36Sopenharmony_ci	       (source_mapping, num_pages * page_size, 'a') != 0,
17062306a36Sopenharmony_ci	       "source should have no ptes");
17162306a36Sopenharmony_ci
17262306a36Sopenharmony_ci	BUG_ON(munmap(dest_mapping, num_pages * page_size) == -1,
17362306a36Sopenharmony_ci	       "unable to unmap destination mapping");
17462306a36Sopenharmony_ci	BUG_ON(munmap(source_mapping, num_pages * page_size) == -1,
17562306a36Sopenharmony_ci	       "unable to unmap source mapping");
17662306a36Sopenharmony_ci}
17762306a36Sopenharmony_ci
17862306a36Sopenharmony_ci// This test validates MREMAP_DONTUNMAP will move page tables to a specific
17962306a36Sopenharmony_ci// destination using MREMAP_FIXED, also while validating that the source
18062306a36Sopenharmony_ci// remains intact.
18162306a36Sopenharmony_cistatic void mremap_dontunmap_simple_fixed()
18262306a36Sopenharmony_ci{
18362306a36Sopenharmony_ci	unsigned long num_pages = 5;
18462306a36Sopenharmony_ci
18562306a36Sopenharmony_ci	// Since we want to guarantee that we can remap to a point, we will
18662306a36Sopenharmony_ci	// create a mapping up front.
18762306a36Sopenharmony_ci	void *dest_mapping =
18862306a36Sopenharmony_ci	    mmap(NULL, num_pages * page_size, PROT_READ | PROT_WRITE,
18962306a36Sopenharmony_ci		 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
19062306a36Sopenharmony_ci	BUG_ON(dest_mapping == MAP_FAILED, "mmap");
19162306a36Sopenharmony_ci	memset(dest_mapping, 'X', num_pages * page_size);
19262306a36Sopenharmony_ci
19362306a36Sopenharmony_ci	void *source_mapping =
19462306a36Sopenharmony_ci	    mmap(NULL, num_pages * page_size, PROT_READ | PROT_WRITE,
19562306a36Sopenharmony_ci		 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
19662306a36Sopenharmony_ci	BUG_ON(source_mapping == MAP_FAILED, "mmap");
19762306a36Sopenharmony_ci	memset(source_mapping, 'a', num_pages * page_size);
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_ci	void *remapped_mapping =
20062306a36Sopenharmony_ci	    mremap(source_mapping, num_pages * page_size, num_pages * page_size,
20162306a36Sopenharmony_ci		   MREMAP_FIXED | MREMAP_DONTUNMAP | MREMAP_MAYMOVE,
20262306a36Sopenharmony_ci		   dest_mapping);
20362306a36Sopenharmony_ci	BUG_ON(remapped_mapping == MAP_FAILED, "mremap");
20462306a36Sopenharmony_ci	BUG_ON(remapped_mapping != dest_mapping,
20562306a36Sopenharmony_ci	       "mremap should have placed the remapped mapping at dest_mapping");
20662306a36Sopenharmony_ci
20762306a36Sopenharmony_ci	// The dest mapping will have been unmap by mremap so we expect the Xs
20862306a36Sopenharmony_ci	// to be gone and replaced with a's.
20962306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte
21062306a36Sopenharmony_ci	       (dest_mapping, num_pages * page_size, 'a') != 0,
21162306a36Sopenharmony_ci	       "pages did not migrate");
21262306a36Sopenharmony_ci
21362306a36Sopenharmony_ci	// And the source mapping will have had its ptes dropped.
21462306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte
21562306a36Sopenharmony_ci	       (source_mapping, num_pages * page_size, 0) != 0,
21662306a36Sopenharmony_ci	       "source should have no ptes");
21762306a36Sopenharmony_ci
21862306a36Sopenharmony_ci	BUG_ON(munmap(dest_mapping, num_pages * page_size) == -1,
21962306a36Sopenharmony_ci	       "unable to unmap destination mapping");
22062306a36Sopenharmony_ci	BUG_ON(munmap(source_mapping, num_pages * page_size) == -1,
22162306a36Sopenharmony_ci	       "unable to unmap source mapping");
22262306a36Sopenharmony_ci}
22362306a36Sopenharmony_ci
22462306a36Sopenharmony_ci// This test validates that we can MREMAP_DONTUNMAP for a portion of an
22562306a36Sopenharmony_ci// existing mapping.
22662306a36Sopenharmony_cistatic void mremap_dontunmap_partial_mapping()
22762306a36Sopenharmony_ci{
22862306a36Sopenharmony_ci	/*
22962306a36Sopenharmony_ci	 *  source mapping:
23062306a36Sopenharmony_ci	 *  --------------
23162306a36Sopenharmony_ci	 *  | aaaaaaaaaa |
23262306a36Sopenharmony_ci	 *  --------------
23362306a36Sopenharmony_ci	 *  to become:
23462306a36Sopenharmony_ci	 *  --------------
23562306a36Sopenharmony_ci	 *  | aaaaa00000 |
23662306a36Sopenharmony_ci	 *  --------------
23762306a36Sopenharmony_ci	 *  With the destination mapping containing 5 pages of As.
23862306a36Sopenharmony_ci	 *  ---------
23962306a36Sopenharmony_ci	 *  | aaaaa |
24062306a36Sopenharmony_ci	 *  ---------
24162306a36Sopenharmony_ci	 */
24262306a36Sopenharmony_ci	unsigned long num_pages = 10;
24362306a36Sopenharmony_ci	void *source_mapping =
24462306a36Sopenharmony_ci	    mmap(NULL, num_pages * page_size, PROT_READ | PROT_WRITE,
24562306a36Sopenharmony_ci		 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
24662306a36Sopenharmony_ci	BUG_ON(source_mapping == MAP_FAILED, "mmap");
24762306a36Sopenharmony_ci	memset(source_mapping, 'a', num_pages * page_size);
24862306a36Sopenharmony_ci
24962306a36Sopenharmony_ci	// We will grab the last 5 pages of the source and move them.
25062306a36Sopenharmony_ci	void *dest_mapping =
25162306a36Sopenharmony_ci	    mremap(source_mapping + (5 * page_size), 5 * page_size,
25262306a36Sopenharmony_ci		   5 * page_size,
25362306a36Sopenharmony_ci		   MREMAP_DONTUNMAP | MREMAP_MAYMOVE, NULL);
25462306a36Sopenharmony_ci	BUG_ON(dest_mapping == MAP_FAILED, "mremap");
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_ci	// We expect the first 5 pages of the source to contain a's and the
25762306a36Sopenharmony_ci	// final 5 pages to contain zeros.
25862306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte(source_mapping, 5 * page_size, 'a') !=
25962306a36Sopenharmony_ci	       0, "first 5 pages of source should have original pages");
26062306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte
26162306a36Sopenharmony_ci	       (source_mapping + (5 * page_size), 5 * page_size, 0) != 0,
26262306a36Sopenharmony_ci	       "final 5 pages of source should have no ptes");
26362306a36Sopenharmony_ci
26462306a36Sopenharmony_ci	// Finally we expect the destination to have 5 pages worth of a's.
26562306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte(dest_mapping, 5 * page_size, 'a') !=
26662306a36Sopenharmony_ci	       0, "dest mapping should contain ptes from the source");
26762306a36Sopenharmony_ci
26862306a36Sopenharmony_ci	BUG_ON(munmap(dest_mapping, 5 * page_size) == -1,
26962306a36Sopenharmony_ci	       "unable to unmap destination mapping");
27062306a36Sopenharmony_ci	BUG_ON(munmap(source_mapping, num_pages * page_size) == -1,
27162306a36Sopenharmony_ci	       "unable to unmap source mapping");
27262306a36Sopenharmony_ci}
27362306a36Sopenharmony_ci
27462306a36Sopenharmony_ci// This test validates that we can remap over only a portion of a mapping.
27562306a36Sopenharmony_cistatic void mremap_dontunmap_partial_mapping_overwrite(void)
27662306a36Sopenharmony_ci{
27762306a36Sopenharmony_ci	/*
27862306a36Sopenharmony_ci	 *  source mapping:
27962306a36Sopenharmony_ci	 *  ---------
28062306a36Sopenharmony_ci	 *  |aaaaa|
28162306a36Sopenharmony_ci	 *  ---------
28262306a36Sopenharmony_ci	 *  dest mapping initially:
28362306a36Sopenharmony_ci	 *  -----------
28462306a36Sopenharmony_ci	 *  |XXXXXXXXXX|
28562306a36Sopenharmony_ci	 *  ------------
28662306a36Sopenharmony_ci	 *  Source to become:
28762306a36Sopenharmony_ci	 *  ---------
28862306a36Sopenharmony_ci	 *  |00000|
28962306a36Sopenharmony_ci	 *  ---------
29062306a36Sopenharmony_ci	 *  With the destination mapping containing 5 pages of As.
29162306a36Sopenharmony_ci	 *  ------------
29262306a36Sopenharmony_ci	 *  |aaaaaXXXXX|
29362306a36Sopenharmony_ci	 *  ------------
29462306a36Sopenharmony_ci	 */
29562306a36Sopenharmony_ci	void *source_mapping =
29662306a36Sopenharmony_ci	    mmap(NULL, 5 * page_size, PROT_READ | PROT_WRITE,
29762306a36Sopenharmony_ci		 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
29862306a36Sopenharmony_ci	BUG_ON(source_mapping == MAP_FAILED, "mmap");
29962306a36Sopenharmony_ci	memset(source_mapping, 'a', 5 * page_size);
30062306a36Sopenharmony_ci
30162306a36Sopenharmony_ci	void *dest_mapping =
30262306a36Sopenharmony_ci	    mmap(NULL, 10 * page_size, PROT_READ | PROT_WRITE,
30362306a36Sopenharmony_ci		 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
30462306a36Sopenharmony_ci	BUG_ON(dest_mapping == MAP_FAILED, "mmap");
30562306a36Sopenharmony_ci	memset(dest_mapping, 'X', 10 * page_size);
30662306a36Sopenharmony_ci
30762306a36Sopenharmony_ci	// We will grab the last 5 pages of the source and move them.
30862306a36Sopenharmony_ci	void *remapped_mapping =
30962306a36Sopenharmony_ci	    mremap(source_mapping, 5 * page_size,
31062306a36Sopenharmony_ci		   5 * page_size,
31162306a36Sopenharmony_ci		   MREMAP_DONTUNMAP | MREMAP_MAYMOVE | MREMAP_FIXED, dest_mapping);
31262306a36Sopenharmony_ci	BUG_ON(dest_mapping == MAP_FAILED, "mremap");
31362306a36Sopenharmony_ci	BUG_ON(dest_mapping != remapped_mapping, "expected to remap to dest_mapping");
31462306a36Sopenharmony_ci
31562306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte(source_mapping, 5 * page_size, 0) !=
31662306a36Sopenharmony_ci	       0, "first 5 pages of source should have no ptes");
31762306a36Sopenharmony_ci
31862306a36Sopenharmony_ci	// Finally we expect the destination to have 5 pages worth of a's.
31962306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte(dest_mapping, 5 * page_size, 'a') != 0,
32062306a36Sopenharmony_ci			"dest mapping should contain ptes from the source");
32162306a36Sopenharmony_ci
32262306a36Sopenharmony_ci	// Finally the last 5 pages shouldn't have been touched.
32362306a36Sopenharmony_ci	BUG_ON(check_region_contains_byte(dest_mapping + (5 * page_size),
32462306a36Sopenharmony_ci				5 * page_size, 'X') != 0,
32562306a36Sopenharmony_ci			"dest mapping should have retained the last 5 pages");
32662306a36Sopenharmony_ci
32762306a36Sopenharmony_ci	BUG_ON(munmap(dest_mapping, 10 * page_size) == -1,
32862306a36Sopenharmony_ci	       "unable to unmap destination mapping");
32962306a36Sopenharmony_ci	BUG_ON(munmap(source_mapping, 5 * page_size) == -1,
33062306a36Sopenharmony_ci	       "unable to unmap source mapping");
33162306a36Sopenharmony_ci}
33262306a36Sopenharmony_ci
33362306a36Sopenharmony_ciint main(void)
33462306a36Sopenharmony_ci{
33562306a36Sopenharmony_ci	page_size = sysconf(_SC_PAGE_SIZE);
33662306a36Sopenharmony_ci
33762306a36Sopenharmony_ci	// test for kernel support for MREMAP_DONTUNMAP skipping the test if
33862306a36Sopenharmony_ci	// not.
33962306a36Sopenharmony_ci	if (kernel_support_for_mremap_dontunmap() != 0) {
34062306a36Sopenharmony_ci		printf("No kernel support for MREMAP_DONTUNMAP\n");
34162306a36Sopenharmony_ci		return KSFT_SKIP;
34262306a36Sopenharmony_ci	}
34362306a36Sopenharmony_ci
34462306a36Sopenharmony_ci	// Keep a page sized buffer around for when we need it.
34562306a36Sopenharmony_ci	page_buffer =
34662306a36Sopenharmony_ci	    mmap(NULL, page_size, PROT_READ | PROT_WRITE,
34762306a36Sopenharmony_ci		 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
34862306a36Sopenharmony_ci	BUG_ON(page_buffer == MAP_FAILED, "unable to mmap a page.");
34962306a36Sopenharmony_ci
35062306a36Sopenharmony_ci	mremap_dontunmap_simple();
35162306a36Sopenharmony_ci	mremap_dontunmap_simple_shmem();
35262306a36Sopenharmony_ci	mremap_dontunmap_simple_fixed();
35362306a36Sopenharmony_ci	mremap_dontunmap_partial_mapping();
35462306a36Sopenharmony_ci	mremap_dontunmap_partial_mapping_overwrite();
35562306a36Sopenharmony_ci
35662306a36Sopenharmony_ci	BUG_ON(munmap(page_buffer, page_size) == -1,
35762306a36Sopenharmony_ci	       "unable to unmap page buffer");
35862306a36Sopenharmony_ci
35962306a36Sopenharmony_ci	printf("OK\n");
36062306a36Sopenharmony_ci	return 0;
36162306a36Sopenharmony_ci}
362