1f08c3bdfSopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 2f08c3bdfSopenharmony_ci/* 3f08c3bdfSopenharmony_ci * Copyright (c) 2023 SUSE LLC 4f08c3bdfSopenharmony_ci * Author: Vlastimil Babka <vbabka@suse.cz> 5f08c3bdfSopenharmony_ci * https://bugzilla.suse.com/attachment.cgi?id=867254 6f08c3bdfSopenharmony_ci * LTP port: Petr Vorel <pvorel@suse.cz> 7f08c3bdfSopenharmony_ci */ 8f08c3bdfSopenharmony_ci 9f08c3bdfSopenharmony_ci/*\ 10f08c3bdfSopenharmony_ci * [Description] 11f08c3bdfSopenharmony_ci * 12f08c3bdfSopenharmony_ci * Bug reproducer for 7e7757876f25 ("mm/mremap: fix vm_pgoff in vma_merge() case 3") 13f08c3bdfSopenharmony_ci */ 14f08c3bdfSopenharmony_ci 15f08c3bdfSopenharmony_ci#define _GNU_SOURCE 16f08c3bdfSopenharmony_ci#include <unistd.h> 17f08c3bdfSopenharmony_ci#include <stdlib.h> 18f08c3bdfSopenharmony_ci#include <stdio.h> 19f08c3bdfSopenharmony_ci#include <fcntl.h> 20f08c3bdfSopenharmony_ci#include <sys/mman.h> 21f08c3bdfSopenharmony_ci 22f08c3bdfSopenharmony_ci#include "tst_test.h" 23f08c3bdfSopenharmony_ci#include "tst_safe_macros.h" 24f08c3bdfSopenharmony_ci 25f08c3bdfSopenharmony_ci#define NUM_PAGES 3 26f08c3bdfSopenharmony_ci 27f08c3bdfSopenharmony_cistatic int fd; 28f08c3bdfSopenharmony_cistatic char *buf, *buf2; 29f08c3bdfSopenharmony_cistatic int page_size, mmap_size, mremap_size; 30f08c3bdfSopenharmony_ci 31f08c3bdfSopenharmony_cistatic struct tcase { 32f08c3bdfSopenharmony_ci size_t incompatible; 33f08c3bdfSopenharmony_ci const char *desc; 34f08c3bdfSopenharmony_ci} tcases[] = { 35f08c3bdfSopenharmony_ci { 36f08c3bdfSopenharmony_ci .desc = "all pages with compatible mapping", 37f08c3bdfSopenharmony_ci }, 38f08c3bdfSopenharmony_ci { 39f08c3bdfSopenharmony_ci .incompatible = 3, 40f08c3bdfSopenharmony_ci .desc = "third page's mapping incompatible", 41f08c3bdfSopenharmony_ci }, 42f08c3bdfSopenharmony_ci { 43f08c3bdfSopenharmony_ci .incompatible = 1, 44f08c3bdfSopenharmony_ci .desc = "first page's mapping incompatible", 45f08c3bdfSopenharmony_ci }, 46f08c3bdfSopenharmony_ci}; 47f08c3bdfSopenharmony_ci 48f08c3bdfSopenharmony_cistatic int check_pages(void) 49f08c3bdfSopenharmony_ci{ 50f08c3bdfSopenharmony_ci int fail = 0, i; 51f08c3bdfSopenharmony_ci char val; 52f08c3bdfSopenharmony_ci 53f08c3bdfSopenharmony_ci for (i = 0; i < (int)ARRAY_SIZE(tcases); i++) { 54f08c3bdfSopenharmony_ci val = buf[i * page_size]; 55f08c3bdfSopenharmony_ci if (val != 0x30 + i) { 56f08c3bdfSopenharmony_ci tst_res(TFAIL, "page %d wrong value %d (0x%x)", i, val - 0x30, val); 57f08c3bdfSopenharmony_ci fail = 1; 58f08c3bdfSopenharmony_ci } 59f08c3bdfSopenharmony_ci } 60f08c3bdfSopenharmony_ci 61f08c3bdfSopenharmony_ci return fail; 62f08c3bdfSopenharmony_ci} 63f08c3bdfSopenharmony_ci 64f08c3bdfSopenharmony_cistatic void do_test(unsigned int n) 65f08c3bdfSopenharmony_ci{ 66f08c3bdfSopenharmony_ci struct tcase *tc = &tcases[n]; 67f08c3bdfSopenharmony_ci int ret; 68f08c3bdfSopenharmony_ci 69f08c3bdfSopenharmony_ci tst_res(TINFO, "%s", tc->desc); 70f08c3bdfSopenharmony_ci 71f08c3bdfSopenharmony_ci buf = SAFE_MMAP(0, mmap_size, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0); 72f08c3bdfSopenharmony_ci 73f08c3bdfSopenharmony_ci buf2 = mremap(buf + page_size, page_size, page_size, 74f08c3bdfSopenharmony_ci MREMAP_MAYMOVE|MREMAP_FIXED, buf + mremap_size); 75f08c3bdfSopenharmony_ci if (buf2 == MAP_FAILED) 76f08c3bdfSopenharmony_ci tst_brk(TBROK, "mremap() failed"); 77f08c3bdfSopenharmony_ci 78f08c3bdfSopenharmony_ci if (tc->incompatible) { 79f08c3bdfSopenharmony_ci ret = mprotect(buf + (tc->incompatible-1)*page_size, page_size, PROT_READ); 80f08c3bdfSopenharmony_ci if (ret == -1) 81f08c3bdfSopenharmony_ci tst_brk(TBROK, "mprotect() failed"); 82f08c3bdfSopenharmony_ci } 83f08c3bdfSopenharmony_ci 84f08c3bdfSopenharmony_ci buf2 = mremap(buf + mremap_size, page_size, page_size, 85f08c3bdfSopenharmony_ci MREMAP_MAYMOVE|MREMAP_FIXED, buf + page_size); 86f08c3bdfSopenharmony_ci if (buf2 == MAP_FAILED) 87f08c3bdfSopenharmony_ci tst_brk(TBROK, "mremap() failed"); 88f08c3bdfSopenharmony_ci 89f08c3bdfSopenharmony_ci if (!check_pages()) 90f08c3bdfSopenharmony_ci tst_res(TPASS, "mmap/mremap work properly"); 91f08c3bdfSopenharmony_ci 92f08c3bdfSopenharmony_ci SAFE_MUNMAP(buf, mremap_size); 93f08c3bdfSopenharmony_ci} 94f08c3bdfSopenharmony_ci 95f08c3bdfSopenharmony_cistatic void setup(void) 96f08c3bdfSopenharmony_ci{ 97f08c3bdfSopenharmony_ci int ret, i; 98f08c3bdfSopenharmony_ci 99f08c3bdfSopenharmony_ci page_size = getpagesize(); 100f08c3bdfSopenharmony_ci mmap_size = (NUM_PAGES+1) * page_size; 101f08c3bdfSopenharmony_ci mremap_size = NUM_PAGES * page_size; 102f08c3bdfSopenharmony_ci 103f08c3bdfSopenharmony_ci fd = SAFE_OPEN("testfile", O_CREAT | O_RDWR | O_TRUNC, 0600); 104f08c3bdfSopenharmony_ci 105f08c3bdfSopenharmony_ci ret = fallocate(fd, 0, 0, mmap_size); 106f08c3bdfSopenharmony_ci if (ret == -1) 107f08c3bdfSopenharmony_ci tst_brk(TBROK, "fallocate() failed"); 108f08c3bdfSopenharmony_ci 109f08c3bdfSopenharmony_ci buf = SAFE_MMAP(0, mmap_size, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0); 110f08c3bdfSopenharmony_ci 111f08c3bdfSopenharmony_ci for (i = 0; i < (int)ARRAY_SIZE(tcases)+1; i++) 112f08c3bdfSopenharmony_ci buf[i*page_size] = 0x30 + i; 113f08c3bdfSopenharmony_ci 114f08c3bdfSopenharmony_ci /* clear the page tables */ 115f08c3bdfSopenharmony_ci SAFE_MUNMAP(buf, mmap_size); 116f08c3bdfSopenharmony_ci} 117f08c3bdfSopenharmony_ci 118f08c3bdfSopenharmony_cistatic void cleanup(void) 119f08c3bdfSopenharmony_ci{ 120f08c3bdfSopenharmony_ci if (fd > 0) 121f08c3bdfSopenharmony_ci SAFE_CLOSE(fd); 122f08c3bdfSopenharmony_ci} 123f08c3bdfSopenharmony_ci 124f08c3bdfSopenharmony_cistatic struct tst_test test = { 125f08c3bdfSopenharmony_ci .setup = setup, 126f08c3bdfSopenharmony_ci .cleanup = cleanup, 127f08c3bdfSopenharmony_ci .test = do_test, 128f08c3bdfSopenharmony_ci .needs_tmpdir = 1, 129f08c3bdfSopenharmony_ci .tcnt = ARRAY_SIZE(tcases), 130f08c3bdfSopenharmony_ci .tags = (struct tst_tag[]) { 131f08c3bdfSopenharmony_ci {"linux-git", "7e7757876f25"}, 132f08c3bdfSopenharmony_ci {} 133f08c3bdfSopenharmony_ci }, 134f08c3bdfSopenharmony_ci}; 135