231 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			231 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===-- sanitizer_solaris.cpp ---------------------------------------------===//
 | |
| //
 | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | |
| // See https://llvm.org/LICENSE.txt for license information.
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // This file is shared between various sanitizers' runtime libraries and
 | |
| // implements Solaris-specific functions.
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "sanitizer_platform.h"
 | |
| #if SANITIZER_SOLARIS
 | |
| 
 | |
| #include <stdio.h>
 | |
| 
 | |
| #include "sanitizer_common.h"
 | |
| #include "sanitizer_flags.h"
 | |
| #include "sanitizer_internal_defs.h"
 | |
| #include "sanitizer_libc.h"
 | |
| #include "sanitizer_placement_new.h"
 | |
| #include "sanitizer_platform_limits_posix.h"
 | |
| #include "sanitizer_procmaps.h"
 | |
| 
 | |
| #include <fcntl.h>
 | |
| #include <pthread.h>
 | |
| #include <sched.h>
 | |
| #include <thread.h>
 | |
| #include <synch.h>
 | |
| #include <signal.h>
 | |
| #include <sys/mman.h>
 | |
| #include <sys/resource.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/types.h>
 | |
| #include <dirent.h>
 | |
| #include <unistd.h>
 | |
| #include <errno.h>
 | |
| #include <stdlib.h>
 | |
| 
 | |
