| |
| /*--------------------------------------------------------------------*/ |
| /*--- File- and socket-related libc stuff. m_libcfile.c ---*/ |
| /*--------------------------------------------------------------------*/ |
| |
| /* |
| This file is part of Valgrind, a dynamic binary instrumentation |
| framework. |
| |
| Copyright (C) 2000-2011 Julian Seward |
| jseward@acm.org |
| |
| This program is free software; you can redistribute it and/or |
| modify it under the terms of the GNU General Public License as |
| published by the Free Software Foundation; either version 2 of the |
| License, or (at your option) any later version. |
| |
| This program is distributed in the hope that it will be useful, but |
| WITHOUT ANY WARRANTY; without even the implied warranty of |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| General Public License for more details. |
| |
| You should have received a copy of the GNU General Public License |
| along with this program; if not, write to the Free Software |
| Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA |
| 02111-1307, USA. |
| |
| The GNU General Public License is contained in the file COPYING. |
| */ |
| |
| #include "pub_core_basics.h" |
| #include "pub_core_vki.h" |
| #include "pub_core_vkiscnums.h" |
| #include "pub_core_debuglog.h" |
| #include "pub_core_libcbase.h" |
| #include "pub_core_libcassert.h" |
| #include "pub_core_libcfile.h" |
| #include "pub_core_libcprint.h" // VG_(sprintf) |
| #include "pub_core_libcproc.h" // VG_(getpid), VG_(getppid) |
| #include "pub_core_xarray.h" |
| #include "pub_core_clientstate.h" // VG_(fd_hard_limit) |
| #include "pub_core_syscall.h" |
| |
| /* IMPORTANT: on Darwin it is essential to use the _nocancel versions |
| of syscalls rather than the vanilla version, if a _nocancel version |
| is available. See docs/internals/Darwin-notes.txt for the reason |
| why. */ |
| |
| /* --------------------------------------------------------------------- |
| File stuff |
| ------------------------------------------------------------------ */ |
| |
| static inline Bool fd_exists(Int fd) |
| { |
| struct vg_stat st; |
| return VG_(fstat)(fd, &st) == 0; |
| } |
| |
| /* Move an fd into the Valgrind-safe range */ |
| Int VG_(safe_fd)(Int oldfd) |
| { |
| Int newfd; |
| |
| vg_assert(VG_(fd_hard_limit) != -1); |
| |
| newfd = VG_(fcntl)(oldfd, VKI_F_DUPFD, VG_(fd_hard_limit)); |
| if (newfd != -1) |
| VG_(close)(oldfd); |
| |
| /* Set the close-on-exec flag for this fd. */ |
| VG_(fcntl)(newfd, VKI_F_SETFD, VKI_FD_CLOEXEC); |
| |
| vg_assert(newfd >= VG_(fd_hard_limit)); |
| return newfd; |
| } |
| |
| /* Given a file descriptor, attempt to deduce its filename. To do |
| this, we use /proc/self/fd/<FD>. If this doesn't point to a file, |
| or if it doesn't exist, we return False. */ |
| Bool VG_(resolve_filename) ( Int fd, HChar* buf, Int n_buf ) |
| { |
| # if defined(VGO_linux) |
| HChar tmp[64]; |
| VG_(sprintf)(tmp, "/proc/self/fd/%d", fd); |
| VG_(memset)(buf, 0, n_buf); |
| if (VG_(readlink)(tmp, buf, n_buf) > 0 && buf[0] == '/') |
| return True; |
| else |
| return False; |
| |
| # elif defined(VGO_darwin) |
| HChar tmp[VKI_MAXPATHLEN+1]; |
| if (0 == VG_(fcntl)(fd, VKI_F_GETPATH, (UWord)tmp)) { |
| if (n_buf > 0) { |
| VG_(strncpy)( buf, tmp, n_buf < sizeof(tmp) ? n_buf : sizeof(tmp) ); |
| buf[n_buf-1] = 0; |
| } |
| if (tmp[0] == '/') return True; |
| } |
| return False; |
| |
| # else |
| # error Unknown OS |
| # endif |
| } |
| |
| SysRes VG_(mknod) ( const Char* pathname, Int mode, UWord dev ) |
| { |
| # if defined(VGO_linux) || defined(VGO_darwin) |
| SysRes res = VG_(do_syscall3)(__NR_mknod, |
| (UWord)pathname, mode, dev); |
| # else |
| # error Unknown OS |
| # endif |
| return res; |
| } |
| |
| SysRes VG_(open) ( const Char* pathname, Int flags, Int mode ) |
| { |
| # if defined(VGO_linux) |
| SysRes res = VG_(do_syscall3)(__NR_open, |
| (UWord)pathname, flags, mode); |
| # elif defined(VGO_darwin) |
| SysRes res = VG_(do_syscall3)(__NR_open_nocancel, |
| (UWord)pathname, flags, mode); |
| # else |
| # error Unknown OS |
| # endif |
| return res; |
| } |
| |
| Int VG_(fd_open) (const Char* pathname, Int flags, Int mode) |
| { |
| SysRes sr; |
| sr = VG_(open) (pathname, flags, mode); |
| if (sr_isError (sr)) |
| return -1; |
| else |
| return sr_Res (sr); |
| } |
| |
| void VG_(close) ( Int fd ) |
| { |
| /* Hmm. Return value is not checked. That's uncool. */ |
| # if defined(VGO_linux) |
| (void)VG_(do_syscall1)(__NR_close, fd); |
| # elif defined(VGO_darwin) |
| (void)VG_(do_syscall1)(__NR_close_nocancel, fd); |
| # else |
| # error Unknown OS |
| # endif |
| } |
| |
| Int VG_(read) ( Int fd, void* buf, Int count) |
| { |
| Int ret; |
| # if defined(VGO_linux) |
| SysRes res = VG_(do_syscall3)(__NR_read, fd, (UWord)buf, count); |
| # elif defined(VGO_darwin) |
| SysRes res = VG_(do_syscall3)(__NR_read_nocancel, fd, (UWord)buf, count); |
| # else |
| # error Unknown OS |
| # endif |
| if (sr_isError(res)) { |
| ret = - (Int)(Word)sr_Err(res); |
| vg_assert(ret < 0); |
| } else { |
| ret = (Int)(Word)sr_Res(res); |
| vg_assert(ret >= 0); |
| } |
| return ret; |
| } |
| |
| Int VG_(write) ( Int fd, const void* buf, Int count) |
| { |
| Int ret; |
| # if defined(VGO_linux) |
| SysRes res = VG_(do_syscall3)(__NR_write, fd, (UWord)buf, count); |
| # elif defined(VGO_darwin) |
| SysRes res = VG_(do_syscall3)(__NR_write_nocancel, fd, (UWord)buf, count); |
| # else |
| # error "Unknown OS" |
| # endif |
| if (sr_isError(res)) { |
| ret = - (Int)(Word)sr_Err(res); |
| vg_assert(ret < 0); |
| } else { |
| ret = (Int)(Word)sr_Res(res); |
| vg_assert(ret >= 0); |
| } |
| return ret; |
| } |
| |
| |
| Int VG_(pipe) ( Int fd[2] ) |
| { |
| # if defined(VGO_linux) |
| SysRes res = VG_(do_syscall1)(__NR_pipe, (UWord)fd); |
| return sr_isError(res) ? -1 : 0; |
| # elif defined(VGO_darwin) |
| /* __NR_pipe is UX64, so produces a double-word result */ |
| SysRes res = VG_(do_syscall0)(__NR_pipe); |
| if (!sr_isError(res)) { |
| fd[0] = (Int)sr_Res(res); |
| fd[1] = (Int)sr_ResHI(res); |
| } |
| return sr_isError(res) ? -1 : 0; |
| # else |
| # error "Unknown OS" |
| # endif |
| } |
| |
| Off64T VG_(lseek) ( Int fd, Off64T offset, Int whence ) |
| { |
| # if defined(VGO_linux) || defined(VGP_amd64_darwin) |
| # if defined(__NR__llseek) |
| Off64T result; |
| SysRes res = VG_(do_syscall5)(__NR__llseek, fd, |
| offset >> 32, offset & 0xffffffff, |
| (UWord)&result, whence); |
| return sr_isError(res) ? (-1) : result; |
| # else |
| SysRes res = VG_(do_syscall3)(__NR_lseek, fd, offset, whence); |
| vg_assert(sizeof(Off64T) == sizeof(Word)); |
| return sr_isError(res) ? (-1) : sr_Res(res); |
| # endif |
| # elif defined(VGP_x86_darwin) |
| SysRes res = VG_(do_syscall4)(__NR_lseek, fd, |
| offset & 0xffffffff, offset >> 32, whence); |
| return sr_isError(res) ? (-1) : sr_Res(res); |
| # else |
| # error "Unknown plat" |
| # endif |
| /* if you change the error-reporting conventions of this, also |
| change VG_(pread) and all other usage points. */ |
| } |
| |
| extern Int VG_(ftruncate) ( Int fd, OffT length ) { |
| #if defined (VGO_linux) |
| SysRes res = VG_(do_syscall2)(__NR_ftruncate, fd, length); |
| return sr_isError(res) ? (-1) : sr_Res(res); |
| #else |
| return -1; /*UNIMPLEMENTED*/ |
| #endif |
| } |
| |
| /* stat/fstat support. It's uggerly. We have impedance-match into a |
| 'struct vg_stat' in order to have a single structure that callers |
| can use consistently on all platforms. */ |
| |
| #define TRANSLATE_TO_vg_stat(_p_vgstat, _p_vkistat) \ |
| do { \ |
| (_p_vgstat)->dev = (ULong)( (_p_vkistat)->st_dev ); \ |
| (_p_vgstat)->ino = (ULong)( (_p_vkistat)->st_ino ); \ |
| (_p_vgstat)->nlink = (ULong)( (_p_vkistat)->st_nlink ); \ |
| (_p_vgstat)->mode = (UInt) ( (_p_vkistat)->st_mode ); \ |
| (_p_vgstat)->uid = (UInt) ( (_p_vkistat)->st_uid ); \ |
| (_p_vgstat)->gid = (UInt) ( (_p_vkistat)->st_gid ); \ |
| (_p_vgstat)->rdev = (ULong)( (_p_vkistat)->st_rdev ); \ |
| (_p_vgstat)->size = (Long) ( (_p_vkistat)->st_size ); \ |
| (_p_vgstat)->blksize = (ULong)( (_p_vkistat)->st_blksize ); \ |
| (_p_vgstat)->blocks = (ULong)( (_p_vkistat)->st_blocks ); \ |
| (_p_vgstat)->atime = (ULong)( (_p_vkistat)->st_atime ); \ |
| (_p_vgstat)->atime_nsec = (ULong)( (_p_vkistat)->st_atime_nsec ); \ |
| (_p_vgstat)->mtime = (ULong)( (_p_vkistat)->st_mtime ); \ |
| (_p_vgstat)->mtime_nsec = (ULong)( (_p_vkistat)->st_mtime_nsec ); \ |
| (_p_vgstat)->ctime = (ULong)( (_p_vkistat)->st_ctime ); \ |
| (_p_vgstat)->ctime_nsec = (ULong)( (_p_vkistat)->st_ctime_nsec ); \ |
| } while (0) |
| |
| SysRes VG_(stat) ( const Char* file_name, struct vg_stat* vgbuf ) |
| { |
| SysRes res; |
| VG_(memset)(vgbuf, 0, sizeof(*vgbuf)); |
| |
| # if defined(VGO_linux) || defined(VGO_darwin) |
| /* First try with stat64. If that doesn't work out, fall back to |
| the vanilla version. */ |
| # if defined(__NR_stat64) |
| { struct vki_stat64 buf64; |
| res = VG_(do_syscall2)(__NR_stat64, (UWord)file_name, (UWord)&buf64); |
| if (!(sr_isError(res) && sr_Err(res) == VKI_ENOSYS)) { |
| /* Success, or any failure except ENOSYS */ |
| if (!sr_isError(res)) |
| TRANSLATE_TO_vg_stat(vgbuf, &buf64); |
| return res; |
| } |
| } |
| # endif /* defined(__NR_stat64) */ |
| { struct vki_stat buf; |
| res = VG_(do_syscall2)(__NR_stat, (UWord)file_name, (UWord)&buf); |
| if (!sr_isError(res)) |
| TRANSLATE_TO_vg_stat(vgbuf, &buf); |
| return res; |
| } |
| |
| # else |
| # error Unknown OS |
| # endif |
| } |
| |
| Int VG_(fstat) ( Int fd, struct vg_stat* vgbuf ) |
| { |
| SysRes res; |
| VG_(memset)(vgbuf, 0, sizeof(*vgbuf)); |
| |
| # if defined(VGO_linux) || defined(VGO_darwin) |
| /* First try with fstat64. If that doesn't work out, fall back to |
| the vanilla version. */ |
| # if defined(__NR_fstat64) |
| { struct vki_stat64 buf64; |
| res = VG_(do_syscall2)(__NR_fstat64, (UWord)fd, (UWord)&buf64); |
| if (!(sr_isError(res) && sr_Err(res) == VKI_ENOSYS)) { |
| /* Success, or any failure except ENOSYS */ |
| if (!sr_isError(res)) |
| TRANSLATE_TO_vg_stat(vgbuf, &buf64); |
| return sr_isError(res) ? (-1) : 0; |
| } |
| } |
| # endif /* if defined(__NR_fstat64) */ |
| { struct vki_stat buf; |
| res = VG_(do_syscall2)(__NR_fstat, (UWord)fd, (UWord)&buf); |
| if (!sr_isError(res)) |
| TRANSLATE_TO_vg_stat(vgbuf, &buf); |
| return sr_isError(res) ? (-1) : 0; |
| } |
| |
| # else |
| # error Unknown OS |
| # endif |
| } |
| |
| #undef TRANSLATE_TO_vg_stat |
| |
| |
| Long VG_(fsize) ( Int fd ) |
| { |
| struct vg_stat buf; |
| Int res = VG_(fstat)( fd, &buf ); |
| return (res == -1) ? (-1LL) : buf.size; |
| } |
| |
| Bool VG_(is_dir) ( const HChar* f ) |
| { |
| struct vg_stat buf; |
| SysRes res = VG_(stat)(f, &buf); |
| return sr_isError(res) ? False |
| : VKI_S_ISDIR(buf.mode) ? True : False; |
| } |
| |
| SysRes VG_(dup) ( Int oldfd ) |
| { |
| return VG_(do_syscall1)(__NR_dup, oldfd); |
| } |
| |
| SysRes VG_(dup2) ( Int oldfd, Int newfd ) |
| { |
| # if defined(VGO_linux) || defined(VGO_darwin) |
| return VG_(do_syscall2)(__NR_dup2, oldfd, newfd); |
| # else |
| # error Unknown OS |
| # endif |
| } |
| |
| /* Returns -1 on error. */ |
| Int VG_(fcntl) ( Int fd, Int cmd, Addr arg ) |
| { |
| # if defined(VGO_linux) |
| SysRes res = VG_(do_syscall3)(__NR_fcntl, fd, cmd, arg); |
| # elif defined(VGO_darwin) |
| SysRes res = VG_(do_syscall3)(__NR_fcntl_nocancel, fd, cmd, arg); |
| # else |
| # error "Unknown OS" |
| # endif |
| return sr_isError(res) ? -1 : sr_Res(res); |
| } |
| |
| Int VG_(rename) ( const Char* old_name, const Char* new_name ) |
| { |
| SysRes res = VG_(do_syscall2)(__NR_rename, (UWord)old_name, (UWord)new_name); |
| return sr_isError(res) ? (-1) : 0; |
| } |
| |
| Int VG_(unlink) ( const Char* file_name ) |
| { |
| SysRes res = VG_(do_syscall1)(__NR_unlink, (UWord)file_name); |
| return sr_isError(res) ? (-1) : 0; |
| } |
| |
| /* The working directory at startup. AIX doesn't provide an easy |
| system call to do getcwd, but fortunately we don't need arbitrary |
| getcwd support. All that is really needed is to note the cwd at |
| process startup. Hence VG_(record_startup_wd) notes it (in a |
| platform dependent way) and VG_(get_startup_wd) produces the noted |
| value. Hence: */ |
| static HChar startup_wd[VKI_PATH_MAX]; |
| static Bool startup_wd_acquired = False; |
| |
| /* Record the process' working directory at startup. Is intended to |
| be called exactly once, at startup, before the working directory |
| changes. Return True for success, False for failure, so that the |
| caller can bomb out suitably without creating module cycles if |
| there is a problem. */ |
| Bool VG_(record_startup_wd) ( void ) |
| { |
| const Int szB = sizeof(startup_wd); |
| vg_assert(!startup_wd_acquired); |
| vg_assert(szB >= 512 && szB <= 16384/*let's say*/); /* stay sane */ |
| VG_(memset)(startup_wd, 0, szB); |
| # if defined(VGO_linux) |
| /* Simple: just ask the kernel */ |
| { SysRes res |
| = VG_(do_syscall2)(__NR_getcwd, (UWord)startup_wd, szB-1); |
| vg_assert(startup_wd[szB-1] == 0); |
| if (sr_isError(res)) { |
| return False; |
| } else { |
| startup_wd_acquired = True; |
| return True; |
| } |
| } |
| # elif defined(VGO_darwin) |
| /* We can't ask the kernel, so instead rely on launcher-*.c to |
| tell us the startup path. Note the env var is keyed to the |
| parent's PID, not ours, since our parent is the launcher |
| process. */ |
| { Char envvar[100]; |
| Char* wd = NULL; |
| VG_(memset)(envvar, 0, sizeof(envvar)); |
| VG_(sprintf)(envvar, "VALGRIND_STARTUP_PWD_%d_XYZZY", |
| (Int)VG_(getppid)()); |
| wd = VG_(getenv)( envvar ); |
| if (wd == NULL || (1+VG_(strlen)(wd) >= szB)) |
| return False; |
| VG_(strncpy_safely)(startup_wd, wd, szB); |
| vg_assert(startup_wd[szB-1] == 0); |
| startup_wd_acquired = True; |
| return True; |
| } |
| # else |
| # error Unknown OS |
| # endif |
| } |
| |
| /* Copy the previously acquired startup_wd into buf[0 .. size-1], |
| or return False if buf isn't big enough. */ |
| Bool VG_(get_startup_wd) ( Char* buf, SizeT size ) |
| { |
| vg_assert(startup_wd_acquired); |
| vg_assert(startup_wd[ sizeof(startup_wd)-1 ] == 0); |
| if (1+VG_(strlen)(startup_wd) >= size) |
| return False; |
| VG_(strncpy_safely)(buf, startup_wd, size); |
| return True; |
| } |
| |
| Int VG_(poll) (struct vki_pollfd *fds, Int nfds, Int timeout) |
| { |
| SysRes res; |
| res = VG_(do_syscall3)(__NR_poll, (UWord)fds, nfds, timeout); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| } |
| |
| |
| Int VG_(readlink) (const Char* path, Char* buf, UInt bufsiz) |
| { |
| SysRes res; |
| /* res = readlink( path, buf, bufsiz ); */ |
| res = VG_(do_syscall3)(__NR_readlink, (UWord)path, (UWord)buf, bufsiz); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| } |
| |
| Int VG_(getdents) (Int fd, struct vki_dirent *dirp, UInt count) |
| { |
| # if defined(VGO_linux) |
| SysRes res; |
| /* res = getdents( fd, dirp, count ); */ |
| res = VG_(do_syscall3)(__NR_getdents, fd, (UWord)dirp, count); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| # elif defined(VGO_darwin) |
| I_die_here; |
| # else |
| # error "Unknown OS" |
| # endif |
| } |
| |
| /* Check accessibility of a file. Returns zero for access granted, |
| nonzero otherwise. */ |
| Int VG_(access) ( const HChar* path, Bool irusr, Bool iwusr, Bool ixusr ) |
| { |
| # if defined(VGO_linux) |
| /* Very annoyingly, I cannot find any definition for R_OK et al in |
| the kernel interfaces. Therefore I reluctantly resort to |
| hardwiring in these magic numbers that I determined by |
| experimentation. */ |
| # define VKI_R_OK 4 |
| # define VKI_W_OK 2 |
| # define VKI_X_OK 1 |
| # endif |
| |
| UWord w = (irusr ? VKI_R_OK : 0) |
| | (iwusr ? VKI_W_OK : 0) |
| | (ixusr ? VKI_X_OK : 0); |
| SysRes res = VG_(do_syscall2)(__NR_access, (UWord)path, w); |
| return sr_isError(res) ? 1 : 0; |
| |
| # if defined(VGO_linux) |
| # undef VKI_R_OK |
| # undef VKI_W_OK |
| # undef VKI_X_OK |
| # endif |
| } |
| |
| /* |
| Emulate the normal Unix permissions checking algorithm. |
| |
| If owner matches, then use the owner permissions, else |
| if group matches, then use the group permissions, else |
| use other permissions. |
| |
| Note that we can't deal properly with SUID/SGID. By default |
| (allow_setuid == False), we refuse to run them (otherwise the |
| executable may misbehave if it doesn't have the permissions it |
| thinks it does). However, the caller may indicate that setuid |
| executables are allowed, for example if we are going to exec them |
| but not trace into them (iow, client sys_execve when |
| clo_trace_children == False). |
| |
| If VKI_EACCES is returned (iow, permission was refused), then |
| *is_setuid is set to True iff permission was refused because the |
| executable is setuid. |
| */ |
| /* returns: 0 = success, non-0 is failure */ |
| Int VG_(check_executable)(/*OUT*/Bool* is_setuid, |
| const HChar* f, Bool allow_setuid) |
| { |
| struct vg_stat st; |
| SysRes res = VG_(stat)(f, &st); |
| |
| if (is_setuid) |
| *is_setuid = False; |
| |
| if (sr_isError(res)) { |
| return sr_Err(res); |
| } |
| |
| if ( (st.mode & (VKI_S_ISUID | VKI_S_ISGID)) && !allow_setuid ) { |
| if (is_setuid) |
| *is_setuid = True; |
| return VKI_EACCES; |
| } |
| |
| if (VG_(geteuid)() == st.uid) { |
| if (!(st.mode & VKI_S_IXUSR)) |
| return VKI_EACCES; |
| } else { |
| Int grpmatch = 0; |
| |
| if (VG_(getegid)() == st.gid) |
| grpmatch = 1; |
| else { |
| UInt groups[32]; |
| Int ngrp = VG_(getgroups)(32, groups); |
| Int i; |
| /* ngrp will be -1 if VG_(getgroups) failed. */ |
| for (i = 0; i < ngrp; i++) { |
| if (groups[i] == st.gid) { |
| grpmatch = 1; |
| break; |
| } |
| } |
| } |
| |
| if (grpmatch) { |
| if (!(st.mode & VKI_S_IXGRP)) { |
| return VKI_EACCES; |
| } |
| } else if (!(st.mode & VKI_S_IXOTH)) { |
| return VKI_EACCES; |
| } |
| } |
| |
| return 0; |
| } |
| |
| /* DDD: Note this moves (or at least, is believed to move) the file |
| pointer on Linux but doesn't on Darwin. This inconsistency should |
| be fixed. (In other words, why isn't the Linux version implemented |
| in terms of pread()?) */ |
| SysRes VG_(pread) ( Int fd, void* buf, Int count, OffT offset ) |
| { |
| SysRes res; |
| # if defined(VGO_linux) |
| OffT off = VG_(lseek)( fd, offset, VKI_SEEK_SET); |
| if (off < 0) |
| return VG_(mk_SysRes_Error)( VKI_EINVAL ); |
| res = VG_(do_syscall3)(__NR_read, fd, (UWord)buf, count ); |
| return res; |
| # elif defined(VGP_amd64_darwin) |
| res = VG_(do_syscall4)(__NR_pread_nocancel, fd, (UWord)buf, count, offset); |
| return res; |
| # elif defined(VGP_x86_darwin) |
| /* ppc32-darwin is the same, but with the args inverted */ |
| res = VG_(do_syscall5)(__NR_pread_nocancel, fd, (UWord)buf, count, |
| offset & 0xffffffff, offset >> 32); |
| return res; |
| # else |
| # error "Unknown platform" |
| # endif |
| } |
| |
| /* Return the name of a directory for temporary files. */ |
| const HChar *VG_(tmpdir)(void) |
| { |
| const HChar *tmpdir; |
| |
| tmpdir = VG_(getenv)("TMPDIR"); |
| if (tmpdir == NULL || *tmpdir == '\0') tmpdir = VG_TMPDIR; |
| if (tmpdir == NULL || *tmpdir == '\0') tmpdir = "/tmp"; /* fallback */ |
| |
| return tmpdir; |
| } |
| |
| /* Create and open (-rw------) a tmp file name incorporating said arg. |
| Returns -1 on failure, else the fd of the file. If fullname is |
| non-NULL, the file's name is written into it. The number of bytes |
| written is guaranteed not to exceed 64+strlen(part_of_name). */ |
| |
| Int VG_(mkstemp) ( HChar* part_of_name, /*OUT*/HChar* fullname ) |
| { |
| HChar buf[200]; |
| Int n, tries, fd; |
| UInt seed; |
| SysRes sres; |
| const HChar *tmpdir; |
| |
| vg_assert(part_of_name); |
| n = VG_(strlen)(part_of_name); |
| vg_assert(n > 0 && n < 100); |
| |
| seed = (VG_(getpid)() << 9) ^ VG_(getppid)(); |
| |
| /* Determine sensible location for temporary files */ |
| tmpdir = VG_(tmpdir)(); |
| |
| tries = 0; |
| while (True) { |
| if (tries++ > 10) |
| return -1; |
| VG_(sprintf)( buf, "%s/valgrind_%s_%08x", |
| tmpdir, part_of_name, VG_(random)( &seed )); |
| if (0) |
| VG_(printf)("VG_(mkstemp): trying: %s\n", buf); |
| |
| sres = VG_(open)(buf, |
| VKI_O_CREAT|VKI_O_RDWR|VKI_O_EXCL|VKI_O_TRUNC, |
| VKI_S_IRUSR|VKI_S_IWUSR); |
| if (sr_isError(sres)) { |
| VG_(umsg)("VG_(mkstemp): failed to create temp file: %s\n", buf); |
| continue; |
| } |
| /* VG_(safe_fd) doesn't return if it fails. */ |
| fd = VG_(safe_fd)( sr_Res(sres) ); |
| if (fullname) |
| VG_(strcpy)( fullname, buf ); |
| return fd; |
| } |
| /* NOTREACHED */ |
| } |
| |
| |
| /* --------------------------------------------------------------------- |
| Socket-related stuff. |
| ------------------------------------------------------------------ */ |
| |
| static |
| Int parse_inet_addr_and_port ( UChar* str, UInt* ip_addr, UShort* port ); |
| |
| static |
| Int my_connect ( Int sockfd, struct vki_sockaddr_in* serv_addr, Int addrlen ); |
| |
| UInt VG_(htonl) ( UInt x ) |
| { |
| # if defined(VG_BIGENDIAN) |
| return x; |
| # else |
| return |
| (((x >> 24) & 0xFF) << 0) | (((x >> 16) & 0xFF) << 8) |
| | (((x >> 8) & 0xFF) << 16) | (((x >> 0) & 0xFF) << 24); |
| # endif |
| } |
| |
| UInt VG_(ntohl) ( UInt x ) |
| { |
| # if defined(VG_BIGENDIAN) |
| return x; |
| # else |
| return |
| (((x >> 24) & 0xFF) << 0) | (((x >> 16) & 0xFF) << 8) |
| | (((x >> 8) & 0xFF) << 16) | (((x >> 0) & 0xFF) << 24); |
| # endif |
| } |
| |
| UShort VG_(htons) ( UShort x ) |
| { |
| # if defined(VG_BIGENDIAN) |
| return x; |
| # else |
| return |
| (((x >> 8) & 0xFF) << 0) | (((x >> 0) & 0xFF) << 8); |
| # endif |
| } |
| |
| UShort VG_(ntohs) ( UShort x ) |
| { |
| # if defined(VG_BIGENDIAN) |
| return x; |
| # else |
| return |
| (((x >> 8) & 0xFF) << 0) | (((x >> 0) & 0xFF) << 8); |
| # endif |
| } |
| |
| |
| /* The main function. |
| |
| Supplied string contains either an ip address "192.168.0.1" or |
| an ip address and port pair, "192.168.0.1:1500". Parse these, |
| and return: |
| -1 if there is a parse error |
| -2 if no parse error, but specified host:port cannot be opened |
| the relevant file (socket) descriptor, otherwise. |
| is used. |
| */ |
| Int VG_(connect_via_socket)( UChar* str ) |
| { |
| # if defined(VGO_linux) || defined(VGO_darwin) |
| Int sd, res; |
| struct vki_sockaddr_in servAddr; |
| UInt ip = 0; |
| UShort port = VG_CLO_DEFAULT_LOGPORT; |
| Bool ok = parse_inet_addr_and_port(str, &ip, &port); |
| if (!ok) |
| return -1; |
| |
| //if (0) |
| // VG_(printf)("ip = %d.%d.%d.%d, port %d\n", |
| // (ip >> 24) & 0xFF, (ip >> 16) & 0xFF, |
| // (ip >> 8) & 0xFF, ip & 0xFF, |
| // (UInt)port ); |
| |
| servAddr.sin_family = VKI_AF_INET; |
| servAddr.sin_addr.s_addr = VG_(htonl)(ip); |
| servAddr.sin_port = VG_(htons)(port); |
| |
| /* create socket */ |
| sd = VG_(socket)(VKI_AF_INET, VKI_SOCK_STREAM, 0 /* IPPROTO_IP ? */); |
| if (sd < 0) { |
| /* this shouldn't happen ... nevertheless */ |
| return -2; |
| } |
| |
| /* connect to server */ |
| res = my_connect(sd, &servAddr, sizeof(servAddr)); |
| if (res < 0) { |
| /* connection failed */ |
| return -2; |
| } |
| |
| return sd; |
| |
| # else |
| # error "Unknown OS" |
| # endif |
| } |
| |
| |
| /* Let d = one or more digits. Accept either: |
| d.d.d.d or d.d.d.d:d |
| */ |
| static Int parse_inet_addr_and_port ( UChar* str, UInt* ip_addr, UShort* port ) |
| { |
| # define GET_CH ((*str) ? (*str++) : 0) |
| UInt ipa, i, j, c, any; |
| ipa = 0; |
| for (i = 0; i < 4; i++) { |
| j = 0; |
| any = 0; |
| while (1) { |
| c = GET_CH; |
| if (c < '0' || c > '9') break; |
| j = 10 * j + (int)(c - '0'); |
| any = 1; |
| } |
| if (any == 0 || j > 255) goto syntaxerr; |
| ipa = (ipa << 8) + j; |
| if (i <= 2 && c != '.') goto syntaxerr; |
| } |
| if (c == 0 || c == ':') |
| *ip_addr = ipa; |
| if (c == 0) goto ok; |
| if (c != ':') goto syntaxerr; |
| j = 0; |
| any = 0; |
| while (1) { |
| c = GET_CH; |
| if (c < '0' || c > '9') break; |
| j = j * 10 + (int)(c - '0'); |
| any = 1; |
| if (j > 65535) goto syntaxerr; |
| } |
| if (any == 0 || c != 0) goto syntaxerr; |
| if (j < 1024) goto syntaxerr; |
| *port = (UShort)j; |
| ok: |
| return 1; |
| syntaxerr: |
| return 0; |
| # undef GET_CH |
| } |
| |
| // GrP fixme safe_fd? |
| Int VG_(socket) ( Int domain, Int type, Int protocol ) |
| { |
| # if defined(VGP_x86_linux) || defined(VGP_ppc32_linux) \ |
| || defined(VGP_ppc64_linux) || defined(VGP_s390x_linux) |
| SysRes res; |
| UWord args[3]; |
| args[0] = domain; |
| args[1] = type; |
| args[2] = protocol; |
| res = VG_(do_syscall2)(__NR_socketcall, VKI_SYS_SOCKET, (UWord)&args); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGP_amd64_linux) || defined(VGP_arm_linux) |
| SysRes res; |
| res = VG_(do_syscall3)(__NR_socket, domain, type, protocol ); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGO_darwin) |
| SysRes res; |
| res = VG_(do_syscall3)(__NR_socket, domain, type, protocol); |
| if (!sr_isError(res)) { |
| // Set SO_NOSIGPIPE so write() returns EPIPE instead of raising SIGPIPE |
| Int optval = 1; |
| SysRes res2; |
| res2 = VG_(do_syscall5)(__NR_setsockopt, sr_Res(res), VKI_SOL_SOCKET, |
| VKI_SO_NOSIGPIPE, (UWord)&optval, |
| sizeof(optval)); |
| // ignore setsockopt() error |
| } |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # else |
| # error "Unknown arch" |
| # endif |
| } |
| |
| |
| static |
| Int my_connect ( Int sockfd, struct vki_sockaddr_in* serv_addr, Int addrlen ) |
| { |
| # if defined(VGP_x86_linux) || defined(VGP_ppc32_linux) \ |
| || defined(VGP_ppc64_linux) || defined(VGP_s390x_linux) |
| SysRes res; |
| UWord args[3]; |
| args[0] = sockfd; |
| args[1] = (UWord)serv_addr; |
| args[2] = addrlen; |
| res = VG_(do_syscall2)(__NR_socketcall, VKI_SYS_CONNECT, (UWord)&args); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGP_amd64_linux) || defined(VGP_arm_linux) |
| SysRes res; |
| res = VG_(do_syscall3)(__NR_connect, sockfd, (UWord)serv_addr, addrlen); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGO_darwin) |
| SysRes res; |
| res = VG_(do_syscall3)(__NR_connect_nocancel, |
| sockfd, (UWord)serv_addr, addrlen); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # else |
| # error "Unknown arch" |
| # endif |
| } |
| |
| Int VG_(write_socket)( Int sd, void *msg, Int count ) |
| { |
| /* This is actually send(). */ |
| |
| /* For Linux, VKI_MSG_NOSIGNAL is a request not to send SIGPIPE on |
| errors on stream oriented sockets when the other end breaks the |
| connection. The EPIPE error is still returned. |
| |
| For Darwin, VG_(socket)() sets SO_NOSIGPIPE to get EPIPE instead of |
| SIGPIPE */ |
| |
| # if defined(VGP_x86_linux) || defined(VGP_ppc32_linux) \ |
| || defined(VGP_ppc64_linux) || defined(VGP_s390x_linux) |
| SysRes res; |
| UWord args[4]; |
| args[0] = sd; |
| args[1] = (UWord)msg; |
| args[2] = count; |
| args[3] = VKI_MSG_NOSIGNAL; |
| res = VG_(do_syscall2)(__NR_socketcall, VKI_SYS_SEND, (UWord)&args); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGP_amd64_linux) || defined(VGP_arm_linux) |
| SysRes res; |
| res = VG_(do_syscall6)(__NR_sendto, sd, (UWord)msg, |
| count, VKI_MSG_NOSIGNAL, 0,0); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGP_x86_darwin) || defined(VGP_amd64_darwin) |
| SysRes res; |
| res = VG_(do_syscall3)(__NR_write_nocancel, sd, (UWord)msg, count); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # else |
| # error "Unknown platform" |
| # endif |
| } |
| |
| Int VG_(getsockname) ( Int sd, struct vki_sockaddr *name, Int *namelen) |
| { |
| # if defined(VGP_x86_linux) || defined(VGP_ppc32_linux) \ |
| || defined(VGP_ppc64_linux) || defined(VGP_s390x_linux) |
| SysRes res; |
| UWord args[3]; |
| args[0] = sd; |
| args[1] = (UWord)name; |
| args[2] = (UWord)namelen; |
| res = VG_(do_syscall2)(__NR_socketcall, VKI_SYS_GETSOCKNAME, (UWord)&args); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGP_amd64_linux) || defined(VGP_arm_linux) |
| SysRes res; |
| res = VG_(do_syscall3)( __NR_getsockname, |
| (UWord)sd, (UWord)name, (UWord)namelen ); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGO_darwin) |
| SysRes res; |
| res = VG_(do_syscall3)( __NR_getsockname, |
| (UWord)sd, (UWord)name, (UWord)namelen ); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # else |
| # error "Unknown platform" |
| # endif |
| } |
| |
| Int VG_(getpeername) ( Int sd, struct vki_sockaddr *name, Int *namelen) |
| { |
| # if defined(VGP_x86_linux) || defined(VGP_ppc32_linux) \ |
| || defined(VGP_ppc64_linux) || defined(VGP_s390x_linux) |
| SysRes res; |
| UWord args[3]; |
| args[0] = sd; |
| args[1] = (UWord)name; |
| args[2] = (UWord)namelen; |
| res = VG_(do_syscall2)(__NR_socketcall, VKI_SYS_GETPEERNAME, (UWord)&args); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGP_amd64_linux) || defined(VGP_arm_linux) |
| SysRes res; |
| res = VG_(do_syscall3)( __NR_getpeername, |
| (UWord)sd, (UWord)name, (UWord)namelen ); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGO_darwin) |
| SysRes res; |
| res = VG_(do_syscall3)( __NR_getpeername, |
| (UWord)sd, (UWord)name, (UWord)namelen ); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # else |
| # error "Unknown platform" |
| # endif |
| } |
| |
| Int VG_(getsockopt) ( Int sd, Int level, Int optname, void *optval, |
| Int *optlen) |
| { |
| # if defined(VGP_x86_linux) || defined(VGP_ppc32_linux) \ |
| || defined(VGP_ppc64_linux) || defined(VGP_s390x_linux) |
| SysRes res; |
| UWord args[5]; |
| args[0] = sd; |
| args[1] = level; |
| args[2] = optname; |
| args[3] = (UWord)optval; |
| args[4] = (UWord)optlen; |
| res = VG_(do_syscall2)(__NR_socketcall, VKI_SYS_GETSOCKOPT, (UWord)&args); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGP_amd64_linux) || defined(VGP_arm_linux) |
| SysRes res; |
| res = VG_(do_syscall5)( __NR_getsockopt, |
| (UWord)sd, (UWord)level, (UWord)optname, |
| (UWord)optval, (UWord)optlen ); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # elif defined(VGO_darwin) |
| SysRes res; |
| res = VG_(do_syscall5)( __NR_getsockopt, |
| (UWord)sd, (UWord)level, (UWord)optname, |
| (UWord)optval, (UWord)optlen ); |
| return sr_isError(res) ? -1 : sr_Res(res); |
| |
| # else |
| # error "Unknown platform" |
| # endif |
| } |
| |
| |
| Char *VG_(basename)(const Char *path) |
| { |
| static Char buf[VKI_PATH_MAX]; |
| |
| const Char *p, *end; |
| |
| if (path == NULL || |
| 0 == VG_(strcmp)(path, "")) |
| { |
| return "."; |
| } |
| |
| p = path + VG_(strlen)(path); |
| while (p > path && *p == '/') { |
| // skip all trailing '/' |
| p--; |
| } |
| |
| if (p == path && *p == '/') return "/"; // all slashes |
| |
| end = p; |
| |
| while (p > path && *p != '/') { |
| // now skip non '/' |
| p--; |
| } |
| |
| if (*p == '/') p++; |
| |
| VG_(strncpy)(buf, p, end-p+1); |
| buf[end-p+1] = '\0'; |
| |
| return buf; |
| } |
| |
| |
| Char *VG_(dirname)(const Char *path) |
| { |
| static Char buf[VKI_PATH_MAX]; |
| |
| const Char *p; |
| |
| if (path == NULL || |
| 0 == VG_(strcmp)(path, "") || |
| 0 == VG_(strcmp)(path, "/")) |
| { |
| return "."; |
| } |
| |
| p = path + VG_(strlen)(path); |
| while (p > path && *p == '/') { |
| // skip all trailing '/' |
| p--; |
| } |
| |
| while (p > path && *p != '/') { |
| // now skip non '/' |
| p--; |
| } |
| |
| if (p == path) { |
| if (*p == '/') return "/"; // all slashes |
| else return "."; // no slashes |
| } |
| |
| while (p > path && *p == '/') { |
| // skip '/' again |
| p--; |
| } |
| |
| VG_(strncpy)(buf, path, p-path+1); |
| buf[p-path+1] = '\0'; |
| |
| return buf; |
| } |
| |
| |
| /*--------------------------------------------------------------------*/ |
| /*--- end ---*/ |
| /*--------------------------------------------------------------------*/ |