mirror of
https://git.libssh.org/projects/libssh.git
synced 2025-04-19 02:24:03 +03:00
includes also additional syscalls for 32b archs. Signed-off-by: Jakub Jelen <jjelen@redhat.com> Reviewed-by: Andreas Schneider <asn@cryptomilk.org> (cherry picked from commit d416ef533ff16628c7c196c21c6bea29aa0756fa)
214 lines
7.8 KiB
C
214 lines
7.8 KiB
C
#define _GNU_SOURCE
|
|
#include <dlfcn.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
|
|
/*******************************************************************************
|
|
* Structs
|
|
******************************************************************************/
|
|
struct file {
|
|
char *name;
|
|
uid_t uid;
|
|
gid_t gid;
|
|
} file = {0};
|
|
|
|
/*******************************************************************************
|
|
* Destructor
|
|
******************************************************************************/
|
|
|
|
void destructor(void) __attribute__((destructor));
|
|
|
|
void
|
|
destructor(void)
|
|
{
|
|
free(file.name);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Chown wrapping
|
|
******************************************************************************/
|
|
|
|
/** Records the UID and GID and pretend syscall worked */
|
|
static int
|
|
chown_helper(const char *pathname, uid_t owner, gid_t group)
|
|
{
|
|
if (strlen(pathname) > 7 && strncmp(pathname, "/dev/pt", 7) == 0) {
|
|
/*
|
|
* The OpenSSH server modified the PTY which requires root permissions
|
|
* see torture_request_pty_modes
|
|
* */
|
|
return 0;
|
|
}
|
|
if (strlen(pathname) > 4 && strncmp(pathname, "/tmp", 4) == 0) {
|
|
/*
|
|
* faking chown because It requires root permissions to modify the owner
|
|
* under /tmp
|
|
* It's also a helper for torture_sftp_setstat
|
|
* */
|
|
if (file.name != NULL) {
|
|
free((char *)file.name);
|
|
}
|
|
file.name = strdup(pathname);
|
|
file.uid = owner;
|
|
file.gid = group;
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
#define WRAP_CHOWN(syscall_name) \
|
|
typedef int (*__libc_##syscall_name)(const char *pathname, \
|
|
uid_t owner, \
|
|
gid_t group); \
|
|
int syscall_name(const char *pathname, uid_t owner, gid_t group); \
|
|
int syscall_name(const char *pathname, uid_t owner, gid_t group) \
|
|
{ \
|
|
__libc_##syscall_name original_##syscall_name = NULL; \
|
|
int rc; \
|
|
\
|
|
rc = chown_helper(pathname, owner, group); \
|
|
if (rc == 0) { \
|
|
return 0; \
|
|
} \
|
|
original_##syscall_name = \
|
|
(__libc_##syscall_name)dlsym(RTLD_NEXT, #syscall_name); \
|
|
return (*original_##syscall_name)(pathname, owner, group); \
|
|
}
|
|
|
|
WRAP_CHOWN(chown)
|
|
WRAP_CHOWN(chown32)
|
|
WRAP_CHOWN(lchown)
|
|
|
|
/* fchownat */
|
|
typedef int (*__libc_fchownat)(int dirfd,
|
|
const char *pathname,
|
|
uid_t owner,
|
|
gid_t group,
|
|
int flags);
|
|
|
|
int
|
|
fchownat(int dirfd, const char *pathname, uid_t owner, gid_t group, int flags);
|
|
|
|
int
|
|
fchownat(int dirfd, const char *pathname, uid_t owner, gid_t group, int flags)
|
|
{
|
|
__libc_fchownat original_fchownat = NULL;
|
|
int rc;
|
|
|
|
rc = chown_helper(pathname, owner, group);
|
|
if (rc == 0) {
|
|
return 0;
|
|
}
|
|
|
|
original_fchownat = (__libc_fchownat)dlsym(RTLD_NEXT, "fchownat");
|
|
return (*original_fchownat)(dirfd, pathname, owner, group, flags);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Stat wrapping
|
|
******************************************************************************/
|
|
|
|
/** Returns previously set UID/GID for the filename */
|
|
static void
|
|
stat_helper(const char *pathname, struct stat *statbuf)
|
|
{
|
|
if (file.name != NULL && strcmp(pathname, file.name) == 0) {
|
|
statbuf->st_uid = file.uid;
|
|
statbuf->st_gid = file.gid;
|
|
}
|
|
}
|
|
|
|
static void
|
|
stat64_helper(const char *pathname, struct stat64 *statbuf)
|
|
{
|
|
if (file.name != NULL && strcmp(pathname, file.name) == 0) {
|
|
statbuf->st_uid = file.uid;
|
|
statbuf->st_gid = file.gid;
|
|
}
|
|
}
|
|
|
|
#define WRAP_STAT(syscall_name, struct_name) \
|
|
typedef int (*__libc_##syscall_name)(const char *pathname, \
|
|
struct struct_name *statbuf); \
|
|
int syscall_name(const char *pathname, struct struct_name *statbuf); \
|
|
int syscall_name(const char *pathname, struct struct_name *statbuf) \
|
|
{ \
|
|
int rc; \
|
|
__libc_##syscall_name original_##syscall_name = NULL; \
|
|
\
|
|
original_##syscall_name = \
|
|
(__libc_##syscall_name)dlsym(RTLD_NEXT, #syscall_name); \
|
|
rc = (*original_##syscall_name)(pathname, statbuf); \
|
|
struct_name##_helper(pathname, statbuf); \
|
|
\
|
|
return rc; \
|
|
}
|
|
|
|
WRAP_STAT(stat, stat)
|
|
WRAP_STAT(lstat, stat)
|
|
/* i686 arch */
|
|
WRAP_STAT(stat64, stat64)
|
|
WRAP_STAT(lstat64, stat64)
|
|
|
|
#define WRAP_XSTAT(syscall_name) \
|
|
typedef int (*__libc_##syscall_name)(int ver, \
|
|
const char *pathname, \
|
|
struct stat *statbuf); \
|
|
int syscall_name(int ver, const char *pathname, struct stat *statbuf); \
|
|
int syscall_name(int ver, const char *pathname, struct stat *statbuf) \
|
|
{ \
|
|
int rc; \
|
|
__libc_##syscall_name original_##syscall_name = NULL; \
|
|
\
|
|
original_##syscall_name = \
|
|
(__libc_##syscall_name)dlsym(RTLD_NEXT, #syscall_name); \
|
|
rc = (*original_##syscall_name)(ver, pathname, statbuf); \
|
|
stat_helper(pathname, statbuf); \
|
|
\
|
|
return rc; \
|
|
}
|
|
|
|
WRAP_XSTAT(__xstat) /* CentOS8 */
|
|
WRAP_XSTAT(__lxstat)
|
|
|
|
/* i686 arch (likely not wrappable) */
|
|
static void
|
|
statx_helper(const char *pathname, struct statx *statbuf)
|
|
{
|
|
if (file.name != NULL && strcmp(pathname, file.name) == 0) {
|
|
statbuf->stx_uid = file.uid;
|
|
statbuf->stx_gid = file.gid;
|
|
}
|
|
}
|
|
|
|
typedef int (*__libc_statx)(int dirfd,
|
|
const char *pathname,
|
|
int flags,
|
|
unsigned int mask,
|
|
struct statx *statbuf);
|
|
int statx(int dirfd,
|
|
const char *pathname,
|
|
int flags,
|
|
unsigned int mask,
|
|
struct statx *statbuf);
|
|
int
|
|
statx(int dirfd,
|
|
const char *pathname,
|
|
int flags,
|
|
unsigned int mask,
|
|
struct statx *statbuf)
|
|
{
|
|
int rc;
|
|
__libc_statx original_statx = NULL;
|
|
|
|
original_statx = (__libc_statx)dlsym(RTLD_NEXT, "statx");
|
|
rc = (*original_statx)(dirfd, pathname, flags, mask, statbuf);
|
|
statx_helper(pathname, statbuf);
|
|
|
|
return rc;
|
|
}
|