#include <stdio.h> #include <stdlib.h> #include <sys/mman.h> #include <unistd.h> #include <fcntl.h> /* This file includes a simple set of memory allocation calls that * a user space program can use to allocate/free or move memory mappings. * The intent of this program is to make it easier to verify if the kernel * internal mappings are correct. */ #define PAGE_SHIFT 12 #define ROUND_PAGES(memsize) ((memsize >> (PAGE_SHIFT)) << PAGE_SHIFT) /* approximately half of memsize, page aligned */ #define HALF_MEM(memsize) ((memsize >> (PAGE_SHIFT))<<(PAGE_SHIFT - 1)) inline void waitnext() { fflush(NULL); getchar(); } int main(int argc, char *argv[]) { unsigned int memsize; char *mem; int i, numpages, fd; if (argc != 2) { printf("Usage: %s <memory_size>\n", argv[0]); exit(EXIT_FAILURE); } memsize = strtoul(argv[1], NULL, 10); memsize = ROUND_PAGES(memsize); /* We should be limited to < 4G so any size other than 0 is ok */ if (memsize == 0) { printf("Invalid memsize\n"); exit(EXIT_FAILURE); } numpages = memsize >> PAGE_SHIFT; mlockall(MCL_FUTURE); mem = sbrk(memsize); if (mem == (void*) -1) { perror("Failed to allocate memory using sbrk\n"); exit(EXIT_FAILURE); } printf("Successfully allocated sbrk memory %d bytes @%p\n", memsize, mem); waitnext(); sbrk(-(memsize)); mem = mmap(0, memsize, PROT_READ | PROT_WRITE, MAP_PRIVATE| MAP_ANONYMOUS, -1, 0); if (mem == (void*) -1) { perror("Failed to allocate anon private memory using mmap\n"); exit(EXIT_FAILURE); } printf("Successfully allocated anon mmap memory %d bytes @%p\n", memsize, mem); waitnext(); if (-1 == mprotect(mem, HALF_MEM(memsize), PROT_READ)) { perror("Failed to W protect memory using mprotect\n"); exit(EXIT_FAILURE); } printf("Successfully write protected %d bytes @%p\n", HALF_MEM(memsize), mem); waitnext(); if (-1 == mprotect(mem, HALF_MEM(memsize), PROT_READ | PROT_WRITE)) { perror("Failed to RW protect memory using mprotect\n"); exit(EXIT_FAILURE); } printf("Successfully cleared write protected %d bytes @%p\n", memsize, mem); waitnext(); /* Mark all pages with a specific pattern */ for (i = 0; i < numpages; i++) { int *ptr = (int *)(mem + i*4096); *ptr = i; } mem = mremap(mem , memsize, memsize + HALF_MEM(memsize), 1 /* MREMAP_MAYMOVE */); if (mem == MAP_FAILED) { perror("Failed to remap expand anon private memory\n"); exit(EXIT_FAILURE); } printf("Successfully remapped %d bytes @%p\n", memsize + HALF_MEM(memsize), mem); waitnext(); /* Mark all pages with a specific pattern */ for (i = 0; i < numpages; i++) { int value = *(int*)(mem + i*4096); if (value != i) { printf("remap error expected %d got %d\n", i, value); exit(EXIT_FAILURE); } } if (munmap(mem, memsize + HALF_MEM(memsize))) { perror("Could not unmap and free memory\n"); exit(EXIT_FAILURE); } fd = open("/dev/zero", O_RDONLY); mem = mmap(0, memsize, PROT_READ | PROT_WRITE, MAP_PRIVATE, fd, 0); if (mem == (void*) -1) { perror("Failed to allocate file backed memory using mmap\n"); exit(EXIT_FAILURE); } printf("Successfully allocated file backed mmap memory %d bytes @%p\n", memsize, mem); waitnext(); if (munmap(mem, memsize)) { perror("Could not unmap and free file backed memory\n"); exit(EXIT_FAILURE); } exit(EXIT_SUCCESS); }