| namespace __sanitizer {
 | |
| 
 | |
| //#include "sanitizer_syscall_generic.inc"
 | |
| 
 | |
| #define _REAL(func) _ ## func
 | |
| #define DECLARE__REAL(ret_type, func, ...) \
 | |
|   extern "C" ret_type _REAL(func)(__VA_ARGS__)
 | |
| #define DECLARE__REAL_AND_INTERNAL(ret_type, func, ...) \
 | |
|   DECLARE__REAL(ret_type, func, __VA_ARGS__); \
 | |
|   ret_type internal_ ## func(__VA_ARGS__)
 | |
| 
 | |
| #if !defined(_LP64) && _FILE_OFFSET_BITS == 64
 | |
| #define _REAL64(func) _ ## func ## 64
 | |
| #else
 | |
| #define _REAL64(func) _REAL(func)
 | |
| #endif
 | |
| #define DECLARE__REAL64(ret_type, func, ...) \
 | |
|   extern "C" ret_type _REAL64(func)(__VA_ARGS__)
 | |
| #define DECLARE__REAL_AND_INTERNAL64(ret_type, func, ...) \
 | |
|   DECLARE__REAL64(ret_type, func, __VA_ARGS__); \
 | |
|   ret_type internal_ ## func(__VA_ARGS__)
 | |
| 
 | |
| // ---------------------- sanitizer_libc.h
 | |
| DECLARE__REAL_AND_INTERNAL64(uptr, mmap, void *addr, uptr /*size_t*/ length,
 | |
|                              int prot, int flags, int fd, OFF_T offset) {
 | |
|   return (uptr)_REAL64(mmap)(addr, length, prot, flags, fd, offset);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, munmap, void *addr, uptr length) {
 | |
|   return _REAL(munmap)(addr, length);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(int, mprotect, void *addr, uptr length, int prot) {
 | |
|   return _REAL(mprotect)(addr, length, prot);
 | |
| }
 | |
| 
 | |
| // Illumos' declaration of madvise cannot be made visible if _XOPEN_SOURCE
 | |
| // is defined as g++ does on Solaris.
 | |
| //
 | |
| // This declaration is consistent with Solaris 11.4. Both Illumos and Solaris
 | |
| // versions older than 11.4 declared madvise with a caddr_t as the first
 | |
| // argument, but we don't currently support Solaris versions older than 11.4,
 | |
| // and as mentioned above the declaration is not visible on Illumos so we can
 | |
| // use any declaration we like on Illumos.
 | |
| extern "C" int madvise(void *, size_t, int);
 | |
| 
 | |
| int internal_madvise(uptr addr, uptr length, int advice) {
 | |
|   return madvise((void *)addr, length, advice);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, close, fd_t fd) {
 | |
|   return _REAL(close)(fd);
 | |
| }
 | |
| 
 | |
| extern "C" int _REAL64(open)(const char *, int, ...);
 | |
| 
 | |
| uptr internal_open(const char *filename, int flags) {
 | |
|   return _REAL64(open)(filename, flags);
 | |
| }
 | |
| 
 | |
| uptr internal_open(const char *filename, int flags, u32 mode) {
 | |
|   return _REAL64(open)(filename, flags, mode);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, read, fd_t fd, void *buf, uptr count) {
 | |
|   return _REAL(read)(fd, buf, count);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, write, fd_t fd, const void *buf, uptr count) {
 | |
|   return _REAL(write)(fd, buf, count);
 | |
| }
 | |
| 
 | |
| // FIXME: There's only _ftruncate64 beginning with Solaris 11.
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, ftruncate, fd_t fd, uptr size) {
 | |
|   return ftruncate(fd, size);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL64(uptr, stat, const char *path, void *buf) {
 | |
|   return _REAL64(stat)(path, (struct stat *)buf);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL64(uptr, lstat, const char *path, void *buf) {
 | |
|   return _REAL64(lstat)(path, (struct stat *)buf);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL64(uptr, fstat, fd_t fd, void *buf) {
 | |
|   return _REAL64(fstat)(fd, (struct stat *)buf);
 | |
| }
 | |
| 
 | |
| uptr internal_filesize(fd_t fd) {
 | |
|   struct stat st;
 | |
|   if (internal_fstat(fd, &st))
 | |
|     return -1;
 | |
|   return (uptr)st.st_size;
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, dup, int oldfd) {
 | |
|   return _REAL(dup)(oldfd);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, dup2, int oldfd, int newfd) {
 | |
|   return _REAL(dup2)(oldfd, newfd);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, readlink, const char *path, char *buf,
 | |
|                            uptr bufsize) {
 | |
|   return _REAL(readlink)(path, buf, bufsize);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, unlink, const char *path) {
 | |
|   return _REAL(unlink)(path);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, rename, const char *oldpath,
 | |
|                            const char *newpath) {
 | |
|   return _REAL(rename)(oldpath, newpath);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, sched_yield, void) {
 | |
|   return sched_yield();
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(void, usleep, u64 useconds) {
 | |
|   struct timespec ts;
 | |
|   ts.tv_sec = useconds / 1000000;
 | |
|   ts.tv_nsec = (useconds % 1000000) * 1000;
 | |
|   nanosleep(&ts, nullptr);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, execve, const char *filename,
 | |
|                            char *const argv[], char *const envp[]) {
 | |
|   return _REAL(execve)(filename, argv, envp);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, waitpid, int pid, int *status, int options) {
 | |
|   return _REAL(waitpid)(pid, status, options);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, getpid, void) {
 | |
|   return _REAL(getpid)();
 | |
| }
 | |
| 
 | |
| // FIXME: This might be wrong: _getdents doesn't take a struct linux_dirent *.
 | |
| DECLARE__REAL_AND_INTERNAL64(uptr, getdents, fd_t fd, struct linux_dirent *dirp,
 | |
|                              unsigned int count) {
 | |
|   return _REAL64(getdents)(fd, dirp, count);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL64(uptr, lseek, fd_t fd, OFF_T offset, int whence) {
 | |
|   return _REAL64(lseek)(fd, offset, whence);
 | |
| }
 | |
| 
 | |
| // FIXME: This might be wrong: _sigfillset doesn't take a
 | |
| // __sanitizer_sigset_t *.
 | |
| DECLARE__REAL_AND_INTERNAL(void, sigfillset, __sanitizer_sigset_t *set) {
 | |
|   _REAL(sigfillset)(set);
 | |
| }
 | |
| 
 | |
| // FIXME: This might be wrong: _sigprocmask doesn't take __sanitizer_sigset_t *.
 | |
| DECLARE__REAL_AND_INTERNAL(uptr, sigprocmask, int how,
 | |
|                            __sanitizer_sigset_t *set,
 | |
|                            __sanitizer_sigset_t *oldset) {
 | |
|   return _REAL(sigprocmask)(how, set, oldset);
 | |
| }
 | |
| 
 | |
| DECLARE__REAL_AND_INTERNAL(int, fork, void) {
 | |
|   // TODO(glider): this may call user's pthread_atfork() handlers which is bad.
 | |
|   return _REAL(fork)();
 | |
| }
 | |
| 
 | |
| u64 NanoTime() {
 | |
|   return gethrtime();
 | |
| }
 | |
| 
 | |
| uptr internal_clock_gettime(__sanitizer_clockid_t clk_id, void *tp) {
 | |
|   // FIXME: No internal variant.
 | |
|   return clock_gettime(clk_id, (timespec *)tp);
 | |
| }
 | |
| 
 | |
| // ----------------- sanitizer_common.h
 | |
| void FutexWait(atomic_uint32_t *p, u32 cmp) {
 | |
|   // FIXME: implement actual blocking.
 | |
|   sched_yield();
 | |
| }
 | |
| 
 | |
| void FutexWake(atomic_uint32_t *p, u32 count) {}
 | |
| 
 | |
| }  // namespace __sanitizer
 | |
| 
 | |
| #endif  // SANITIZER_SOLARIS
 |