1 //===-- sanitizer_solaris.cpp ---------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // This file is shared between various sanitizers' runtime libraries and
10 // implements Solaris-specific functions.
11 //===----------------------------------------------------------------------===//
13 #include "sanitizer_platform.h"
18 #include "sanitizer_common.h"
19 #include "sanitizer_flags.h"
20 #include "sanitizer_internal_defs.h"
21 #include "sanitizer_libc.h"
22 #include "sanitizer_placement_new.h"
23 #include "sanitizer_platform_limits_posix.h"
24 #include "sanitizer_procmaps.h"
33 #include <sys/resource.h>
35 #include <sys/types.h>
41 namespace __sanitizer
{
43 //#include "sanitizer_syscall_generic.inc"
45 #define _REAL(func) _ ## func
46 #define DECLARE__REAL(ret_type, func, ...) \
47 extern "C" ret_type _REAL(func)(__VA_ARGS__)
48 #define DECLARE__REAL_AND_INTERNAL(ret_type, func, ...) \
49 DECLARE__REAL(ret_type, func, __VA_ARGS__); \
50 ret_type internal_ ## func(__VA_ARGS__)
52 #if !defined(_LP64) && _FILE_OFFSET_BITS == 64
53 #define _REAL64(func) _ ## func ## 64
55 #define _REAL64(func) _REAL(func)
57 #define DECLARE__REAL64(ret_type, func, ...) \
58 extern "C" ret_type _REAL64(func)(__VA_ARGS__)
59 #define DECLARE__REAL_AND_INTERNAL64(ret_type, func, ...) \
60 DECLARE__REAL64(ret_type, func, __VA_ARGS__); \
61 ret_type internal_ ## func(__VA_ARGS__)
63 // ---------------------- sanitizer_libc.h
64 DECLARE__REAL_AND_INTERNAL64(uptr
, mmap
, void *addr
, uptr
/*size_t*/ length
,
65 int prot
, int flags
, int fd
, OFF_T offset
) {
66 return (uptr
)_REAL64(mmap
)(addr
, length
, prot
, flags
, fd
, offset
);
69 DECLARE__REAL_AND_INTERNAL(uptr
, munmap
, void *addr
, uptr length
) {
70 return _REAL(munmap
)(addr
, length
);
73 DECLARE__REAL_AND_INTERNAL(int, mprotect
, void *addr
, uptr length
, int prot
) {
74 return _REAL(mprotect
)(addr
, length
, prot
);
77 // Illumos' declaration of madvise cannot be made visible if _XOPEN_SOURCE
78 // is defined as g++ does on Solaris.
80 // This declaration is consistent with Solaris 11.4. Both Illumos and Solaris
81 // versions older than 11.4 declared madvise with a caddr_t as the first
82 // argument, but we don't currently support Solaris versions older than 11.4,
83 // and as mentioned above the declaration is not visible on Illumos so we can
84 // use any declaration we like on Illumos.
85 extern "C" int madvise(void *, size_t, int);
87 int internal_madvise(uptr addr
, uptr length
, int advice
) {
88 return madvise((void *)addr
, length
, advice
);
91 DECLARE__REAL_AND_INTERNAL(uptr
, close
, fd_t fd
) {
92 return _REAL(close
)(fd
);
95 extern "C" int _REAL64(open
)(const char *, int, ...);
97 uptr
internal_open(const char *filename
, int flags
) {
98 return _REAL64(open
)(filename
, flags
);
101 uptr
internal_open(const char *filename
, int flags
, u32 mode
) {
102 return _REAL64(open
)(filename
, flags
, mode
);
105 DECLARE__REAL_AND_INTERNAL(uptr
, read
, fd_t fd
, void *buf
, uptr count
) {
106 return _REAL(read
)(fd
, buf
, count
);
109 DECLARE__REAL_AND_INTERNAL(uptr
, write
, fd_t fd
, const void *buf
, uptr count
) {
110 return _REAL(write
)(fd
, buf
, count
);
113 // FIXME: There's only _ftruncate64 beginning with Solaris 11.
114 DECLARE__REAL_AND_INTERNAL(uptr
, ftruncate
, fd_t fd
, uptr size
) {
115 return ftruncate(fd
, size
);
118 DECLARE__REAL_AND_INTERNAL64(uptr
, stat
, const char *path
, void *buf
) {
119 return _REAL64(stat
)(path
, (struct stat
*)buf
);
122 DECLARE__REAL_AND_INTERNAL64(uptr
, lstat
, const char *path
, void *buf
) {
123 return _REAL64(lstat
)(path
, (struct stat
*)buf
);
126 DECLARE__REAL_AND_INTERNAL64(uptr
, fstat
, fd_t fd
, void *buf
) {
127 return _REAL64(fstat
)(fd
, (struct stat
*)buf
);
130 uptr
internal_filesize(fd_t fd
) {
132 if (internal_fstat(fd
, &st
))
134 return (uptr
)st
.st_size
;
137 DECLARE__REAL_AND_INTERNAL(uptr
, dup
, int oldfd
) {
138 return _REAL(dup
)(oldfd
);
141 DECLARE__REAL_AND_INTERNAL(uptr
, dup2
, int oldfd
, int newfd
) {
142 return _REAL(dup2
)(oldfd
, newfd
);
145 DECLARE__REAL_AND_INTERNAL(uptr
, readlink
, const char *path
, char *buf
,
147 return _REAL(readlink
)(path
, buf
, bufsize
);
150 DECLARE__REAL_AND_INTERNAL(uptr
, unlink
, const char *path
) {
151 return _REAL(unlink
)(path
);
154 DECLARE__REAL_AND_INTERNAL(uptr
, rename
, const char *oldpath
,
155 const char *newpath
) {
156 return _REAL(rename
)(oldpath
, newpath
);
159 DECLARE__REAL_AND_INTERNAL(uptr
, sched_yield
, void) {
160 return sched_yield();
163 DECLARE__REAL_AND_INTERNAL(void, usleep
, u64 useconds
) {
165 ts
.tv_sec
= useconds
/ 1000000;
166 ts
.tv_nsec
= (useconds
% 1000000) * 1000;
167 nanosleep(&ts
, nullptr);
170 DECLARE__REAL_AND_INTERNAL(uptr
, execve
, const char *filename
,
171 char *const argv
[], char *const envp
[]) {
172 return _REAL(execve
)(filename
, argv
, envp
);
175 DECLARE__REAL_AND_INTERNAL(uptr
, waitpid
, int pid
, int *status
, int options
) {
176 return _REAL(waitpid
)(pid
, status
, options
);
179 DECLARE__REAL_AND_INTERNAL(uptr
, getpid
, void) {
180 return _REAL(getpid
)();
183 // FIXME: This might be wrong: _getdents doesn't take a struct linux_dirent *.
184 DECLARE__REAL_AND_INTERNAL64(uptr
, getdents
, fd_t fd
, struct linux_dirent
*dirp
,
185 unsigned int count
) {
186 return _REAL64(getdents
)(fd
, dirp
, count
);
189 DECLARE__REAL_AND_INTERNAL64(uptr
, lseek
, fd_t fd
, OFF_T offset
, int whence
) {
190 return _REAL64(lseek
)(fd
, offset
, whence
);
193 // FIXME: This might be wrong: _sigfillset doesn't take a
194 // __sanitizer_sigset_t *.
195 DECLARE__REAL_AND_INTERNAL(void, sigfillset
, __sanitizer_sigset_t
*set
) {
196 _REAL(sigfillset
)(set
);
199 // FIXME: This might be wrong: _sigprocmask doesn't take __sanitizer_sigset_t *.
200 DECLARE__REAL_AND_INTERNAL(uptr
, sigprocmask
, int how
,
201 __sanitizer_sigset_t
*set
,
202 __sanitizer_sigset_t
*oldset
) {
203 return _REAL(sigprocmask
)(how
, set
, oldset
);
206 DECLARE__REAL_AND_INTERNAL(int, fork
, void) {
207 // TODO(glider): this may call user's pthread_atfork() handlers which is bad.
208 return _REAL(fork
)();
215 uptr
internal_clock_gettime(__sanitizer_clockid_t clk_id
, void *tp
) {
216 // FIXME: No internal variant.
217 return clock_gettime(clk_id
, (timespec
*)tp
);
220 // ----------------- sanitizer_common.h
221 void FutexWait(atomic_uint32_t
*p
, u32 cmp
) {
222 // FIXME: implement actual blocking.
226 void FutexWake(atomic_uint32_t
*p
, u32 count
) {}
228 } // namespace __sanitizer
230 #endif // SANITIZER_SOLARIS