1 //===-- sanitizer_solaris.cc ----------------------------------------------===//
3 // This file is distributed under the University of Illinois Open Source
4 // License. See LICENSE.TXT for details.
6 //===----------------------------------------------------------------------===//
8 // This file is shared between various sanitizers' runtime libraries and
9 // implements Solaris-specific functions.
10 //===----------------------------------------------------------------------===//
12 #include "sanitizer_platform.h"
17 #include "sanitizer_common.h"
18 #include "sanitizer_flags.h"
19 #include "sanitizer_internal_defs.h"
20 #include "sanitizer_libc.h"
21 #include "sanitizer_placement_new.h"
22 #include "sanitizer_platform_limits_posix.h"
23 #include "sanitizer_procmaps.h"
32 #include <sys/resource.h>
34 #include <sys/types.h>
40 namespace __sanitizer
{
42 //#include "sanitizer_syscall_generic.inc"
44 #define _REAL(func) _ ## func
45 #define DECLARE__REAL(ret_type, func, ...) \
46 extern "C" ret_type _REAL(func)(__VA_ARGS__)
47 #define DECLARE__REAL_AND_INTERNAL(ret_type, func, ...) \
48 DECLARE__REAL(ret_type, func, __VA_ARGS__); \
49 ret_type internal_ ## func(__VA_ARGS__)
51 #if !defined(_LP64) && _FILE_OFFSET_BITS == 64
52 #define _REAL64(func) _ ## func ## 64
54 #define _REAL64(func) _REAL(func)
56 #define DECLARE__REAL64(ret_type, func, ...) \
57 extern "C" ret_type _REAL64(func)(__VA_ARGS__)
58 #define DECLARE__REAL_AND_INTERNAL64(ret_type, func, ...) \
59 DECLARE__REAL64(ret_type, func, __VA_ARGS__); \
60 ret_type internal_ ## func(__VA_ARGS__)
62 // ---------------------- sanitizer_libc.h
63 DECLARE__REAL_AND_INTERNAL64(uptr
, mmap
, void *addr
, uptr
/*size_t*/ length
,
64 int prot
, int flags
, int fd
, OFF_T offset
) {
65 return (uptr
)_REAL64(mmap
)(addr
, length
, prot
, flags
, fd
, offset
);
68 DECLARE__REAL_AND_INTERNAL(uptr
, munmap
, void *addr
, uptr length
) {
69 return _REAL(munmap
)(addr
, length
);
72 DECLARE__REAL_AND_INTERNAL(int, mprotect
, void *addr
, uptr length
, int prot
) {
73 return _REAL(mprotect
)(addr
, length
, prot
);
76 DECLARE__REAL_AND_INTERNAL(uptr
, close
, fd_t fd
) {
77 return _REAL(close
)(fd
);
80 extern "C" int _REAL64(open
)(const char *, int, ...);
82 uptr
internal_open(const char *filename
, int flags
) {
83 return _REAL64(open
)(filename
, flags
);
86 uptr
internal_open(const char *filename
, int flags
, u32 mode
) {
87 return _REAL64(open
)(filename
, flags
, mode
);
90 uptr
OpenFile(const char *filename
, bool write
) {
91 return internal_open(filename
,
92 write
? O_WRONLY
| O_CREAT
: O_RDONLY
, 0660);
95 DECLARE__REAL_AND_INTERNAL(uptr
, read
, fd_t fd
, void *buf
, uptr count
) {
96 return _REAL(read
)(fd
, buf
, count
);
99 DECLARE__REAL_AND_INTERNAL(uptr
, write
, fd_t fd
, const void *buf
, uptr count
) {
100 return _REAL(write
)(fd
, buf
, count
);
103 // FIXME: There's only _ftruncate64 beginning with Solaris 11.
104 DECLARE__REAL_AND_INTERNAL(uptr
, ftruncate
, fd_t fd
, uptr size
) {
105 return ftruncate(fd
, size
);
108 DECLARE__REAL_AND_INTERNAL64(uptr
, stat
, const char *path
, void *buf
) {
109 return _REAL64(stat
)(path
, (struct stat
*)buf
);
112 DECLARE__REAL_AND_INTERNAL64(uptr
, lstat
, const char *path
, void *buf
) {
113 return _REAL64(lstat
)(path
, (struct stat
*)buf
);
116 DECLARE__REAL_AND_INTERNAL64(uptr
, fstat
, fd_t fd
, void *buf
) {
117 return _REAL64(fstat
)(fd
, (struct stat
*)buf
);
120 uptr
internal_filesize(fd_t fd
) {
122 if (internal_fstat(fd
, &st
))
124 return (uptr
)st
.st_size
;
127 DECLARE__REAL_AND_INTERNAL(uptr
, dup2
, int oldfd
, int newfd
) {
128 return _REAL(dup2
)(oldfd
, newfd
);
131 DECLARE__REAL_AND_INTERNAL(uptr
, readlink
, const char *path
, char *buf
,
133 return _REAL(readlink
)(path
, buf
, bufsize
);
136 DECLARE__REAL_AND_INTERNAL(uptr
, unlink
, const char *path
) {
137 return _REAL(unlink
)(path
);
140 DECLARE__REAL_AND_INTERNAL(uptr
, rename
, const char *oldpath
,
141 const char *newpath
) {
142 return _REAL(rename
)(oldpath
, newpath
);
145 DECLARE__REAL_AND_INTERNAL(uptr
, sched_yield
, void) {
146 return sched_yield();
149 DECLARE__REAL_AND_INTERNAL(void, _exit
, int exitcode
) {
153 DECLARE__REAL_AND_INTERNAL(uptr
, execve
, const char *filename
,
154 char *const argv
[], char *const envp
[]) {
155 return _REAL(execve
)(filename
, argv
, envp
);
158 DECLARE__REAL_AND_INTERNAL(uptr
, waitpid
, int pid
, int *status
, int options
) {
159 return _REAL(waitpid
)(pid
, status
, options
);
162 DECLARE__REAL_AND_INTERNAL(uptr
, getpid
, void) {
163 return _REAL(getpid
)();
166 // FIXME: This might be wrong: _getdents doesn't take a struct linux_dirent *.
167 DECLARE__REAL_AND_INTERNAL64(uptr
, getdents
, fd_t fd
, struct linux_dirent
*dirp
,
168 unsigned int count
) {
169 return _REAL64(getdents
)(fd
, dirp
, count
);
172 DECLARE__REAL_AND_INTERNAL64(uptr
, lseek
, fd_t fd
, OFF_T offset
, int whence
) {
173 return _REAL64(lseek
)(fd
, offset
, whence
);
176 // FIXME: This might be wrong: _sigfillset doesn't take a
177 // __sanitizer_sigset_t *.
178 DECLARE__REAL_AND_INTERNAL(void, sigfillset
, __sanitizer_sigset_t
*set
) {
179 _REAL(sigfillset
)(set
);
182 // FIXME: This might be wrong: _sigprocmask doesn't take __sanitizer_sigset_t *.
183 DECLARE__REAL_AND_INTERNAL(uptr
, sigprocmask
, int how
,
184 __sanitizer_sigset_t
*set
,
185 __sanitizer_sigset_t
*oldset
) {
186 return _REAL(sigprocmask
)(how
, set
, oldset
);
189 DECLARE__REAL_AND_INTERNAL(int, fork
, void) {
190 // TODO(glider): this may call user's pthread_atfork() handlers which is bad.
191 return _REAL(fork
)();
198 uptr
internal_clock_gettime(__sanitizer_clockid_t clk_id
, void *tp
) {
199 // FIXME: No internal variant.
200 return clock_gettime(clk_id
, (timespec
*)tp
);
203 // ----------------- sanitizer_common.h
204 BlockingMutex::BlockingMutex() {
205 CHECK(sizeof(mutex_t
) <= sizeof(opaque_storage_
));
206 internal_memset(this, 0, sizeof(*this));
207 CHECK_EQ(mutex_init((mutex_t
*)&opaque_storage_
, USYNC_THREAD
, NULL
), 0);
210 void BlockingMutex::Lock() {
211 CHECK(sizeof(mutex_t
) <= sizeof(opaque_storage_
));
212 CHECK_NE(owner_
, (uptr
)thr_self());
213 CHECK_EQ(mutex_lock((mutex_t
*)&opaque_storage_
), 0);
215 owner_
= (uptr
)thr_self();
218 void BlockingMutex::Unlock() {
219 CHECK(owner_
== (uptr
)thr_self());
221 CHECK_EQ(mutex_unlock((mutex_t
*)&opaque_storage_
), 0);
224 void BlockingMutex::CheckLocked() {
225 CHECK_EQ((uptr
)thr_self(), owner_
);
228 } // namespace __sanitizer
230 #endif // SANITIZER_SOLARIS