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);
502f3571eaSMasami Hiramatsu 	/*
512f3571eaSMasami Hiramatsu 	 * va_max must be enough bigger than vm.mmap_min_addr, which is
522f3571eaSMasami Hiramatsu 	 * 64KB/32KB by default. (depends on CONFIG_LSM_MMAP_MIN_ADDR)
532f3571eaSMasami Hiramatsu 	 */
542f3571eaSMasami 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