optimize locking against vm changes for mmap/munmap
[musl] / src / mman / mmap.c
1 #include <unistd.h>
2 #include <sys/mman.h>
3 #include <errno.h>
4 #include <stdint.h>
5 #include <limits.h>
6 #include "syscall.h"
7 #include "libc.h"
8
9 static void dummy1(int x) { }
10 static void dummy0(void) { }
11 weak_alias(dummy1, __vm_lock);
12 weak_alias(dummy0, __vm_unlock);
13
14 #define UNIT SYSCALL_MMAP2_UNIT
15 #define OFF_MASK ((-0x2000ULL << (8*sizeof(long)-1)) | (UNIT-1))
16
17 void *__mmap(void *start, size_t len, int prot, int flags, int fd, off_t off)
18 {
19         if (off & OFF_MASK) {
20                 errno = EINVAL;
21                 return MAP_FAILED;
22         }
23         if (len >= PTRDIFF_MAX) {
24                 errno = ENOMEM;
25                 return MAP_FAILED;
26         }
27         if (flags & MAP_FIXED) {
28                 __vm_lock(-1);
29                 __vm_unlock();
30         }
31 #ifdef SYS_mmap2
32         return (void *)syscall(SYS_mmap2, start, len, prot, flags, fd, off/UNIT);
33 #else
34         return (void *)syscall(SYS_mmap, start, len, prot, flags, fd, off);
35 #endif
36 }
37
38 weak_alias(__mmap, mmap);
39
40 LFS64(mmap);