/* Unix SMB/Netbios implementation. Version 2.0 SMB wrapper functions Copyright (C) Andrew Tridgell 1998 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., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "wrapper.h" #ifdef HAVE__OPEN __asm__(".globl _open; _open = open"); #elif HAVE___OPEN __asm__(".globl __open; __open = open"); #endif int open(const char *name, int flags, mode_t mode) { if (smbw_path(name)) { return smbw_open(name, flags, mode); } return real_open(name, flags, mode); } #ifdef HAVE___CHDIR __asm__(".globl __chdir; __chdir = chdir"); #elif HAVE__CHDIR __asm__(".globl _chdir; _chdir = chdir"); #endif int chdir(const char *name) { return smbw_chdir(name); } #ifdef HAVE___CLOSE __asm__(".globl __close; __close = close"); #elif HAVE__CLOSE __asm__(".globl _close; _close = close"); #endif ssize_t close(int fd) { if (smbw_fd(fd)) { return smbw_close(fd); } return real_close(fd); } #ifdef HAVE___FCHDIR __asm__(".globl __fchdir; __fchdir = fchdir"); #elif HAVE__FCHDIR __asm__(".globl _fchdir; _fchdir = fchdir"); #endif int fchdir(int fd) { if (smbw_fd(fd)) { return smbw_fchdir(fd); } return real_fchdir(fd); } #ifdef HAVE___FCNTL __asm__(".globl __fcntl; __fcntl = fcntl"); #elif HAVE__FCNTL __asm__(".globl _fcntl; _fcntl = fcntl"); #endif int fcntl(int fd, int cmd, long arg) { if (smbw_fd(fd)) { return smbw_fcntl(fd, cmd, arg); } return real_fcntl(fd, cmd, arg); } #ifdef HAVE___GETDENTS __asm__(".globl __getdents; __getdents = getdents"); #elif HAVE__GETDENTS __asm__(".globl _getdents; _getdents = getdents"); #endif int getdents(int fd, struct dirent *dirp, unsigned int count) { if (smbw_fd(fd)) { return smbw_getdents(fd, dirp, count); } return real_getdents(fd, dirp, count); } #ifdef HAVE___LSEEK __asm__(".globl __lseek; __lseek = lseek"); #elif HAVE__LSEEK __asm__(".globl _lseek; _lseek = lseek"); #endif ssize_t lseek(int fd, off_t offset, int whence) { if (smbw_fd(fd)) { return smbw_lseek(fd, offset, whence); } return real_lseek(fd, offset, whence); } #ifdef HAVE___READ __asm__(".globl __read; __read = read"); #elif HAVE__READ __asm__(".globl _read; _read = read"); #endif ssize_t read(int fd, void *buf, size_t count) { if (smbw_fd(fd)) { return smbw_read(fd, buf, count); } return real_read(fd, buf, count); } #ifdef HAVE___WRITE __asm__(".globl __write; __write = write"); #elif HAVE__WRITE __asm__(".globl _write; _write = write"); #endif ssize_t write(int fd, void *buf, size_t count) { if (smbw_fd(fd)) { return smbw_write(fd, buf, count); } return real_write(fd, buf, count); } int access(const char *name, int mode) { if (smbw_path(name)) { return smbw_access(name, mode); } return real_access(name, mode); } int chmod(const char *name,mode_t mode) { if (smbw_path(name)) { return smbw_chmod(name, mode); } return real_chmod(name, mode); } int chown(const char *name,uid_t owner, gid_t group) { if (smbw_path(name)) { return smbw_chown(name, owner, group); } return real_chown(name, owner, group); } #ifdef LINUX int __fxstat(int vers, int fd, struct stat *st) { struct kernel_stat kbuf; int ret; if (smbw_fd(fd)) { return smbw_fstat(fd, st); } switch (vers) { case _STAT_VER_LINUX_OLD: /* Nothing to do. The struct is in the form the kernel expects it to be. */ return real_fstat(fd, (struct kernel_stat *)st); break; case _STAT_VER_LINUX: /* Do the system call. */ ret = real_fstat(fd, &kbuf); st->st_dev = kbuf.st_dev; #ifdef _HAVE___PAD1 st->__pad1 = 0; #endif st->st_ino = kbuf.st_ino; st->st_mode = kbuf.st_mode; st->st_nlink = kbuf.st_nlink; st->st_uid = kbuf.st_uid; st->st_gid = kbuf.st_gid; st->st_rdev = kbuf.st_rdev; #ifdef _HAVE___PAD2 st->__pad2 = 0; #endif st->st_size = kbuf.st_size; st->st_blksize = kbuf.st_blksize; st->st_blocks = kbuf.st_blocks; st->st_atime = kbuf.st_atime; #ifdef _HAVE___UNUSED1 st->__unused1 = 0; #endif st->st_mtime = kbuf.st_mtime; #ifdef _HAVE___UNUSED2 st->__unused2 = 0; #endif st->st_ctime = kbuf.st_ctime; #ifdef _HAVE___UNUSED3 st->__unused3 = 0; #endif #ifdef _HAVE___UNUSED4 st->__unused4 = 0; #endif #ifdef _HAVE___UNUSED5 st->__unused5 = 0; #endif return ret; default: errno = EINVAL; return -1; } } #endif char *getcwd(char *buf, size_t size) { return smbw_getcwd(buf, size); } #ifdef LINUX int __lxstat(int vers, const char *name, struct stat *st) { struct kernel_stat kbuf; int ret; if (smbw_path(name)) { return smbw_stat(name, st); } switch (vers) { case _STAT_VER_LINUX_OLD: /* Nothing to do. The struct is in the form the kernel expects it to be. */ return real_lstat(name, (struct kernel_stat *)st); break; case _STAT_VER_LINUX: /* Do the system call. */ ret = real_lstat(name, &kbuf); st->st_dev = kbuf.st_dev; #ifdef _HAVE___PAD1 st->__pad1 = 0; #endif st->st_ino = kbuf.st_ino; st->st_mode = kbuf.st_mode; st->st_nlink = kbuf.st_nlink; st->st_uid = kbuf.st_uid; st->st_gid = kbuf.st_gid; st->st_rdev = kbuf.st_rdev; #ifdef _HAVE___PAD2 st->__pad2 = 0; #endif st->st_size = kbuf.st_size; st->st_blksize = kbuf.st_blksize; st->st_blocks = kbuf.st_blocks; st->st_atime = kbuf.st_atime; #ifdef _HAVE___UNUSED1 st->__unused1 = 0; #endif st->st_mtime = kbuf.st_mtime; #ifdef _HAVE___UNUSED2 st->__unused2 = 0; #endif st->st_ctime = kbuf.st_ctime; #ifdef _HAVE___UNUSED3 st->__unused3 = 0; #endif #ifdef _HAVE___UNUSED4 st->__unused4 = 0; #endif #ifdef _HAVE___UNUSED5 st->__unused5 = 0; #endif return ret; default: errno = EINVAL; return -1; } } #endif int mkdir(const char *name, mode_t mode) { if (smbw_path(name)) { return smbw_mkdir(name, mode); } return real_mkdir(name, mode); } #ifdef LINUX int __xstat(int vers, const char *name, struct stat *st) { struct kernel_stat kbuf; int ret; if (smbw_path(name)) { return smbw_stat(name, st); } switch (vers) { case _STAT_VER_LINUX_OLD: /* Nothing to do. The struct is in the form the kernel expects it to be. */ return real_stat(name, (struct kernel_stat *)st); break; case _STAT_VER_LINUX: /* Do the system call. */ ret = real_stat(name, &kbuf); st->st_dev = kbuf.st_dev; #ifdef _HAVE___PAD1 st->__pad1 = 0; #endif st->st_ino = kbuf.st_ino; st->st_mode = kbuf.st_mode; st->st_nlink = kbuf.st_nlink; st->st_uid = kbuf.st_uid; st->st_gid = kbuf.st_gid; st->st_rdev = kbuf.st_rdev; #ifdef _HAVE___PAD2 st->__pad2 = 0; #endif st->st_size = kbuf.st_size; st->st_blksize = kbuf.st_blksize; st->st_blocks = kbuf.st_blocks; st->st_atime = kbuf.st_atime; #ifdef _HAVE___UNUSED1 st->__unused1 = 0; #endif st->st_mtime = kbuf.st_mtime; #ifdef _HAVE___UNUSED2 st->__unused2 = 0; #endif st->st_ctime = kbuf.st_ctime; #ifdef _HAVE___UNUSED3 st->__unused3 = 0; #endif #ifdef _HAVE___UNUSED4 st->__unused4 = 0; #endif #ifdef _HAVE___UNUSED5 st->__unused5 = 0; #endif return ret; default: errno = EINVAL; return -1; } } #endif int stat(const char *name, struct stat *st) { #if HAVE___XSTAT return __xstat(_STAT_VER, name, st); #else if (smbw_path(name)) { return smbw_stat(name, st); } return real_stat(name, st); #endif } int lstat(const char *name, struct stat *st) { #if HAVE___LXSTAT return __lxstat(_STAT_VER, name, st); #else if (smbw_path(name)) { return smbw_stat(name, st); } return real_lstat(name, st); #endif } int fstat(int fd, struct stat *st) { #if HAVE___LXSTAT return __fxstat(_STAT_VER, fd, st); #else if (smbw_fd(fd)) { return smbw_fstat(fd, st); } return real_fstat(fd, st); #endif } int unlink(const char *name) { if (smbw_path(name)) { return smbw_unlink(name); } return real_unlink(name); } int utime(const char *name,void *tvp) { if (smbw_path(name)) { return smbw_utime(name, tvp); } return real_utime(name, tvp); } int readlink(char *path, char *buf, size_t bufsize) { if (smbw_path(path)) { return smbw_readlink(path, buf, bufsize); } return real_readlink(path, buf, bufsize); } int rename(const char *oldname,const char *newname) { int p1, p2; p1 = smbw_path(oldname); p2 = smbw_path(newname); if (p1 ^ p2) { /* can't cross filesystem boundaries */ errno = EXDEV; return -1; } if (p1 && p2) { return smbw_rename(oldname, newname); } return real_rename(oldname, newname); } int rmdir(const char *name) { if (smbw_path(name)) { return smbw_rmdir(name); } return real_rmdir(name); } int symlink(const char *topath,const char *frompath) { int p1, p2; p1 = smbw_path(topath); p2 = smbw_path(frompath); if (p1 || p2) { /* can't handle symlinks */ errno = EPERM; return -1; } return real_symlink(topath, frompath); } int dup(int fd) { if (smbw_fd(fd)) { return smbw_dup(fd); } return real_dup(fd); } int dup2(int oldfd, int newfd) { if (smbw_fd(newfd)) { close(newfd); } if (smbw_fd(oldfd)) { return smbw_dup2(oldfd, newfd); } return real_dup2(oldfd, newfd); } DIR *opendir(const char *name) { DIR *ret; if (smbw_path(name)) { return smbw_opendir(name); } return real_opendir(name); } struct dirent *readdir(DIR *dir) { if (smbw_dirp(dir)) { return smbw_readdir(dir); } return real_readdir(dir); } int closedir(DIR *dir) { if (smbw_dirp(dir)) { return smbw_closedir(dir); } return real_closedir(dir); } #ifndef NO_TELLDIR_WRAPPER off_t telldir(DIR *dir) { if (smbw_dirp(dir)) { return smbw_telldir(dir); } return real_telldir(dir); } #endif #ifndef NO_SEEKDIR_WRAPPER void seekdir(DIR *dir, off_t offset) { if (smbw_dirp(dir)) { smbw_seekdir(dir, offset); return; } real_seekdir(dir, offset); } #endif #ifndef NO_ACL_WRAPPER int acl(const char *pathp, int cmd, int nentries, aclent_t *aclbufp) { if (smbw_path(pathp)) { switch (cmd) { case GETACL: case GETACLCNT: return 0; default: errno = ENOSYS; return -1; } } real_acl(pathp, cmd, nentries, aclbufp); } #endif #ifndef NO_FACL_WRAPPER int facl(int fd, int cmd, int nentries, aclent_t *aclbufp) { if (smbw_fd(fd)) { switch (cmd) { case GETACL: case GETACLCNT: return 0; default: errno = ENOSYS; return -1; } } real_facl(fd, cmd, nentries, aclbufp); } #endif #ifndef LINUX int creat(const char *path, mode_t mode) { return open(path, O_WRONLY|O_CREAT|O_TRUNC, mode); } #endif #ifdef HAVE_STAT64 static void stat64_convert(struct stat *st, struct stat64 *st64) { st64->st_size = st->st_size; st64->st_mode = st->st_mode; st64->st_ino = st->st_ino; st64->st_dev = st->st_dev; st64->st_rdev = st->st_rdev; st64->st_nlink = st->st_nlink; st64->st_uid = st->st_uid; st64->st_gid = st->st_gid; st64->st_atime = st->st_atime; st64->st_mtime = st->st_mtime; st64->st_ctime = st->st_ctime; st64->st_blksize = st->st_blksize; st64->st_blocks = st->st_blocks; } int stat64(const char *name, struct stat64 *st64) { if (smbw_path(name)) { struct stat st; int ret = stat(name, &st); stat64_convert(&st, st64); return ret; } return real_stat64(name, st64); } int fstat64(int fd, struct stat64 *st64) { if (smbw_fd(fd)) { struct stat st; int ret = fstat(fd, &st); stat64_convert(&st, st64); return ret; } return real_fstat64(fd, st64); } int lstat64(const char *name, struct stat64 *st64) { if (smbw_path(name)) { struct stat st; int ret = lstat(name, &st); stat64_convert(&st, st64); return ret; } return real_lstat64(name, st64); } #endif #ifdef HAVE_LLSEEK offset_t llseek(int fd, offset_t ofs, int whence) { if (smbw_fd(fd)) { return lseek(fd, ofs, whence); } return real_llseek(fd, ofs, whence); } #endif #ifdef HAVE_READDIR64 static void dirent64_convert(struct dirent *d, struct dirent64 *d64) { d64->d_ino = d->d_ino; d64->d_off = d->d_off; d64->d_reclen = d->d_reclen; strcpy(d64->d_name, d->d_name); } struct dirent64 *readdir64(DIR *dir) { if (smbw_dirp(dir)) { struct dirent *d; static union { char buf[DIRP_SIZE]; struct dirent64 d64; } dbuf; d = readdir(dir); if (!d) return NULL; dirent64_convert(d, &dbuf.d64); return &dbuf.d64; } return real_readdir64(dir); } #endif