135318db5SAlexey Dobriyan /*
2ee9294d6SAlexey Dobriyan * Copyright © 2018 Alexey Dobriyan <adobriyan@gmail.com>
335318db5SAlexey Dobriyan *
435318db5SAlexey Dobriyan * Permission to use, copy, modify, and distribute this software for any
535318db5SAlexey Dobriyan * purpose with or without fee is hereby granted, provided that the above
635318db5SAlexey Dobriyan * copyright notice and this permission notice appear in all copies.
735318db5SAlexey Dobriyan *
835318db5SAlexey Dobriyan * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
935318db5SAlexey Dobriyan * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
1035318db5SAlexey Dobriyan * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
1135318db5SAlexey Dobriyan * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
1235318db5SAlexey Dobriyan * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
1335318db5SAlexey Dobriyan * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
1435318db5SAlexey Dobriyan * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1535318db5SAlexey Dobriyan */
16dbd4af54SAlexey Dobriyan /* Test readlink /proc/self/map_files/... with minimum address. */
1735318db5SAlexey Dobriyan #include <errno.h>
1835318db5SAlexey Dobriyan #include <sys/types.h>
1935318db5SAlexey Dobriyan #include <sys/stat.h>
2035318db5SAlexey Dobriyan #include <fcntl.h>
2135318db5SAlexey Dobriyan #include <stdio.h>
2235318db5SAlexey Dobriyan #include <unistd.h>
2335318db5SAlexey Dobriyan #include <sys/mman.h>
2435318db5SAlexey Dobriyan #include <stdlib.h>
2535318db5SAlexey Dobriyan
pass(const char * fmt,unsigned long a,unsigned long b)2635318db5SAlexey Dobriyan static void pass(const char *fmt, unsigned long a, unsigned long b)
2735318db5SAlexey Dobriyan {
2835318db5SAlexey Dobriyan char name[64];
2935318db5SAlexey Dobriyan char buf[64];
3035318db5SAlexey Dobriyan
3135318db5SAlexey Dobriyan snprintf(name, sizeof(name), fmt, a, b);
3235318db5SAlexey Dobriyan if (readlink(name, buf, sizeof(buf)) == -1)
3335318db5SAlexey Dobriyan exit(1);
3435318db5SAlexey Dobriyan }
3535318db5SAlexey Dobriyan
fail(const char * fmt,unsigned long a,unsigned long b)3635318db5SAlexey Dobriyan static void fail(const char *fmt, unsigned long a, unsigned long b)
3735318db5SAlexey Dobriyan {
3835318db5SAlexey Dobriyan char name[64];
3935318db5SAlexey Dobriyan char buf[64];
4035318db5SAlexey Dobriyan
4135318db5SAlexey Dobriyan snprintf(name, sizeof(name), fmt, a, b);
4235318db5SAlexey Dobriyan if (readlink(name, buf, sizeof(buf)) == -1 && errno == ENOENT)
4335318db5SAlexey Dobriyan return;
4435318db5SAlexey Dobriyan exit(1);
4535318db5SAlexey Dobriyan }
4635318db5SAlexey Dobriyan
main(void)4735318db5SAlexey Dobriyan int main(void)
4835318db5SAlexey Dobriyan {
498cd40d1dSAlexey Dobriyan const int PAGE_SIZE = sysconf(_SC_PAGESIZE);
50*2f3571eaSMasami Hiramatsu /*
51*2f3571eaSMasami Hiramatsu * va_max must be enough bigger than vm.mmap_min_addr, which is
52*2f3571eaSMasami Hiramatsu * 64KB/32KB by default. (depends on CONFIG_LSM_MMAP_MIN_ADDR)
53*2f3571eaSMasami Hiramatsu */
54*2f3571eaSMasami Hiramatsu const unsigned long va_max = 1UL << 20;
558cd40d1dSAlexey Dobriyan unsigned long va;
5635318db5SAlexey Dobriyan void *p;
5735318db5SAlexey Dobriyan int fd;
5835318db5SAlexey Dobriyan unsigned long a, b;
5935318db5SAlexey Dobriyan
6035318db5SAlexey Dobriyan fd = open("/dev/zero", O_RDONLY);
6135318db5SAlexey Dobriyan if (fd == -1)
6235318db5SAlexey Dobriyan return 1;
6335318db5SAlexey Dobriyan
648cd40d1dSAlexey Dobriyan for (va = 0; va < va_max; va += PAGE_SIZE) {
65dbd4af54SAlexey Dobriyan p = mmap((void *)va, PAGE_SIZE, PROT_NONE, MAP_PRIVATE|MAP_FILE|MAP_FIXED, fd, 0);
668cd40d1dSAlexey Dobriyan if (p == (void *)va)
678cd40d1dSAlexey Dobriyan break;
688cd40d1dSAlexey Dobriyan }
698cd40d1dSAlexey Dobriyan if (va == va_max) {
708cd40d1dSAlexey Dobriyan fprintf(stderr, "error: mmap doesn't like you\n");
7135318db5SAlexey Dobriyan return 1;
7235318db5SAlexey Dobriyan }
7335318db5SAlexey Dobriyan
7435318db5SAlexey Dobriyan a = (unsigned long)p;
7535318db5SAlexey Dobriyan b = (unsigned long)p + PAGE_SIZE;
7635318db5SAlexey Dobriyan
7735318db5SAlexey Dobriyan pass("/proc/self/map_files/%lx-%lx", a, b);
7835318db5SAlexey Dobriyan fail("/proc/self/map_files/ %lx-%lx", a, b);
7935318db5SAlexey Dobriyan fail("/proc/self/map_files/%lx -%lx", a, b);
8035318db5SAlexey Dobriyan fail("/proc/self/map_files/%lx- %lx", a, b);
8135318db5SAlexey Dobriyan fail("/proc/self/map_files/%lx-%lx ", a, b);
8235318db5SAlexey Dobriyan fail("/proc/self/map_files/0%lx-%lx", a, b);
8335318db5SAlexey Dobriyan fail("/proc/self/map_files/%lx-0%lx", a, b);
8435318db5SAlexey Dobriyan if (sizeof(long) == 4) {
8535318db5SAlexey Dobriyan fail("/proc/self/map_files/100000000%lx-%lx", a, b);
8635318db5SAlexey Dobriyan fail("/proc/self/map_files/%lx-100000000%lx", a, b);
8735318db5SAlexey Dobriyan } else if (sizeof(long) == 8) {
8835318db5SAlexey Dobriyan fail("/proc/self/map_files/10000000000000000%lx-%lx", a, b);
8935318db5SAlexey Dobriyan fail("/proc/self/map_files/%lx-10000000000000000%lx", a, b);
9035318db5SAlexey Dobriyan } else
9135318db5SAlexey Dobriyan return 1;
9235318db5SAlexey Dobriyan
9335318db5SAlexey Dobriyan return 0;
9435318db5SAlexey Dobriyan }
95